騷操作:利用強(qiáng)弱符號(hào)制作插件庫
本文轉(zhuǎn)載自微信公眾號(hào)「編程珠璣」,作者守望先生。轉(zhuǎn)載本文請聯(lián)系編程珠璣(ID:shouwangxiansheng)公眾號(hào)。
在《什么是強(qiáng)符號(hào)和弱符號(hào)》中簡單介紹了強(qiáng)弱符號(hào),那么強(qiáng)弱符號(hào)的性質(zhì)有什么用呢?
還記得在《什么是強(qiáng)符號(hào)和弱符號(hào)》中提到的鏈接原則嗎?
- 當(dāng)有強(qiáng)符號(hào)和弱符號(hào)時(shí),選擇使用強(qiáng)符號(hào)
那么我們正可以利用這個(gè)原則做以下事情:
- 定義為弱符號(hào),如果是弱符號(hào),使用默認(rèn)行為
- 如果鏈接了庫,是強(qiáng)符號(hào),則使用外部定義行為
以此來實(shí)現(xiàn)一個(gè)類似插件的功能。通俗一點(diǎn)說:
- 當(dāng)沒有插件時(shí),使用默認(rèn)行為
- 鏈接了插件時(shí),使用插件的功能
原理和示例
其原理也非常簡單:
- 外部引用弱符號(hào)
- 如果符號(hào)地址為0,則說明外部沒有鏈接插件庫,未有強(qiáng)符號(hào),走默認(rèn)流程
- 如果符號(hào)地址不為0,則說明鏈接了插件庫,執(zhí)行插件庫的功能。
示例程序如下:
- // 來源:公眾號(hào)【編程珠璣】
- // 作者:守望先生
- #include<stdio.h>
- __attribute__((weak)) void my_print();
- void test_print()
- {
- // 如果是強(qiáng)符號(hào),說明鏈接了外部插件,使用外部定義
- if(my_print)
- {
- my_print();
- }
- else
- {
- // 弱符號(hào),走默認(rèn)邏輯
- printf("this is weak print\n");
- }
- }
- int main(void)
- {
- test_print();
- return 0;
- }
上面的test_print函數(shù)是弱符號(hào),在沒有其他地方定義的情況下,也是能夠正常編譯運(yùn)行的:
- $ gcc -o main main.c
- $ ./main
- this is weak print
觀察可執(zhí)行文件:
- $ nm main |grep my_print
- w my_print
通過nm命令我們也可以知道test_print是弱符號(hào),它前面的修飾字符是W,代表weak。
插件庫
前面的示例程序已經(jīng)能否工作了,如何讓它能否支持插件庫呢?或者說,如何讓它支持外部的插件功能呢?
關(guān)于制作庫(靜態(tài)庫或動(dòng)態(tài)庫制作可以參考《手把手教你制作靜態(tài)庫》)
這里以靜態(tài)庫為例:
- // print_plugin.c
- #include<stdio.h>
- void my_print()
- {
- printf("this is plugin print\n");
- }
制作靜態(tài)庫:
- $ gcc -c print_plugin.c
- $ ar -rcs libprint_plugin.a print_plugin.o
鏈接插件庫
現(xiàn)在重新編譯main程序,并使用插件庫:
- $ gcc -o main main.c -L./ -lprint_plugin
- $ gcc -o main main.c -L. -Wl,--whole-archive -lprint_plugin -Wl,--no-whole-archive
- $ nm main |grep my_print
- 000000000000067a T my_print
- $ ./main
- this is plugin print
需要注意的是,這里在鏈接插件庫之前,需要加上:
- -Wl,--whole-archive
該選項(xiàng)會(huì)將插件庫中所有符號(hào)都鏈接進(jìn)來,若非如此,在main.c中已經(jīng)有了my_print符號(hào),將不會(huì)鏈接進(jìn)來,而在此之后,又要將該選項(xiàng)恢復(fù)。最終我們可以通過nm命令看到my_print符號(hào)已經(jīng)不再是W了。也就看到了最后:
- this is plugin print
的打印了。
也就實(shí)現(xiàn)了我們所謂插件的功能,換句話說,可以對(duì)目標(biāo)程序進(jìn)行功能的裁剪或者增加。
總結(jié)
由于以下幾點(diǎn)原因,我們可以自己做一些支持插件庫的程序:
1.重復(fù)強(qiáng)弱符號(hào)同存在時(shí),使用強(qiáng)符號(hào)
2.弱符號(hào)鏈接不存在時(shí),不會(huì)報(bào)錯(cuò)
3.未鏈接的外部符號(hào),地址為0,可通過判斷避免訪問非法地址
再結(jié)合前面的例子分別解釋一下:
1.這一點(diǎn)在《什么是強(qiáng)符號(hào)和弱符號(hào)》一文中已經(jīng)有解釋說明了
2.在開始的程序中,即便沒有鏈接插件庫,程序也可以正常編譯鏈接通過,而不會(huì)報(bào)錯(cuò)
3.沒有鏈接插件庫時(shí),由于其函數(shù)地址為0,因此,我們程序內(nèi)判斷,if(xxx),當(dāng)?shù)刂窞?時(shí),執(zhí)行默認(rèn)的行為語句。