| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359 | program busexample;{$ifdef fpc}  {$mode objfpc}{$H+}{$endif}uses  {$IFDEF UNIX}{$IFDEF UseCThreads}  cthreads,  {$ENDIF}{$ENDIF}  SysUtils,  ctypes,  dbus;const  SINTAX_TEXT = 'Syntax: dbus-example [send|receive|listen|query] [<param>]';var  err: DBusError;  conn: PDBusConnection;  ret: cint;{ * Send a broadcast signal }procedure BusSend(sigvalue: PChar);var  msg: PDBusMessage;  args: DBusMessageIter;  serial: dbus_uint32_t = 0;begin  WriteLn('Sending signal with value ', string(sigvalue));  { Request the name of the bus }  ret := dbus_bus_request_name(conn, 'test.signal.source', DBUS_NAME_FLAG_REPLACE_EXISTING, @err);  if dbus_error_is_set(@err) <> 0 then  begin    WriteLn('Name Error: ' + err.message);    dbus_error_free(@err);  end;  if ret <> DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER then Exit;  // create a signal & check for errors  msg := dbus_message_new_signal('/test/signal/Object', // object name of the signal                                 'test.signal.Type', // interface name of the signal                                 'Test'); // name of the signal  if (msg = nil) then  begin    WriteLn('Message Null');    Exit;  end;  // append arguments onto signal  dbus_message_iter_init_append(msg, @args);  if (dbus_message_iter_append_basic(@args, DBUS_TYPE_STRING, @sigvalue) = 0) then  begin    WriteLn('Out Of Memory!');    Exit;  end;  // send the message and flush the connection  if (dbus_connection_send(conn, msg, @serial) = 0) then  begin    WriteLn('Out Of Memory!');    Exit;  end;    dbus_connection_flush(conn);  WriteLn('Signal Sent');  // free the message and close the connection  dbus_message_unref(msg);end;{ * Listens for signals on the bus }procedure BusReceive;var  msg: PDBusMessage;  args: DBusMessageIter;  sigvalue: PChar;begin  WriteLn('Listening for signals');  { Request the name of the bus }  ret := dbus_bus_request_name(conn, 'test.signal.sink', DBUS_NAME_FLAG_REPLACE_EXISTING, @err);  if dbus_error_is_set(@err) <> 0 then  begin    WriteLn('Name Error: ' + err.message);    dbus_error_free(@err);  end;  // add a rule for which messages we want to see  dbus_bus_add_match(conn, 'type=''signal'',interface=''test.signal.Type''', @err); // see signals from the given interface  dbus_connection_flush(conn);  if (dbus_error_is_set(@err) <> 0) then  begin    WriteLn('Match Error (', err.message, ')');    Exit;  end;  WriteLn('Match rule sent');  // loop listening for signals being emmitted  while (true) do  begin    // non blocking read of the next available message    dbus_connection_read_write(conn, 0);    msg := dbus_connection_pop_message(conn);    // loop again if we haven't read a message    if (msg = nil) then    begin      sleep(1);      Continue;    end;    // check if the message is a signal from the correct interface and with the correct name    if (dbus_message_is_signal(msg, 'test.signal.Type', 'Test') <> 0) then    begin      // read the parameters      if (dbus_message_iter_init(msg, @args) = 0) then         WriteLn('Message Has No Parameters')      else if (DBUS_TYPE_STRING <> dbus_message_iter_get_arg_type(@args)) then         WriteLn('Argument is not string!')      else         dbus_message_iter_get_basic(@args, @sigvalue);      WriteLn('Got Signal with value ', sigvalue);    end;    // free the message    dbus_message_unref(msg);  end;end;procedure reply_to_method_call(msg: PDBusMessage; conn: PDBusConnection);var  reply: PDBusMessage;  args: DBusMessageIter;  stat: Boolean = true;  level: dbus_uint32_t = 21614;  serial: dbus_uint32_t = 0;  param: PChar = '';begin   // read the arguments   if (dbus_message_iter_init(msg, @args) = 0) then      WriteLn('Message has no arguments!')   else if (DBUS_TYPE_STRING <> dbus_message_iter_get_arg_type(@args)) then      WriteLn('Argument is not string!')   else      dbus_message_iter_get_basic(@args, @param);   WriteLn('Method called with ', param);   // create a reply from the message   reply := dbus_message_new_method_return(msg);   // add the arguments to the reply   dbus_message_iter_init_append(reply, @args);   if (dbus_message_iter_append_basic(@args, DBUS_TYPE_BOOLEAN, @stat) = 0) then   begin     WriteLn('Out Of Memory!');     Exit;   end;   if (dbus_message_iter_append_basic(@args, DBUS_TYPE_UINT32, @level) = 0) then   begin     WriteLn('Out Of Memory!');     Exit;   end;   // send the reply && flush the connection   if (dbus_connection_send(conn, reply, @serial) = 0) then   begin     WriteLn('Out Of Memory!');     Exit;   end;   dbus_connection_flush(conn);   // free the reply   dbus_message_unref(reply);end;{ * Server that exposes a method call and waits for it to be called }procedure BusListen;var  msg, reply: PDBusMessage;  args: DBusMessageIter;  param: PChar;begin  WriteLn('Listening for method calls');  { Request the name of the bus }  ret := dbus_bus_request_name(conn, 'test.method.server', DBUS_NAME_FLAG_REPLACE_EXISTING, @err);  if dbus_error_is_set(@err) <> 0 then  begin    WriteLn('Name Error: ' + err.message);    dbus_error_free(@err);  end;  if ret <> DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER then Exit;  // loop, testing for new messages  while (true) do  begin    // non blocking read of the next available message    dbus_connection_read_write(conn, 0);    msg := dbus_connection_pop_message(conn);    // loop again if we haven't got a message    if (msg = nil) then    begin      sleep(1);      Continue;    end;    // check this is a method call for the right interface & method    if (dbus_message_is_method_call(msg, 'test.method.Type', 'Method') <> 0) then       reply_to_method_call(msg, conn);    // free the message    dbus_message_unref(msg);  end;end;{ * Call a method on a remote object }procedure BusCall(param: PChar);var  msg: PDBusMessage;  args: DBusMessageIter;  pending: PDBusPendingCall;  stat: Boolean;  level: dbus_uint32_t;begin  WriteLn('Calling remote method with ', param);  { Request the name of the bus }  ret := dbus_bus_request_name(conn, 'test.method.caller', DBUS_NAME_FLAG_REPLACE_EXISTING, @err);  if dbus_error_is_set(@err) <> 0 then  begin    WriteLn('Name Error: ' + err.message);    dbus_error_free(@err);  end;  if ret <> DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER then Exit;  // create a new method call and check for errors  msg := dbus_message_new_method_call('test.method.server', // target for the method call                                      '/test/method/Object', // object to call on                                      'test.method.Type', // interface to call on                                      'Method'); // method name  if (msg = nil) then  begin    WriteLn('Message Null');    Exit;  end;  // append arguments  dbus_message_iter_init_append(msg, @args);  if (dbus_message_iter_append_basic(@args, DBUS_TYPE_STRING, @param) = 0) then  begin    WriteLn('Out Of Memory!');    Exit;  end;  // send message and get a handle for a reply  if (dbus_connection_send_with_reply(conn, msg, @pending, -1) = 0) then // -1 is default timeout  begin    WriteLn('Out Of Memory!');    Exit;  end;  if (pending = nil) then  begin    WriteLn('Pending Call Null');    Exit;  end;  dbus_connection_flush(conn);  WriteLn('Request Sent');  // free message  dbus_message_unref(msg);  // block until we recieve a reply  dbus_pending_call_block(pending);  // get the reply message  msg := dbus_pending_call_steal_reply(pending);  if (msg = nil) then  begin    WriteLn('Reply Null');    Exit;  end;  // free the pending message handle  dbus_pending_call_unref(pending);  // read the parameters  if (dbus_message_iter_init(msg, @args) = 0) then     WriteLn('Message has no arguments!')  else if (DBUS_TYPE_BOOLEAN <> dbus_message_iter_get_arg_type(@args)) then     WriteLn('Argument is not boolean!')  else     dbus_message_iter_get_basic(@args, @stat);  if (dbus_message_iter_next(@args) = 0) then     WriteLn('Message has too few arguments!')  else if (DBUS_TYPE_UINT32 <> dbus_message_iter_get_arg_type(@args)) then     WriteLn('Argument is not int!')  else     dbus_message_iter_get_basic(@args, @level);  WriteLn('Got Reply: ', stat, ', ', level);  // free reply  dbus_message_unref(msg);end;begin  { Initializes the errors }  dbus_error_init(@err);    { Connection }  conn := dbus_bus_get(DBUS_BUS_SESSION, @err);  if dbus_error_is_set(@err) <> 0 then  begin    WriteLn('Connection Error: ' + err.message);    dbus_error_free(@err);  end;    if conn = nil then Exit;    { Parses parameters }    if (ParamCount <> 1) and (ParamCount <> 2) then WriteLn(SINTAX_TEXT)  else  begin    if ParamStr(1) = 'send' then BusSend(PChar(ParamStr(2)))    else if ParamStr(1) = 'receive' then BusReceive()    else if ParamStr(1) = 'listen' then BusListen()    else if ParamStr(1) = 'call' then BusCall(PChar(ParamStr(2)))    else WriteLn(SINTAX_TEXT);  end;  { Finalization }  dbus_connection_close(conn);end.
 |