1、Linux中EXPORT_SYMBOL的用法
EXPORT_SYMBOL标签内定义的函数对全部内核代码公开,不用修改内核代码就可以在您的内核模块中直接调用。您还可以手工修改内核源代码来导出另外的函数,用于重新编译并加载新内核后的测试。
//mod1.c
#include<linux/init.h> #include<linux/module.h> #include<linux/kernel.h> MODULE_LICENSE("DUAL BSD/GPL"); int func1(void) { printk("In Func: %s.../n",__func__); return 0; } static int __init hello_init(void) { printk("Module 1,Init!/n"); return 0; } static void __exit hello_exit(void) { printk("Module 1,Exit!/n"); } EXPORT_SYMBOL(func1); module_init(hello_init); module_exit(hello_exit);
//mod2.c
#include<linux/init.h>
#include<linux/kernel.h>
#include<linux/module.h>
MODULE_LICENSE("DUAL BSD/GPL");
extern int func1(void);
static int func2(void)
{
func1();
printk("In Func: %s.../n",__func__);
return 0;
}
static int __init hello_init(void)
{
printk("Module 2,Init!/n");
func2();
return 0;
}
static void __exit hello_exit(void)
{
printk("Module 2,Exit!/n");
}
module_init(hello_init);
module_exit(hello_exit);
//Makefile
ifneq ($(KERNELRELEASE),) obj-m := xxx.o else KDIR := /lib/modules/$(shell uname -r)/build PWD := $(shell pwd) default: $(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules clean: rm -rf Module.symvers *.ko *.o *.mod.c .*.cmd .tmp_versions *~ *.*~ endif
2、 运行
有两个模块,mod1和mod2。
在mod1中定义了func()1函数,并且经EXPORT_SYMBOL()导出。
在mod2中extern func()1,调用func()1。
①insmod mod1.ko
②insmod mod2.ko
编译模块mod2,成功。
加载mod2时,输出:
insmod: error inserting ‘mod2.ko‘: -1 Invalid parameters
dmesg查看:
mod2: no symbol version for func
mod2: Unknown symbol func (err -22)
③cat /proc/kallsyms |grep func1
了解了什么是内核符号表之后,我们回到之前的问题。
我们查看/proc/kallsyms,发现mod1的func函数的标志为t,而此标志表示函数是局部的。
此问题是内核2.6.26之后版本的bug,并且不会被修复
3、解决办法
(1)把mod1的Module.symvers放到mod2中,这样在编译mod2时符号信息会自动链接进去。
(2)在mod2的Makefile中添加符号信息
echo ‘0x01873e3f func mod1 EXPORT_SYMBOL_GPL‘ > Module.symvers
这样mod2在编译时就知道mod1中func符号的地址。
Q:这个问题是由什么引起的呢?
A:生成mod2的时候不知道mod1中func的校验码,mod2加载时检查校验码出错。
在内核主线代码树的一个提交修改了内核挂载模块时的函数版本校验机制,使得在挂载模块时候对于编译
时个别函数没有确定CRC校验值无法通过check_version函数检查。
这是内核有意要禁止存在个别无版本校验信息的函数的模块挂载。