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

本文详细介绍了Linux内核中EXPORT_SYMBOL宏的用途,它用于将函数或符号公开给其他内核模块使用,允许模块之间直接调用。文章通过实例展示了如何在模块中定义和使用EXPORT_SYMBOL,以及加载模块的顺序。此外,还对比了System.map和EXPORT_SYMBOL的区别,并提及了EXPORT_SYMBOL_GPL的使用场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、查看内核驱动代码你会发现很多的函数带有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、使用方法

<think>好的,用户想了解 Linux 内核中的 EXPORT_SYMBOL 函数的使用说明和示例。首先,我需要回忆一下相关知识,并结合提供的引用内容来组织回答。 首先,用户提到的引用[1]和[2]已经解释了EXPORT_SYMBOL的基本作用,即导出符号供其他模块使用。需要先介绍它的基本概念和用途,然后说明其工作原理,比如符号表管理和模块加载过程。引用[4]提到宏定义展开和链接脚本,这部分可能需要稍微深入一点,但用户可能更关注用法,所以可以简要提及。 接下来,用户需要使用说明,应该分步骤讲解如何在代码中使用EXPORT_SYMBOL,比如定义函数、导出符号、其他模块声明并使用。同时,要给出示例代码,像引用[1]中的示例,但可能需要更详细,比如两个模块的代码示例,一个导出,一个使用。 注意事项部分需要包括作用域、依赖关系、版本兼容性等,引用[2]和[4]可能有相关内容,比如符号表管理和版本问题。此外,要提到GPL协议的问题,因为内核模块如果使用EXPORT_SYMBOL_GPL的话需要注意许可证兼容性。 然后生成相关问题,要围绕EXPORT_SYMBOL的相关知识点,比如原理、与GPL版本的区别、调试方法、应用场景等。 需要注意的是,回答要符合用户的要求,结构清晰,分点说明,同时正确使用LaTeX格式,比如变量名用$...$包裹,代码块用三个反引号标注语言。另外,引用标识要正确添加到对应的引用内容后面,比如在提到导出符号时引用[1],在原理部分引用[2]和[4]。 最后,检查是否有遗漏的信息,比如是否覆盖了使用步骤、示例、注意事项,以及相关问题是否相关。确保语言口语化,避免使用专业术语过多,保持自然的中文表达。</think>### EXPORT_SYMBOL 函数介绍及用法 #### 1. 核心功能 `EXPORT_SYMBOL` 是 Linux 内核提供的宏,用于将模块中的**函数或变量**导出到内核符号表,使其他内核模块能跨模块调用这些符号[^1]。这是实现模块间动态交互的基础机制。 #### 2. 工作原理 - **符号表管理**:导出的符号会被记录在`__ksymtab`段中,形成全局可见的符号表[^4] - **模块依赖**:加载模块时,内核通过符号表解析跨模块调用关系[^2] - **地址绑定**:编译时生成未解析的符号引用,运行时通过符号表完成地址绑定[^4] #### 3. 使用步骤 1. **定义待导出函数** ```c // module_a.c void my_exported_func(void) { printk("Exported function called\n"); } EXPORT_SYMBOL(my_exported_func); // 关键导出语句 ``` 2. **外部模块调用** ```c // module_b.c extern void my_exported_func(void); // 声明外部符号 static int __init mod_init(void) { my_exported_func(); // 跨模块调用 return 0; } ``` #### 4. 完整示例 **导出模块 (export_demo.c):** ```c #include <linux/init.h> #include <linux/module.h> int shared_var = 100; EXPORT_SYMBOL(shared_var); // 导出变量 void demo_func(void) { printk(KERN_INFO "Demo function executed\n"); } EXPORT_SYMBOL(demo_func); // 导出函数 MODULE_LICENSE("GPL"); ``` **调用模块 (use_demo.c):** ```c #include <linux/init.h> #include <linux/module.h> extern int shared_var; // 导入变量 extern void demo_func();// 导入函数 static int __init mod_init(void) { printk("Shared value: %d\n", shared_var); demo_func(); return 0; } module_init(mod_init); MODULE_LICENSE("GPL"); ``` #### 5. 关键注意事项 1. **符号可见性**:导出的符号必须是非静态(non-static)的 2. **依赖顺序**:被调用模块需要先于调用模块加载 3. **版本控制**:使用`EXPORT_SYMBOL_GPL`限制仅GPL兼容模块可用[^4] 4. **命名冲突**:避免与内核已有符号重名 5. **内存安全**:确保导出符号的生命周期覆盖所有调用者 #### 6. 调试技巧 - 查看符号表:`cat /proc/kallsyms | grep <symbol>` - 模块依赖检查:`modinfo <module_name>` - 动态加载测试:`insmod`/`rmmod`顺序验证
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值