EXPORT_SYMBOL使用
EXPORT_SYMBOL只出現(xiàn)在2.6內(nèi)核中,在2.4內(nèi)核默認(rèn)的非static 函數(shù)和變量都會(huì)自動(dòng)導(dǎo)入到kernel 空間的, 都不用EXPORT_SYMBOL() 做標(biāo)記的。
2.6就必須用EXPORT_SYMBOL() 來(lái)導(dǎo)出來(lái)(因?yàn)?.6默認(rèn)不到處所有的符號(hào))。?
1、EXPORT_SYMBOL的作用
EXPORT_SYMBOL標(biāo)簽內(nèi)定義的函數(shù)或者符號(hào)對(duì)全部?jī)?nèi)核代碼公開(kāi),不用修改內(nèi)核代碼就可以在您的內(nèi)核模塊中直接調(diào)用,即使用EXPORT_SYMBOL可以將一個(gè)函數(shù)以符號(hào)的方式導(dǎo)出給其他模塊使用。
這里要和System.map做一下對(duì)比:
System.map 中的是連接時(shí)的函數(shù)地址。連接完成以后,在2.6內(nèi)核運(yùn)行過(guò)程中,是不知道哪個(gè)符號(hào)在哪個(gè)地址的。
EXPORT_SYMBOL 的符號(hào), 是把這些符號(hào)和對(duì)應(yīng)的地址保存起來(lái),在內(nèi)核運(yùn)行的過(guò)程中,可以找到這些符號(hào)對(duì)應(yīng)的地址。
而模塊在加載過(guò)程中,其本質(zhì)就是能動(dòng)態(tài)連接到內(nèi)核,如果在模塊中引用了內(nèi)核或其它模塊的符號(hào),就要EXPORT_SYMBOL這些符號(hào),這樣才能找到對(duì)應(yīng)的地址連接。
使用方法:
? ?第一、在模塊函數(shù)定義之后使用EXPORT_SYMBOL(函數(shù)名)
?? 第二、在掉用該函數(shù)的模塊中使用extern對(duì)之聲明
?? 第三、首先加載定義該函數(shù)的模塊,再加載調(diào)用該函數(shù)的模塊
另外,在編譯調(diào)用某導(dǎo)出函數(shù)的模塊時(shí),往往會(huì)有WARNING: "****" [**********] undefined!
使用dmesg命令后會(huì)看到相同的信息。開(kāi)始我以為只要有這個(gè)錯(cuò)誤就不能加載模塊,后來(lái)上網(wǎng)查了一下,發(fā)現(xiàn)這主要是因?yàn)樵诰幾g連接的時(shí)候還沒(méi)有和內(nèi)核打交道,當(dāng)然找不到symbol了,但是由于你生成的是一個(gè)內(nèi)核模塊,所以LD不提示error,而是給出一個(gè)warning,寄希望于在insmod的時(shí)候,內(nèi)核能夠把這個(gè)symbol連接上。
EXPORT_SYMBOL示范
比如有兩個(gè)驅(qū)動(dòng)模塊:Module A和Module B,其中Module B使用了Module A中的export的函數(shù),因此在Module B的Makefile文件中必須添加:
KBUILD_EXTRA_SYMBOLS += /path/to/ModuleA/Module.symvers
export KBUILD_EXTRA_SYMBOLS
這樣在編譯Module B時(shí),才不會(huì)出現(xiàn)Warning,提示說(shuō)func1這個(gè)符號(hào)找不到,而導(dǎo)致編譯得到的ko加載時(shí)也會(huì)出錯(cuò)。
// Module A (mod_a.c) #include<linux/init.h> #include<linux/module.h> #include<linux/kernel.h> static int func1(void) { printk("In Func: %s...\n",__func__); return 0; } // Export symbol func1 EXPORT_SYMBOL(func1); static int __init hello_init(void) { printk("Module 1,say hello world!\n"); return 0; } static void __exit hello_exit(void) { printk("Module 1,Exit!\n"); } module_init(hello_init); module_exit(hello_exit); // Module B (mod_b.c) #include<linux/init.h> #include<linux/kernel.h> #include<linux/module.h> extern int functl(void); static int func2(void) { func1(); printk("In Func: %s...\n",__func__); return 0; } static int __init hello_init(void) { printk("Module 2,is used Module 1 function!\n"); func2(); return 0; } static void __exit hello_exit(void) { printk("Module 2,Exit!\n"); } module_init(hello_init); module_exit(hello_exit); 在驅(qū)動(dòng)加載的時(shí)候,一定要先加載定義function1的Module A模塊,然后再加載調(diào)用function1的Module B的驅(qū)動(dòng)模塊。 如下操作: insmod Module_A.ko insmod Module_B.ko?
轉(zhuǎn)載于:https://www.cnblogs.com/Caden-liu8888/p/7725293.html
總結(jié)
以上是生活随笔為你收集整理的EXPORT_SYMBOL使用的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 慧创印刷电脑系统(慧印刷官网)
- 下一篇: Jzoj4782 Math