EXPORT_SYMBOL只出現在2.6內核中,在2.4內核默認的非static 函數和變量都會自動導入到kernel 空間的, 都不用EXPORT_SYMBOL() 做標記的。
2.6就必須用EXPORT_SYMBOL() 來導出來(因為2.6默認不到處所有的符號)。
1、EXPORT_SYMBOL的作用
EXPORT_SYMBOL標簽內定義的函數或者符號對全部內核代碼公開,不用修改內核代碼就可以在您的內核模塊中直接調用,即使用EXPORT_SYMBOL可以將一個函數以符號的方式導出給其他模塊使用。
這里要和System.map做一下對比:
System.map 中的是連接時的函數地址。連接完成以后,在2.6內核運行過程中,是不知道哪個符號在哪個地址的。
EXPORT_SYMBOL 的符號, 是把這些符號和對應的地址保存起來,在內核運行的過程中,可以找到這些符號對應的地址。
而模塊在加載過程中,其本質就是能動態連接到內核,如果在模塊中引用了內核或其它模塊的符號,就要EXPORT_SYMBOL這些符號,這樣才能找到對應的地址連接。
使用方法:
第一、在模塊函數定義之后使用EXPORT_SYMBOL(函數名)
第二、在掉用該函數的模塊中使用extern對之聲明
第三、首先加載定義該函數的模塊,再加載調用該函數的模塊
另外,在編譯調用某導出函數的模塊時,往往會有WARNING: "****" [**********] undefined!
使用dmesg命令后會看到相同的信息。開始我以為只要有這個錯誤就不能加載模塊,后來上網查了一下,發現這主要是因為在編譯連接的時候還沒有和內核打交道,當然找不到symbol了,但是由於你生成的是一個內核模塊,所以LD不提示error,而是給出一個warning,寄希望於在insmod的時候,內核能夠把這個symbol連接上。
一個模塊mod1中定義一個函數func1;在另外一個模塊mod2中定義一個函數func2,func2調用func1。 在模塊mod1中,EXPORT_SYMBOL(func1); 在模塊mod2中,extern int func1(); 就可以在mod2中調用func1了
EXPORT_SYMBOL示范
比如有兩個驅動模塊:Module A和Module B,其中Module B使用了Module A中的export的函數,因此在Module B的Makefile文件中必須添加:
KBUILD_EXTRA_SYMBOLS += /path/to/ModuleA/Module.symvers
export KBUILD_EXTRA_SYMBOLS
這樣在編譯Module B時,才不會出現Warning,提示說func1這個符號找不到,而導致編譯得到的ko加載時也會出錯。
// 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);
insmod Module_A.ko
insmod Module_B.ko