2012-03-24
本文介紹了AndroidRIL的總體架構(gòu),具體分析了本地庫的實現(xiàn)[c/cpp部分]
一、總體架構(gòu)
Android RIL (Radio Interface Layer)提供了Telephony服務(wù)和Radio硬件之間的抽象層。RIL負(fù)責(zé)數(shù)據(jù)的可靠傳輸、AT命令的發(fā)送以及response的解析。一般的,應(yīng)用處理器(AP)通過AT命令集與無線通訊模塊(基帶/BP)通信。把標(biāo)準(zhǔn)的GSM27.007中常用的如Dial這些主動請求的操作稱之為 request ;另一類GSM主動上報的例如信號強(qiáng)度、基站信息、來短信等,稱之為 unsolicited response 。
二、實現(xiàn)文件分布
SMS/MMS, call, …
Telephony的應(yīng)用層實現(xiàn)在具體的apps中。Call在packages/apps/Phone中;SMS/MMS在packages/apps/Mms中;網(wǎng)絡(luò)選擇等在packages/apps/Settings中。
android.telephony.*
Telephony的JavaFramework代碼,供上層使用
com.android.internal.telephony.*
Telephony的JavaFramework代碼的內(nèi)部實現(xiàn),該包是隱藏的,外面無法訪問。
hardware/ril/include/telephony/ril.h
本地代碼的頭文件
hardware/ril/libril
ril本地庫的實現(xiàn)源代碼,生成libril.so
hardware/ril/rild
ril守護(hù)進(jìn)程rild的實現(xiàn)源代碼,生成可執(zhí)行文件rild
hardware/ril/reference-ril
ril實現(xiàn)庫的參考實現(xiàn)源代碼,生成libreference-ril.so
針對不同的硬件平臺可以仿照這個來實現(xiàn)具體的功能庫。
三、ril初始化
1.Rild解析ril的實現(xiàn)庫<rillibPath>——可以通過命令行或property來指定。
Rild在init.rc中可以通過命令行參數(shù) -l <rillibPath> 指定ril的具體實現(xiàn)庫;
如果命令行中未指定實現(xiàn)庫,通過property_get(“ rild.libpath ”, …)獲取實現(xiàn)庫。
而,在模擬環(huán)境下的話,就用/system/lib/libreference-ril.so,這也是上節(jié)所提到的libreference-ril.so最終被放的路徑。
2.通過dlopen(<rillibPath>, )加載ril實現(xiàn)庫;
3.調(diào)用libril中的RIL_startEventLoop()開啟并確保 eventLoop線程 已經(jīng)啟動;
4.通過dlsym()獲取<rillibPath>中定義的RIL_Init()函數(shù)。
5.獲取RIL_Init()所需的參數(shù)——可以通過命令行或property來指定。
Rild在init.rc中可以通過命令行參數(shù) -- 指定RIL_Init的參數(shù);
如果命令行中未指定實現(xiàn)庫,通過property_get(“ rild.libargs ”, …)獲取參數(shù)。
6.由4&5獲取的函數(shù)以及參數(shù),執(zhí)行 RIL_Init ()開始 mainLoop線程 ,并獲取RIL_RadioFunctions;
7.以6獲取的RIL_RadioFunctions為參數(shù),執(zhí)行l(wèi)ibril中的 RIL_Register ()保留這些callback函數(shù)到s_callbacks,開啟命名Socket“rild”,接受上層的Socket指令。
時序圖如下圖所示:
注意:圖中的序號跟上面描述的步驟并不是一一對應(yīng)的。
在上面的時序中,有幾個重要的數(shù)據(jù): s_rilEnv : RIL_Env定義在rild中,但函數(shù)的具體實現(xiàn)是在標(biāo)準(zhǔn)庫libril.so中,注冊給實現(xiàn)庫,供RIL實現(xiàn)庫調(diào)用; s_callbacks : RIL_RadioFunctions定義在reference-ril中,實現(xiàn)Request等操作。
四、Request過程
初始化過程步驟3啟動的eventLoop會調(diào)用ril_event_loop()處理Request請求。
通過select()多路選擇Socket,偵聽是否有Java層到來的Socket請求。執(zhí)行:
ril_event_loop() ->
-> processTimeouts()
-> processReadReadies()把ril_event從watch_table[]移到pendling_list
-> firePending() -> ev->func() [listenCallback()[ril.cpp]]
-> record_stream_new()新創(chuàng)建一個RecordStream
-> 收到完整的Request后,執(zhí)行processCommandsCallback()
-> processCommandBuffer () [ril.cpp]
具體的執(zhí)行都在processCommandBuffer()
Ril.cpp中定義了s_comamnds:CommandInfo[]
processCommandBuffer()中,通過request的索引,找到s_commands中的CommandInfo,然后通過CommandInfo的dispatchFunction,具體執(zhí)行dispatch操作。
可以打開ril_commands.h中看具體某個Request的dispatch函數(shù)。
dispatchXYZ()是一系列對特定參數(shù)的Reqeust的封裝,看dispatchString(Parcel&p, RequestInfo *pRI)的實現(xiàn):
解析出參數(shù),通過s_callbacks.onRequest()調(diào)用reference-ril具體實現(xiàn)庫中的onRequest的實現(xiàn)。[s_callbacks是在第三節(jié)中初始化時注冊保留的]
Reference-ril中的onRequest()有一個很大的switch… case語句來處理各種request。
requestSendSMS()中調(diào)用at_send_command_sms()用“+CMGS”發(fā)送短信,并獲取返回值。
requestSendSMS()中調(diào)用 RIL_onRequestComplete (t,e: RIL_Errno)完成Request。
五、Response過程
Response有Solicited Response也就是上節(jié)講的Request的響應(yīng),另外還有一種就是主動上報的unsolicitedresponse,比如來電話,來短信等信息。
5.1 solicited response
對于Solicited response,上節(jié)中最后調(diào)用的是RIL_onRequestComplete(),reference-ril中該函數(shù)調(diào)用s_rilenv->OnRequestComplete()。從第三節(jié)的初始化知道,具體的實現(xiàn)是在ril.cpp中RIL_onRequestComplete()里。
在RIL_onRequestComplete()中,
如果“ril_commsnds.h”中定義的該CommandInfo中有response函數(shù),完成對responseFunction()的調(diào)用;
轉(zhuǎn)換ril定義的響應(yīng)信息結(jié)構(gòu),調(diào)用sendResponse(),將響應(yīng)結(jié)果通過Socket反饋到上層。
5.2 unsolicited response
第三節(jié)步驟六開始的mainLoop中會通過at_open()把onUnsolicited()傳遞給AtChannel。AtChannel的at_open()中開啟readerLoop線程。
readerLoop()中:
readLine() -> processLine()對Unsolicited message調(diào)用handleUnsolicited()作出處理;而handleUnsolicited()中回調(diào)reference-ril中注冊的onUnsolicited()函數(shù)。
reference-ril的onUnsolicited()中對各種特殊指令作出處理之后,調(diào)用初始化時[第三節(jié)]注冊的RIL_Env中的RIL_onUnsolicitedResponse(),最終實現(xiàn)是在ril.cpp的RIL_onUnsolicitedResponse()。
同Request一樣,Ril.cpp中定義了s_comamnds:CommandInfo[]
RIL_onUnsolicitedResponse中,通過unsolResponseIndex,找到s_unsolResponses中的UnsolResponseInfo,然后通過UnsolResponseInfo的WakeType決定喚醒手機(jī)的WakeLock,再通過responseFunction,具體執(zhí)行response操作。
可以打開ril_unsol_commands.h中看具體某個Unsolicitedresponse的定義。
最后,unsolicited response與solicited response一樣,也是通過調(diào)用sendResponse(),通過Socket反饋到Java上層。
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯(lián)系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機(jī)微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
