上一篇咱們分析了android HAL層的主要的兩個結構體hw_module_t(硬件模塊)和hw_device_t(硬件設備)的成員,下面咱們來具體看看上層app究竟是怎麼實現操做硬件的?android
咱們知道,一些硬件廠商不肯意將本身的一些核心代碼開放出去,因此將這些代碼放到HAL層,可是怎麼保證它不開放呢?HAL層代碼不是也讓你們知道下載嗎?其實硬件廠商的HAL核心代碼是以共享庫的形式出現的,每次在須要的時候,hal會自動加載調用相關共享庫。那麼是怎麼加載找到某一硬件設備對應的共享庫的呢?這也是咱們這篇都要說的。數組
上層app經過jni調用hal層的hw_get_module函數獲取硬件模塊,這個函數是上層與hal打交道的入口。因此若是咱們以程序調用執行的流程去看源碼的話,這個函數就是hal層第一個被調用的函數,下面咱們就app
從這個函數開始,沿着程序執行的流程走下去。函數
hw_get_module函數定義在/hardware/libhardware/hardware.c中,打開這個文件能夠看到定義以下:oop
1 int hw_get_module(const char *id, const struct hw_module_t **module) 2 { 3 int status; 4 int i; 5 const struct hw_module_t *hmi = NULL; 6 char prop[PATH_MAX]; 7 char path[PATH_MAX]; 8 9 /* 10 * Here we rely on the fact that calling dlopen multiple times on 11 * the same .so will simply increment a refcount (and not load 12 * a new copy of the library). 13 * We also assume that dlopen() is thread-safe. 14 */ 15 16 /* Loop through the configuration variants looking for a module */ 17 for (i=0 ; i<HAL_VARIANT_KEYS_COUNT+1 ; i++) { 18 if (i < HAL_VARIANT_KEYS_COUNT) { 19 if (property_get(variant_keys[i], prop, NULL) == 0) {//獲取屬性 20 continue; 21 } 22 snprintf(path, sizeof(path), "%s/%s.%s.so", 23 HAL_LIBRARY_PATH1, id, prop); 24 if (access(path, R_OK) == 0) break;//檢查system路徑是否有庫文件 25 26 snprintf(path, sizeof(path), "%s/%s.%s.so", 27 HAL_LIBRARY_PATH2, id, prop); 28 if (access(path, R_OK) == 0) break;//檢查vender路徑是否有庫文件 29 } else { 30 snprintf(path, sizeof(path), "%s/%s.default.so",//若是都沒有,則使用缺省的 31 HAL_LIBRARY_PATH1, id); 32 if (access(path, R_OK) == 0) break; 33 } 34 } 35 36 status = -ENOENT; 37 if (i < HAL_VARIANT_KEYS_COUNT+1) { 38 /* load the module, if this fails, we're doomed, and we should not try 39 * to load a different variant. */ 40 status = load(id, path, module);//裝載庫,獲得module 41 } 42 43 return status; 44 }
看第一行咱們知道有兩個參數,第一參數id就是要獲取的硬件模塊的id,第二個參數module就是咱們想獲得的硬件模塊結構體的指針。this
因此能夠看出,上層首先給hal須要獲取的硬件模塊的id,hw_get_module函數根據這個id去查找匹配和這個id對應的硬件模塊結構體的。spa
下面看看怎麼找的。指針
17行有個for循環,上限是HAL_VARIANT_KEYS_COUNT+1,那麼這個HAL_VARIANT_KEYS_COUNT是什麼呢?查看同文件下找到有:orm
static const int HAL_VARIANT_KEYS_COUNT =
(sizeof(variant_keys)/sizeof(variant_keys[0]));ip
原來它是ariant_keys這個數組的元素個數。那麼這個數組又是什麼呢?在本文件找,有:
/** * There are a set of variant filename for modules. The form of the filename * is "<MODULE_ID>.variant.so" so for the led module the Dream variants * of base "ro.product.board", "ro.board.platform" and "ro.arch" would be: * * led.trout.so * led.msm7k.so * led.ARMV6.so * led.default.so */ static const char *variant_keys[] = { "ro.hardware", /* This goes first so that it can pick up a different file on the emulator. */ "ro.product.board", "ro.board.platform", "ro.arch" };
能夠看到它實際上是個字符串數組。站且不知道幹什麼的。繼續看hw_get_module函數,進入for循環裏面,看22行,其實它是將HAL_LIBRARY_PATH1, id, prop這三個串拼湊一個路徑出來,
HAL_LIBRARY_PATH1定義以下:
/** Base path of the hal modules */
#define HAL_LIBRARY_PATH1 "/system/lib/hw"
#define HAL_LIBRARY_PATH2 "/vendor/lib/hw"
id是上層提供的,prop這個變量的值是前面19行property_get(variant_keys[i], prop, NULL)函數獲取到的,其實這個函數是經過ariant_keys數組的的屬性查找到系統中對應的變種名稱。不一樣的平臺獲取到prop值是不同的。
假如在獲取到的prop值是tout,須要獲取的硬件模塊的id是leds,那麼最後path組成的串是/system/lib/hw/leds.tout.so。
後面24行access是檢查這個路徑下是否存在,若是有就break,跳出循環。若是沒有,繼續走下面,
能夠看到下面幾行和剛纔形式差很少,
snprintf(path, sizeof(path), "%s/%s.%s.so", HAL_LIBRARY_PATH2, id, prop);
if (access(path, R_OK) == 0) break;//檢查vender路徑是否有庫文件
結合 HAL_LIBRARY_PATH2 爲"/vendor/lib/hw",假設一樣獲取到的prop值是tout,須要獲取的硬件模塊的id是leds,這種狀況下path拼出來的值是/vender/lib/hw/leds.tout.so,而後在判斷文件是否存在。若是存在跳出循環。
從以上分析,其實這就是hal層搜索動態共享庫的方式,從中咱們能夠獲得兩點:
1.動態共享庫通常放在 "/system/lib/hw"和"/vendor/lib/hw"這兩個路徑下。
2.動態庫的名稱是以"id.variant.so"的形式命名的,其中id爲上層提供,中間variant爲變種名稱,是隨系統平臺變化的。
接着,從29到32行咱們能夠看到,當全部變種名稱形式的包都不存在時,就以"id.default.so"形式包名查找是否存在。
37行, if (i < HAL_VARIANT_KEYS_COUNT+1),若是i小於變種名稱數組的話,表示找到了對應的庫,那麼38行load(id, path, module);//裝載庫,獲得module。
以上就對hal層搜索庫的規則搞清楚了。
下一篇咱們將進入load函數,看看共享庫是如何被加載的。