Linux模块导出符号及其引用

Linux模块间通讯方法非常的多,最便捷的方法莫过于函数符号导出,然后直接调用。
然而在linux2.6.26以后的 内核中模块的符号导出经常会出现问题,一个模块中的导出符号不能被另外一个模块进行调用。这个使得处理有依赖关系的模块非常的头疼。

  1. 模块符号导出

     EXPORT_SYMBOL() :
    
     EXPORT_SYMBOL标签内定义的函数对全部内核代码公开,不用修改内核代码就可以 在您的内核模块中直接调用。
    
     EXPORT_SYMBOL_GPL() :
    
     和前面一个类 似,但这个范围只适合GPL许可的模块进行调用。
    
  2. 使用说明

     假设,模块B调用模块A的导出函数。
    
     [1] 在模块A中.c源文件或者.h头文件中使用EXPORT_SYMBOL(xxxx) 导出函数.
     有些需要添加编译选项 -DEXPORT_SYMTAB。
    
     [2] 在模块B中用 "extern" 申明函数(如, extern int xxxx),申明以后就能够直接使用导出的函数了。
    
     另外:在导出函数以后,可以使用 cat  proc/kallsyms来查看所有的导出符号,其中属性为 t的标识是不能被调用的,所以如果导出符号是t类型,那么无法直接被其他模块使用。
    
  3. 无法导出问题解决

     情况一: 在模块A编译好后会生成符号表文件Module.symvers, 里面有函数地址和函数名对应关系,把这个文件拷贝到需要调用的模块B的源代码下,替换模块B的该文件。
    
     然后重新编译B模块.这样就能够让模块B调用模块A的函数,以后加载模块顺序也必须先A后B,卸载相反。
    
    
    
     情况二: 将两个模块放在一个目录下,进行编译。其实和方法一类似。
    
    这样就能够成功的实现两个模块之间的函数调用,比如KVM如果需要和驱动模块相互调用,就能使用这个方法。如果是两个模块之间需要相互调用,可以让驱动模块函数导出,KVM模块将函数指针当做回调函数传给驱动,是想双方的函数调用通讯。
    

使用nm -l *.ko可以查看模块所需的所有导出符号。
cat /proc/kallsyms用于输出内核中导出的所有符号。

你可能感兴趣的:(linux驱动)