(资料图片仅供参考)
不知大家有没有想过,在一个内核模块代码中,会用到printk
函数,而这个函数不是我们实现的,它是内核代码的一部分,但我们为什么能够编译通过呢?
我们的代码之所以能够编译通过,是因为对模块的编译仅仅是编译,并没有链接。
而printk
的 符号类型是U
,表示它是一个未决符号。意思是说在编译阶段不知道这个符号的地址,因为它被定义在其他文件中,没有放在模块代码一起编译。
1、EXPORT_SYMBOL导出符号
2、内核模块依赖
#include#include#includeexternintexpval;externvoidexpfun(void);staticint__initvser_init(void){printk("vser_init\");printk("expval:%d\n",expval);expfun();return0;}staticvoid__exitvser_exit(void){printk("vser_exit\n");}module_init(vser_init);module_exit(vser_exit);
dep.c
#include#includestaticintexpval=5;EXPORT_SYMBOL(expval);staticvoidexpfun(void){printk("expfun");}EXPORT_SYMBOL_GPL(expfun);
Makefile关键处:
obj-m:=vser.oobj-m+=dep.o
上述代码中,dep.c定义了一个变量expval
和一个函数expfun
,并分别用EXPORT_SYMBOL
和EXPORT_SYMBOL_GPL
导出。而vser.c
里则调用了dep.c
的变量和函数,编译安装后:
#modprobevser#dmesg[58278.204677]vser_init[58278.204683]expval:5[58287.206464]expfun
从输出信息中可以看到,vser.c
正确引用到了dep.c
的变量和函数。
#cat/lib/modules/5.10.111-64-generic/modules.dep......extra/vser.ko:extra/dep.koextra/dep.ko:
WARNING:"expfun"[/home/ubuntu/driver/module/vser.ko]undefined!WARNING:"expval"[/home/ubuntu/driver/module/vser.ko]undefined!#sudoinsmoddep.ko#sudoinsmodvser.koinsmod:errorinserting"vser.ko":-1Invalidparameters
这是因为在编译vser
模块时在内核的符号表中找不到expval
和expfun
的项,而vser
模块又完全不知道dep
模块的存在。
内核将会创建模块依赖关系的链接,只有当依赖于这个模块的链表为空时,模块才能被卸载。