Linux内核跨模块函数调用:EXPORT_SYMBOL()宏定义

一、查看内核驱动代码你会发现很多的函数带有EXPORT_SYMBOL()宏定义。

二、那么EXPORT_SYMBOL的作用是什么?
           EXPORT_SYMBOL标签内定义的函数或者符号对全部内核代码公开,不用修改内核代码就可以在您的内核模块中直接调用,即使用EXPORT_SYMBOL可以将一个函数以符号的方式导出给其他模块使用。


三、使用方法
         1、在模块函数定义之后使用“EXPORT_SYMBOL(函数名)”来声明。
         2、在调用该函数的另外一个模块中使用extern对之声明。
         3、先加载定义该函数的模块,然后再加载调用该函数的模块,请注意这个先后顺序。

四、实例测试。

          1、测试思路:在模块export_symbol_one中定义一个函数function_one(void);在另外一个模块export_symbol_two中定义一个函数function_two(void),function_two(void)里面会调用function_one(void)。

          2、测试代码:

export_symbol_one.c

    #include<linux/init.h>
    #include<linux/module.h>
    #include<linux/kernel.h>
     
    static int function_one(void)
    {
            printk("EXPORT_SYMBOL  In Func: %s...\n",__func__);
            return 0;
    }
     
    EXPORT_SYMBOL(function_one);
     
    static int __init export_symbol_init(void)
    {
            printk("EXPORT_SYMBOL  Module one,Init!\n");
            return 0;
    }
     
    static void __exit export_symbol_exit(void)
    {
            printk("EXPORT_SYMBOL  Module one,Exit!\n");
    }
     
    module_init(export_symbol_init);
    module_exit(export_symbol_exit);

Makefile:

    obj-m += export_symbol_one.o
     
    KDIR := /home/weifanghai/Android_4.4_git/xunwei/kernel/iTop4412_Kernel_3.0
    PWD ?= $(shell pwd)
     
     
    all:
        make -C $(KDIR) M=$(PWD) modules
            
    clean:
        rm -rf *.o


export_symbol_two.c

    #include<linux/init.h>
    #include<linux/kernel.h>
    #include<linux/module.h>
     
    static int function_two(void)
    {
            extern int function_one(void);
            function_one();
            printk("EXPORT_SYMBOL In Func: %s...\n",__func__);
            return 0;
    }
     
    static int __init export_symbol_init(void)
    {
            printk("EXPORT_SYMBOL  Module two,Init!\n");
            function_two();
            return 0;
    }
     
    static void __exit export_symbol_exit(void)
    {
            printk("EXPORT_SYMBOL  Module two,Exit!\n");
    }
     
    module_init(export_symbol_init);
    module_exit(export_symbol_exit);


Makefile

    obj-m += export_symbol_two.o
     
    KDIR := /home/weifanghai/Android_4.4_git/xunwei/kernel/iTop4412_Kernel_3.0
    PWD ?= $(shell pwd)
     
     
    all:
        make -C $(KDIR) M=$(PWD) modules
            
    clean:
        rm -rf *.o

        3、依次加载两个模块:

          [root@iTOP-4412]# insmod export_symbol_one.ko

          [root@iTOP-4412]# insmod export_symbol_two.ko

       4、运行结果效果图:


      5、小结:从上面的打印信息看,模块export_symbol_two.ko里面的函数function_two(void)可以调用模块export_symbol_one.ko里面的函数function_one(void)。
 





Linux的EXPORT_SYMBOL和EXPORT_SYMBOL_GPL的使用和区别

简要说明使用方法:

一个模块mod1中定义一个函数func1;在另外一个模块mod2中定义一个函数func2,func2调用func1。
在模块mod1中,EXPORT_SYMBOL(func1);
在模块mod2中,extern int func1();
就可以在mod2中调用func1了。

同理EXPORT_SYMBOL_GPL使用相同。
1、EXPORT_SYMBOL的作用是什么?

EXPORT_SYMBOL标签内定义的函数或者符号对全部内核代码公开,不用修改内核代码就可以在您的内核模块中直接调用,即使用EXPORT_SYMBOL可以将一个函数以符号的方式导出给其他模块使用。
这里要和System.map做一下对比:
System.map 中的是连接时的函数地址。连接完成以后,在2.6内核运行过程中,是不知道哪个符号在哪个地址的。
EXPORT_SYMBOL的符号,是把这些符号和对应的地址保存起来,在内核运行的过程中,可以找到这些符号对应的地址。而模块在加载过程中,其本质就是能动态连接到内核,
如果在模块中引用了内核或其它模块的符号,就要EXPORT_SYMBOL这些符号,这样才能找到对应的地址连接。
2、使用方法

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值