EXPORT_SYMBOL使用


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);  
在驅動加載的時候,一定要先加載定義function1的Module A模塊,然后再加載調用function1的Module B的驅動模塊。
如下操作:
insmod Module_A.ko  
insmod Module_B.ko  

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM