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