最近在學(xué) Dbus,不過總是不得其門而入。
大部分資料都講了很多東西卻最終沒有讓我搞清楚怎么用 DBus,不就是一個 IPC 通信的工具么?就沒有一點實用些的資料么?看了很多資料之后還是覺得只見樹木不見森林。仔細(xì)整理下思路,覺得還是應(yīng)該從最基本的方面入門,先從 DBus 的 C API 入手學(xué)習(xí),有了這些知識,就算麻煩,也可以先在完成一個基本功能的例子程序的同時大概的知道 DBus 的運行機制。
在網(wǎng)上找到這么一篇文章:http://www.matthew.ath.cx/misc/dbus, 正合我意,下面的內(nèi)容基本是對這篇文章的翻譯和擴(kuò)充。
注意:
翻譯沒有得到原文作者同意,原文也很簡單易懂,最好去讀原文。如果收到投訴,我會立即撤掉本文的。本文不是一篇好的 DBus 入門,有很多基本的東西不在記述之內(nèi)。一般情況下不會直接使用 C API 進(jìn)行 DBus 的編程,而是使用某種 DBus-binding,但我覺得理解 DBus 的 C API 對完整地理解 DBus 是非常重要的。雖然 DBus 是用 C 寫的,而且本文寫的是 C API,但是 DBus 設(shè)計中充滿的面向?qū)ο蟮乃枷?,請注意?/p>
一、共通部分的代碼
在使用 DBus 進(jìn)行通信的時候,有一些代碼是無論如何都會使用到的。首先,你必須要連接上 Dbus,一般來說,系統(tǒng)中會有一個 System Bus 和一個 Session Bus(他們的差別,請參考我另外的筆記)。其次,你需要在 Dbus 中注冊一個名字,用于標(biāo)識自己。為了簡單起見,這里先不考慮重名的情況:
DBusError?err; DBusConnection*?conn; int?ret; //?initialise?the?errors dbus_error_init(&err); ? //?connect?to?the?bus conn?=?dbus_bus_get(DBUS_BUS_SESSION,?&err); if?(dbus_error_is_set(&err))?{ ????fprintf(stderr,?"Connection?Error?(%s)n",?err.message); ????dbus_error_free(&err); } if?(NULL?==?conn)?{ ????exit(1); } //?request?a?name?on?the?bus ret?=?dbus_bus_request_name(conn,?"test.method.server", ????????????????????????????DBUS_NAME_FLAG_REPLACE_EXISTING ????????????????????????????,?&err); if?(dbus_error_is_set(&err))?{ ????fprintf(stderr,?"Name?Error?(%s)n",?err.message); ????dbus_error_free(&err); } if?(DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER?!=?ret)?{ ????exit(1); }
一般來說,連接上 Dbus 和注冊一個名稱,應(yīng)該是在程序最開始運行的時候就會進(jìn)行的操作。
當(dāng)然,在程序的結(jié)束的時候,需要關(guān)閉掉與 Dbus 的連接。使用下面的函數(shù):
dbus_connection_close(conn);
二、發(fā)送信號(Sending Signal)
信號是一種廣播的消息,你可以簡單的發(fā)出一個信號,這樣,所有連接在 DBus 總線上并注冊了接受對應(yīng)信號的進(jìn)程,都會收到這個信號。為了發(fā)出一個信號,需要的只是創(chuàng)建一個 DBusMessage 對象來代表信號,然后追加上一些需要發(fā)出的參數(shù),就可以發(fā)向總線了。發(fā)完之后還需要釋放掉 Message。如果內(nèi)存不足的話,這下面不少函數(shù)都會返回 false,所以一般情況下你都需要處理這些情況的返回值。
dbus_uint32_t?serial?=?0;?//?unique?number?to?associate?replies?with?requests DBusMessage*?msg; DBusMessageIter?args; ? //?create?a?signal?and?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?(NULL?==?msg) { ????fprintf(stderr,?"Message?Nulln"); ????exit(1); } ? //?append?arguments?onto?signal dbus_message_iter_init_append(msg,?&args); if?(!dbus_message_iter_append_basic(&args,?DBUS_TYPE_STRING,?&sigvalue))?{ ????fprintf(stderr,?"Out?Of?Memory!n"); ????exit(1); } ? //?send?the?message?and?flush?the?connection if?(!dbus_connection_send(conn,?msg,?&serial))?{ ????fprintf(stderr,?"Out?Of?Memory!n"); ????exit(1); } dbus_connection_flush(conn); ? //?free?the?message dbus_message_unref(msg);
三、調(diào)用方法(Calling a Method)
調(diào)用一個遠(yuǎn)程方法(remote method)與發(fā)送一個信號(sending a signal)是很類似的。需要創(chuàng)建一個 DBusMessage,然后通過注冊在 DBus 上的名稱指定發(fā)送的對象。然后追加相應(yīng)的參數(shù),但調(diào)用方法分為兩種,一種是阻塞式的,另一種則可以異步調(diào)用。異步調(diào)用的時候會得到一個 DBusMessage* 的返回,從這個 DBusMessage 中可以獲取一些返回的參數(shù)。
調(diào)用方法一:
DBusMessage*?msg; DBusMessageIter?args; DBusPendingCall*?pending; ? 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?(NULL?==?msg)?{ ????fprintf(stderr,?"Message?Nulln"); ????exit(1); } ? //?append?arguments dbus_message_iter_init_append(msg,?&args); if?(!dbus_message_iter_append_basic(&args,?DBUS_TYPE_STRING,??m))?{ ????fprintf(stderr,?"Out?Of?Memory!n"); ????exit(1); } ? //?send?message?and?get?a?handle?for?a?reply if?(!dbus_connection_send_with_reply?(conn,?msg,?&pending,?-1))?{?//?-1?is?default?timeout ????fprintf(stderr,?"Out?Of?Memory!n"); ????exit(1); } if?(NULL?==?pending)?{ ????fprintf(stderr,?"Pending?Call?Nulln"); ????exit(1); } dbus_connection_flush(conn); ? //?free?message dbus_message_unref(msg);
調(diào)用方法二:
bool?stat; dbus_uint32_t?level; ? //?block?until?we?receive?a?reply dbus_pending_call_block(pending); ? //?get?the?reply?message msg?=?dbus_pending_call_steal_reply(pending); if?(NULL?==?msg)?{ ????fprintf(stderr,?"Reply?Nulln"); ????exit(1); } //?free?the?pending?message?handle dbus_pending_call_unref(pending); ? //?read?the?parameters if?(!dbus_message_iter_init(msg,?&args)) ????fprintf(stderr,?"Message?has?no?arguments!n"); else?if?(DBUS_TYPE_BOOLEAN?!=?dbus_message_iter_get_arg_type(&args)) ????fprintf(stderr,?"Argument?is?not?boolean!n"); else ????dbus_message_iter_get_basic(&args,?&stat); ? if?(!dbus_message_iter_next(&args)) ????fprintf(stderr,?"Message?has?too?few?arguments!n"); else?if?(DBUS_TYPE_UINT32?!=?dbus_message_iter_get_arg_type(&args)) ????fprintf(stderr,?"Argument?is?not?int!n"); else ????dbus_message_iter_get_basic(&args,?&level); ? printf("Got?Reply:?%d,?%dn",?stat,?level); ? //?free?reply?and?close?connection dbus_message_unref(msg);
四、接收消息(Receiving a Signal)
接下來的兩種操作主要是從總線從讀取消息并處理這些消息。
要接收一個消息,你首先需要告訴 DBus 你對什么樣的消息感興趣:
//?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))?{ ????fprintf(stderr,?"Match?Error?(%s)n",?err.message); ????exit(1); }
然后,進(jìn)程就可以在一個循環(huán)中等待這類消息的發(fā)生了:
/?loop?listening?for?signals?being?emmitted while?(true)?{ ? ????//?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?(NULL?==?msg)?{ ????????sleep(1); ????????continue; ????} ? ????//?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"))?{ ????????//?read?the?parameters ????????if?(!dbus_message_iter_init(msg,?&args)) ????????????fprintf(stderr,?"Message?has?no?arguments!n"); ????????else?if?(DBUS_TYPE_STRING?!=?dbus_message_iter_get_arg_type(&args)) ????????????fprintf(stderr,?"Argument?is?not?string!n"); ????????else?{ ????????????dbus_message_iter_get_basic(&args,?&sigvalue); ????????????printf("Got?Signal?with?value?%sn",?sigvalue); ????????} ????} ? ????//?free?the?message ????dbus_message_unref(msg); }
五、提供被遠(yuǎn)程調(diào)用的方法(Exposing a Method to be called)
在第二節(jié)中,我們看到了調(diào)用一個遠(yuǎn)程方法,這節(jié)就是告訴我們怎么樣提供一個方法讓別的應(yīng)用程序調(diào)用。用下面的程序,就可以把方法關(guān)聯(lián)在那些提供給外部的方法上,并解析出相應(yīng)的參數(shù),最后構(gòu)建一個消息返回給調(diào)用方法的應(yīng)用程序。
提供遠(yuǎn)程調(diào)用方法1:
//?loop,?testing?for?new?messages while?(true)?{ ????//?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?(NULL?==?msg)?{ ????????sleep(1); ????????continue; ????} ? ????//?check?this?is?a?method?call?for?the?right?interface?and?method ????if?(dbus_message_is_method_call(msg,?"test.method.Type",?"Method")) ????????reply_to_method_call(msg,?conn); ? ????//?free?the?message ????dbus_message_unref(msg); }
void?reply_to_method_call(DBusMessage*?msg,?DBusConnection*?conn)
{
????DBusMessage*?reply; ????DBusMessageIter?args; ????DBusConnection*?conn; ????bool?stat?=?true; ????dbus_uint32_t?level?=?21614; ????dbus_uint32_t?serial?=?0; ????char*?param?=?""; ? ????//?read?the?arguments ????if?(!dbus_message_iter_init(msg,?&args)) ????????fprintf(stderr,?"Message?has?no?arguments!n"); ????else?if?(DBUS_TYPE_STRING?!=?dbus_message_iter_get_arg_type(&args)) ????????fprintf(stderr,?"Argument?is?not?string!n"); ????else ????????dbus_message_iter_get_basic(&args,??m); ????printf("Method?called?with?%sn",?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))?{ ????????fprintf(stderr,?"Out?Of?Memory!n"); ????????exit(1); ????} ????if?(!dbus_message_iter_append_basic(&args,?DBUS_TYPE_UINT32,?&level))?{ ????????fprintf(stderr,?"Out?Of?Memory!n"); ????????exit(1); ????} ? ????//?send?the?reply?&&?flush?the?connection ????if?(!dbus_connection_send(conn,?reply,?&serial))?{ ????????fprintf(stderr,?"Out?Of?Memory!n"); ????????exit(1); ????} ????dbus_connection_flush(conn); ? ????//?free?the?reply ????dbus_message_unref(reply); }
這就基本上全部了。但用這些來理解 DBus 顯然還遠(yuǎn)遠(yuǎn)不夠。接下來,就要對這些程序以及背后的理念進(jìn)行具體的探究了。