Linux 驱动之EXPORT_SYMBOL 函数以及2.6内核 Unknown symbol bug解决办法

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函数检查。

这是内核有意要禁止存在个别无版本校验信息的函数的模块挂载。






郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。