當(dāng)前位置:首頁 > 芯聞號 > 充電吧
[導(dǎo)讀]最近在學(xué) Dbus,不過總是不得其門而入。大部分資料都講了很多東西卻最終沒有讓我搞清楚怎么用 DBus,不就是一個 IPC 通信的工具么?就沒有一點實用些的資料么?看了很多資料之后還是覺得只見樹木不見


最近在學(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)行具體的探究了。

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉