driver: linux2.6 内核模块导出函数实例(EXPORT_SYMBOL) 【转】

简介:

转自:http://blog.chinaunix.net/uid-23381466-id-3837650.html

内核版本:2.6.38-11-generic

    内核自己都大量利用内核符号表导出函数,那么应该导出呢,ldd3上面说只需要EXPORT_SYMBOL一类的宏导出即可,结果试了很久都不行,最后查看文档,算是明白一点了。

    对于导出符号表,内核文档给出了三种解决方案,见尾部,现在忽略。

    现在有两个模块,a模块导出函数myprint,b模块使用该函数,想象一下如果a模块 EXPORT_SYMBOL(myprint) ,实际上b模块知道吗,很明显b模块对这件事情不是很清楚(这么说不是很准确),要调用a模块的myprint函数,需要知道myprint函数在内存中的位置,首先在内核符号表中是没有说明的,所以...

    当我们编译完a模块后,看看有些什么文件,是不是有一个Module.symvers文件,打开看看什么状况?
0x705034f7    myprint    /home/darren/Desktop/darren/print/myprint    EXPORT_SYMBOL
好了,这一行对b模块来说已经足够了,指定了内存位置,符号名称,模块路径。最简单的方法就是把这个文件复制到b模块所在目录,然后编译就没有讨厌的错误了,可以正常insmod模块。这种方法是内核文档中提到的方法之一。

    但是每次调用该函数都要复制一次,更糟糕的是a模块每编译一次,都要重新复制一次,为什么内核自己导出的函数我们可以直接用呢?现在就就解决:

    编译内核的时候同样会生成一个Module.symvers文件,内核导出的所有符号都在里面,我们在编译模块的时候实际上会调用内核的顶层makefile,也就是说内核的Module.symvers对我们的模块是可见的,和我们自己的Module.symvers文件一样,OK,把a模块的Module.symvers文件合并到内核的Module.symvers文件中,这时候myprint函数就成了真正的导出函数了,其他的模块只需要生命一下就可以用了。

代码如下
a模块代码

  1. #include <linux/module.h>
  2. #include <linux/init.h>
  3. #include <linux/kernel.h> 
  4. MODULE_LICENSE("GPL");
  5. int myprint(void)
  6. {
  7.     printk("c");
  8.     return 0;
  9. }
  10. static int darren_init(void)
  11. {
  12.     return 0;
  13. }
  14. static void darren_exit(void)
  15. {
  16. }
  17. module_init(darren_init);
  18. module_exit(darren_exit);
  19. EXPORT_SYMBOL(myprint);

b模块代码

  1. #include <linux/seq_file.h>
  2. #include <linux/cdev.h>
  3. #include <asm/system.h> 
  4. MODULE_LICENSE("GPL");
  5. extern int print(void);
  6. static int darren_init(void)
  7. {
  8.     int i=0;
  9.     printk("b module init\n");
  10.     for(;i<10;i++)print();
  11.     return 0;
  12. }
  13. static void darren_exit(void)
  14. {
  15. }
  16. module_init(darren_init);
  17. module_exit(darren_exit);

a模块的Makefile如下:

  1. NAME:=a
  2. SYM:=/usr/src/linux-headers-2.6.38-8-generic/Module.symvers
  3. DIR:=/lib/modules/$(shell uname -r)/build/
  4. PWD:=$(shell pwd)
  5. obj-m = $(NAME).o
  6. build: 
  7.     
  8.     make -C (DIR)M=(DIR)M=(PWD)
  9.     sudo chmod 777 $(SYM)
  10.     sudo sed -i '/myprint/d' $(SYM)
  11.     sudo cat Module.symvers>>$(SYM)
  12.     sudo chmod 644 $(SYM)


b模块的makefile:

  1. NAME:=b
  2. DIR:=/lib/modules/$(shell uname -r)/build/
  3. PWD:=$(shell pwd)
  4. obj-m = $(NAME).o
  5. build: 
  6.     make -C (DIR)M=(DIR)M=(PWD)


注意:路径/usr/src/linux-headers-2.6.38-8-generic/Module.symvers 有可能不对如果不行就改成/usr/src/linux-headers-`uname -r`-generic/Module.symvers

内核文档:

  1. Sometimes, an external module uses exported symbols from
  2.     another external module. kbuild needs to have full knowledge of
  3.     all symbols to avoid spitting out warnings about undefined
  4.     symbols. Three solutions exist for this situation.
  5.     NOTE: The method with a top-level kbuild file is recommended
  6.     but may be impractical in certain situations.
  7.     Use a top-level kbuild file
  8.         If you have two modules, foo.ko and bar.ko, where
  9.         foo.ko needs symbols from bar.ko, you can use a
  10.         common top-level kbuild file so both modules are
  11.         compiled in the same build. Consider the following
  12.         directory layout:
  13.         ./foo/ <= contains foo.ko
  14.         ./bar/ <= contains bar.ko
  15.         The top-level kbuild file would then look like:
  16.         #./Kbuild (or ./Makefile):
  17.             obj-y := foo/ bar/
  18.         And executing
  19.             makeCmake−CKDIR M=$PWD
  20.         will then do the expected and compile both modules with
  21.         full knowledge of symbols from either module.
  22.     Use an extra Module.symvers file
  23.         When an external module is built, a Module.symvers file
  24.         is generated containing all exported symbols which are
  25.         not defined in the kernel. To get access to symbols
  26.         from bar.ko, copy the Module.symvers file from the
  27.         compilation of bar.ko to the directory where foo.ko is
  28.         built. During the module build, kbuild will read the
  29.         Module.symvers file in the directory of the external
  30.         module, and when the build is finished, a new
  31.         Module.symvers file is created containing the sum of
  32.         all symbols defined and not part of the kernel.
  33.     Use "make" variable KBUILD_EXTRA_SYMBOLS
  34.         If it is impractical to copy Module.symvers from
  35.         another module, you can assign a space separated list
  36.         of files to KBUILD_EXTRA_SYMBOLS in your build file.
  37.         These files will be loaded by modpost during the
  38.         initialization of its symbol tables.













本文转自张昺华-sky博客园博客,原文链接:http://www.cnblogs.com/sky-heaven/p/4551671.html,如需转载请自行联系原作者


相关文章
|
11月前
|
存储 Linux
linux中的目录操作函数
本文详细介绍了Linux系统编程中常用的目录操作函数,包括创建目录、删除目录、读取目录内容、遍历目录树以及获取和修改目录属性。这些函数是进行文件系统操作的基础,通过示例代码展示了其具体用法。希望本文能帮助您更好地理解和应用这些目录操作函数,提高系统编程的效率和能力。
388 26
|
缓存 安全 Linux
Linux系统查看操作系统版本信息、CPU信息、模块信息
在Linux系统中,常用命令可帮助用户查看操作系统版本、CPU信息和模块信息
2497 23
|
Linux
【Linux】System V信号量详解以及semget()、semctl()和semop()函数讲解
System V信号量的概念及其在Linux中的使用,包括 `semget()`、`semctl()`和 `semop()`函数的具体使用方法。通过实际代码示例,演示了如何创建、初始化和使用信号量进行进程间同步。掌握这些知识,可以有效解决多进程编程中的同步问题,提高程序的可靠性和稳定性。
735 19
|
Linux Android开发 开发者
linux m、mm、mmm函数和make的区别
通过理解和合理使用这些命令,可以更高效地进行项目构建和管理,特别是在复杂的 Android 开发环境中。
726 18
|
关系型数据库 MySQL Linux
Linux下mysql数据库的导入与导出以及查看端口
本文详细介绍了在Linux下如何导入和导出MySQL数据库,以及查看MySQL运行端口的方法。通过这些操作,用户可以轻松进行数据库的备份与恢复,以及确认MySQL服务的运行状态和端口。掌握这些技能,对于日常数据库管理和维护非常重要。
601 8
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
581 13
|
Linux Shell
Linux系统编程:掌握popen函数的使用
记得在使用完 `popen`打开的流后,总是使用 `pclose`来正确关闭它,并回收资源。这种做法符合良好的编程习惯,有助于保持程序的健壮性和稳定性。
788 6
|
Unix Linux 网络安全
python中连接linux好用的模块paramiko(附带案例)
该文章详细介绍了如何使用Python的Paramiko模块来连接Linux服务器,包括安装配置及通过密码或密钥进行身份验证的示例。
854 1
|
Linux Shell
Linux系统编程:掌握popen函数的使用
记得在使用完 `popen`打开的流后,总是使用 `pclose`来正确关闭它,并回收资源。这种做法符合良好的编程习惯,有助于保持程序的健壮性和稳定性。
440 3
在Linux内核中根据函数指针输出函数名称
在Linux内核中根据函数指针输出函数名称

热门文章

最新文章