Chinaunix首页 | 论坛 | 博客
  • 博客访问: 750560
  • 博文数量: 215
  • 博客积分: 291
  • 博客等级: 二等列兵
  • 技术积分: 1031
  • 用 户 组: 普通用户
  • 注册时间: 2010-05-12 18:17
文章分类

全部博文(215)

文章存档

2016年(16)

2015年(16)

2014年(123)

2013年(60)

分类: 嵌入式

2013-09-12 13:14:21

linux-3.1.4下的驱动模块 "Unknown symbol in module" 问题(by liukun321咕唧咕唧)

分类: linux 移植 3035人阅读 评论(1) 举报

    在加载自己写的3.1.4内核的adc驱动时,频频出现下面的错误提示,以前在2.6.29的内核上从未碰到:

[root@LK2440 drivers]# insmod /lib/modules/adc.ko
adc: module license 'unspecified' taints kernel.
Disabling lock debugging due to kernel taint
adc: Unknown symbol platform_get_irq (err 0)
adc: Unknown symbol platform_driver_register (err 0)
adc: Unknown symbol platform_get_resource (err 0)
insmod: cannot insert '/lib/modules/adc.ko': unknown symbol in module or invalid parameter

   对于经常写驱动的大牛来说这可能根本不是问题,但是它却困扰我一段时间。反复查阅网上的资料,找到可以从以下几点来解决这个问题,在此记录:

据CSDN的一位网友  所介绍的:

这是linux kernel 2.6.26 之后版本的bug (详细描述, 请看)
并且这个bug不会被fix

解决办法是把mod_a的Module.symvers放到mod_b的当前路径,从而编译mod_b,符号信息会自动连接进去.
或者在mod_b的makefile中使用KBUILD_EXTRA_SYMBOLS指定mod_a的Module.symvers, 如:
KBUILD_EXTRA_SYMBOLS=/mod_a/Module.symvers

编译mod_b时,搜索Module.symvers的路径是:
1, kernel source path, e.g. /usr/src/kernels/linux-2.6.28.10
2, makefile中M=所指定的路径, 它等效于变量KBUILD_EXTMOD的值
3, 变量KBUILD_EXTRA_SYMBOLS的值

   我按照上面所叙述的对我的内核模块的Makefile 做了修改,在编译命令那行加了“KBUILD_EXTRA_SYMBOLS=/linux-3.1.4/Module.symvers”参数,编译问题仍 没解决。我想细心地朋友,也许发现了问题根本所在,adc: module license 'unspecified' taints kernel.
   对,主要问题就是没有模块许可证声明。而我在添加MODULE_LICENSE("GPL");  后问题解决了。

=====================================================================
可能很多高手早就知道这样处理,但是俺是最近才琢磨明白,呵呵

(1)对编译所得的.ko进行strip -S,处理掉调试信息,这样可以大大缩小ko文件的大小

(2)使用KBUILD_EXTRA_SYMBOLS
主要使用于下面这样的场合:
有两个我们自己的模块,其中Module B使用了Module A中export的函数,因此在Module B的Makefile中必须加上这样的选项
KBUILD_EXTRA_SYMBOLS += /path/to/Module A/Module.symvers
export KBUILD_EXTRA_SYMBOLS

这样在编译Module B时,才不会出现Warning,提示说func1这个符号找不到,而编译得到的ko加载时也会出错。

需要注意的是:
(1)KBUILD_EXTRA_SYMBOLS 赋值时必须使用+=,而不能使用=及:=,这个是实验出来的,至于为什么必须这样,还不明白。如果哪位高手知道,还请指教~~
(2)KBUILD_EXTRA_SYMBOLS 必须使用export处理一下

  1. // Module A (mod_a.c)
  2. #include
  3. #include
  4. #include

  5. static int func1(void)
  6. {
  7.        printk("In Func: %s...\n",__func__);
  8.        return 0;
  9. }

  10. // Export symbol func1
  11. EXPORT_SYMBOL(func1);

  12. static int __init hello_init(void)
  13. {
  14.        printk("Module 1,Init!\n");
  15.        return 0;
  16. }

  17. static void __exit hello_exit(void)
  18. {
  19.        printk("Module 1,Exit!\n");
  20. }

  21. module_init(hello_init);
  22. module_exit(hello_exit);
复制代码
  1. // Module B (mod_b.c)
  2. #include
  3. #include
  4. #include

  5. static int func2(void)
  6. {
  7.        extern int func1(void);
  8.        func1();
  9.        printk("In Func: %s...\n",__func__);
  10.        return 0;
  11. }

  12. static int __init hello_init(void)
  13. {
  14.        printk("Module 2,Init!\n");
  15.        func2();
  16.        return 0;
  17. }

  18. static void __exit hello_exit(void)
  19. {
  20.        printk("Module 2,Exit!\n");
  21. }

  22. module_init(hello_init);
  23. module_exit(hello_exit);
复制代码

阅读(3229) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~