先说结论:linux内核模块间通信可以使用符号导出函数
-
EXPORT_SYMBOL( )
标签内定义函数对全部内核代码公开,不用修改内核代码就可以在其他内核模块中直接调用。EXPORT_SYMBOL_GPL( )
和前面一个类似,但这个范围只适合GPL许可的模块进行调用。
函数标签内定义函数对内核全部代码公开,不用修改内核代码就可以在另外一个内核模块中直接调用。
- 如何使用?
假设模块B要调用模块A中的函数:
在模块A的c文件或者头文件中使用 EXPORT_SYMBOL(xxxx) 导出函数。
有些需要添加编译选项 -DEXPORT_SYMTAB
在模块B中用 “extern” 声明函数即可: extern int xxxx。
这种方式引用的时候需要在引用此变量的模块中修改Makefile.
KBUILD_EXTRA_SYMBOLS += /path/to/Modules.symvers
export KBUILD_EXTRA_SYMBOLS
- 说穿了:linux内核仍旧是一个大程序,当动态加载进入一个模块时,这个模块还是内核的一部分。所以可以像一个工程项目那样,将函数导出到内核符号表中,再在需要调用的外部声明为extern关键字,就可以方便的调用了。
- 在导出函数以后,可以使用 cat proc/kallsyms来查看所有的导出符号,其中属性为 t 的标识是不能被调用的,所以如果导出符号是 t 类型,那么无法直接被其他模块使用。
- 在linux2.6.26版本以后的内核中,模块的符号导出经常会出现问题,一个模块中的导出符号不能被另外一个模块进行调用。这个使得处理有依赖关系的模块非常的麻烦。
-
无法导出问题解决:
(1)在A模块编译好后会生成符号表文件 Module_symvers,里面有函数地址和函数名对应关系,把这个文件拷贝到需要调用的B的源代码下,替换B的该文件。然后重新编译B模块.这样就能够让B调用A的函数,以后加载模块顺序也必须先A后B,卸载相反。
(2)将两个模块放在一个目录下,进行编译。
这样就能够成功的实现两个模块之间的函数调用,比如KVM如果需要和驱动模块相互调用,就能使用这个方法。如果是两个模块之间需要相互调用,可以让驱动模块函数导出,KVM模块将函数指针当做回调函数传给驱动,是想双方的函数调用通讯。