linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
2017-03-25 12:29
393 查看
转自http://blog.csdn.net/lisan04/article/details/4076013
linux2.6的“/prob/kallsyms”文件对应着内核符号表,记录了符号以及符号所在的内存地址。
内核符号表:记录了内核中所有的符号(函数、全局变量等)的地址以及名字,这个符号表被嵌入到内核镜像中,使得内核可以在运行过程中随时获得一个符号地址对应的符号名
模块可以使用如下宏导出符号到内核符号表:
[c-sharp]
view plain
copy
print?
EXPORT_SYMBOL(符号名);
EXPORT_SYMBOL_GPL(符号名)
导出的符号可以被其他模块使用,不过使用之前一定要声明一下。EXPORT_SYMBOL_GPL()只适用于包含GPL许可权的模块。
代码演示:
[c-sharp]
view plain
copy
print?
//hello.c文件,定义2个函数,用于导出
#include <<a href="http://lib.csdn.net/base/linux" class='replace_word' title="Linux知识库" target='_blank' style='color:#df3434; font-weight:bold;'>Linux</a>/init.h>
#include <linux/module.h>
MODULE_LICENSE("Dual BSD/GPL");
int add_integar(int a,int b)
{
return a + b;
}
int sub_integar(int a,int b)
{
return a - b;
}
EXPORT_SYMBOL(add_integar);
EXPORT_SYMBOL(sub_integar);
//test.c 用于调用hello模块导出的函数
#include <linux/init.h>
#include <linux/module.h>
MODULE_LICENSE("Dual BSD/GPL");
extern int add_integar(int ,int); //声明要调用的函数
extern int sub_integar(int ,int); //声明要调用的函数
int result(void)
{
int a,b;
a = add_integar(1,1);
b = sub_integar(1,1);
printk("%d/n",a);
printk("%d/n",b);
return 0;
}
make后,先加在hello模块,再加载test模块。
然后cat /proc/kallsyms | grep integer
显示:
[c-sharp]
view plain
copy
print?
[root@localhost test]# cat /proc/kallsyms |grep integar
e053d000 u add_integar [test]
e053d004 u sub_integar [test]
e053d02c r __ksymtab_sub_integar [hello]
e053d03c r __kstrtab_sub_integar [hello]
e053d034 r __ksymtab_add_integar [hello]
e053d048 r __kstrtab_add_integar [hello]
e053d000 T add_integar [hello]
e053d004 T sub_integar [hello]
这个应该可以很容易看出,第一列为符号地址,第二列为类型,第三列为符号名,第四列为所在的文件。
注意:如果发现符号地址均为0,那是因为系统保护。使用root权限查看即可。
第二列的类型:
有的符号是大写的,有的是小写。大写的符号是全局的。
b 符号在未初始化数据区(BSS)
c 普通符号,是未初始化区域
d 符号在初始化数据区
g 符号针对小object,在初始化数据区
i 非直接引用其他符号的符号
n 调试符号
r 符号在只读数据区
s 符号针对小object,在未初始化数据区
t 符号在代码段
u 符号未定义
linux2.6的“/prob/kallsyms”文件对应着内核符号表,记录了符号以及符号所在的内存地址。
内核符号表:记录了内核中所有的符号(函数、全局变量等)的地址以及名字,这个符号表被嵌入到内核镜像中,使得内核可以在运行过程中随时获得一个符号地址对应的符号名
模块可以使用如下宏导出符号到内核符号表:
[c-sharp]
view plain
copy
print?
EXPORT_SYMBOL(符号名);
EXPORT_SYMBOL_GPL(符号名)
EXPORT_SYMBOL(符号名); EXPORT_SYMBOL_GPL(符号名)
导出的符号可以被其他模块使用,不过使用之前一定要声明一下。EXPORT_SYMBOL_GPL()只适用于包含GPL许可权的模块。
代码演示:
[c-sharp]
view plain
copy
print?
//hello.c文件,定义2个函数,用于导出
#include <<a href="http://lib.csdn.net/base/linux" class='replace_word' title="Linux知识库" target='_blank' style='color:#df3434; font-weight:bold;'>Linux</a>/init.h>
#include <linux/module.h>
MODULE_LICENSE("Dual BSD/GPL");
int add_integar(int a,int b)
{
return a + b;
}
int sub_integar(int a,int b)
{
return a - b;
}
EXPORT_SYMBOL(add_integar);
EXPORT_SYMBOL(sub_integar);
//test.c 用于调用hello模块导出的函数
#include <linux/init.h>
#include <linux/module.h>
MODULE_LICENSE("Dual BSD/GPL");
extern int add_integar(int ,int); //声明要调用的函数
extern int sub_integar(int ,int); //声明要调用的函数
int result(void)
{
int a,b;
a = add_integar(1,1);
b = sub_integar(1,1);
printk("%d/n",a);
printk("%d/n",b);
return 0;
}
//hello.c文件,定义2个函数,用于导出 #include <<a href="http://lib.csdn.net/base/linux" class='replace_word' title="Linux知识库" target='_blank' style='color:#df3434; font-weight:bold;'>Linux</a>/init.h> #include <linux/module.h> MODULE_LICENSE("Dual BSD/GPL"); int add_integar(int a,int b) { return a + b; } int sub_integar(int a,int b) { return a - b; } EXPORT_SYMBOL(add_integar); EXPORT_SYMBOL(sub_integar); //test.c 用于调用hello模块导出的函数 #include <linux/init.h> #include <linux/module.h> MODULE_LICENSE("Dual BSD/GPL"); extern int add_integar(int ,int); //声明要调用的函数 extern int sub_integar(int ,int); //声明要调用的函数 int result(void) { int a,b; a = add_integar(1,1); b = sub_integar(1,1); printk("%d/n",a); printk("%d/n",b); return 0; }
make后,先加在hello模块,再加载test模块。
然后cat /proc/kallsyms | grep integer
显示:
[c-sharp]
view plain
copy
print?
[root@localhost test]# cat /proc/kallsyms |grep integar
e053d000 u add_integar [test]
e053d004 u sub_integar [test]
e053d02c r __ksymtab_sub_integar [hello]
e053d03c r __kstrtab_sub_integar [hello]
e053d034 r __ksymtab_add_integar [hello]
e053d048 r __kstrtab_add_integar [hello]
e053d000 T add_integar [hello]
e053d004 T sub_integar [hello]
列表的项:
这个应该可以很容易看出,第一列为符号地址,第二列为类型,第三列为符号名,第四列为所在的文件。注意:如果发现符号地址均为0,那是因为系统保护。使用root权限查看即可。
第二列的类型:
有的符号是大写的,有的是小写。大写的符号是全局的。
b 符号在未初始化数据区(BSS)
c 普通符号,是未初始化区域
d 符号在初始化数据区
g 符号针对小object,在初始化数据区
i 非直接引用其他符号的符号
n 调试符号
r 符号在只读数据区
s 符号针对小object,在未初始化数据区
t 符号在代码段
u 符号未定义
相关文章推荐
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL .
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- Linux模块导出符号 EXPORT_SYMBOL_GPL EXPORT_SYMBOL
- linux模块导出符号 EXPORT_SYMBOL_GPL&EXPORT_SYMBOL(转)
- driver: linux2.6 内核模块导出函数实例(EXPORT_SYMBOL)
- driver: linux2.6 内核模块导出函数实例(EXPORT_SYMBOL)
- driver: linux2.6 内核模块导出函数实例(EXPORT_SYMBOL)
- 内核模块编程---符号导出(EXPORT_SYMBOL()) (3)
- driver: linux2.6 内核模块导出函数实例(EXPORT_SYMBOL) 【转】
- Linux内核导出符号宏定义EXPORT_SYMBOL的源码分析
- driver: linux2.6 内核模块导出函数实例(EXPORT_SYMBOL)
- Linux 驱动开发之内核模块开发(四)—— 符号表的导出