内核模块符号的导出

复制代码
grep -rn "EXPORT_SYMBOL" 

搜索EXPORT_SYSMBOL这个宏

这一节讲的是 **"内核模块之间如何共享函数 / 变量" 的核心机制 **,可以拆解为 3 个关键部分理解:

1. 为什么需要 "内核模块符号导出"?

内核模块编译后是独立的.ko文件,默认情况下:

  • 模块 A 里的函数 / 变量,模块 B 是 "看不见" 的,没法直接调用;
  • 但复杂驱动需要分层(比如把 "硬件控制" 和 "业务逻辑" 拆成两个模块),这时候就需要让模块 A 把自己的函数 / 变量 "暴露出来",供模块 B 调用 ------ 这个 "暴露" 的操作,就是 "符号导出"。

2. 什么是 "内核模块符号导出"?

简单说:在模块里,用特定宏把函数 / 变量标记为 "公共可用",当这个模块加载到内核时,这些函数 / 变量会被记录到内核公共符号表里,其他模块加载时就能从这个表中找到并调用它们。

3. 具体怎么实现(用到的工具)?

这一节里提到的两个宏,就是实现符号导出的核心:

作用
EXPORT_SYMBOL(sym) 导出符号(函数 / 变量,sym是要导出的名字),所有内核模块都能调用
EXPORT_SYMBOL_GPL(sym) 仅允许遵循 GPL 协议的内核模块调用(因为 Linux 内核是 GPL 协议的)

4. 额外注意点

  • 头文件不用单独引 :这两个宏定义在include/linux/export.h里,但内核模块的常用头文件module.h已经包含了export.h,所以写模块代码时,只要包含了#include <linux/module.h>,就可以直接用这两个宏。
  • 符号表的记录 :导出的符号会被记录到内核的Module.symvers文件中(之前提到的那个文件),其他模块编译时需要依赖这个文件,才能找到导出的符号。

举个简单例子(帮助理解)

比如你写了两个模块:

  • 模块 A(导出方)

    c

    复制代码
    #include <linux/module.h>
    // 定义一个要导出的函数
    void my_shared_func(void) {
        printk("这是模块A导出的函数\n");
    }
    // 导出这个函数,供其他模块调用
    EXPORT_SYMBOL(my_shared_func);
    
    // 模块加载/卸载函数(省略)
    module_init(xxx_init);
    module_exit(xxx_exit);
    MODULE_LICENSE("GPL");
  • 模块 B(调用方)

    c

    复制代码
    #include <linux/module.h>
    // 声明要调用的、模块A导出的函数
    extern void my_shared_func(void);
    
    static int __init b_init(void) {
        my_shared_func(); // 直接调用模块A导出的函数
        return 0;
    }
    module_init(b_init);
    MODULE_LICENSE("GPL");

当模块 A 先加载到内核,模块 B 再加载时,就能成功调用my_shared_func------ 这就是符号导出的实际作用。

相关推荐
十日十行10 小时前
Linux和window共享文件夹
linux
Sinclair14 小时前
简单几步,安卓手机秒变服务器,安装 CMS 程序
android·服务器
木心月转码ing18 小时前
WSL+Cpp开发环境配置
linux
Rockbean2 天前
用40行代码搭建自己的无服务器OCR
服务器·python·deepseek
蝎子莱莱爱打怪2 天前
Centos7中一键安装K8s集群以及Rancher安装记录
运维·后端·kubernetes
茶杯梦轩2 天前
CompletableFuture 在 项目实战 中 创建异步任务 的核心优势及使用场景
服务器·后端·面试
崔小汤呀2 天前
最全的docker安装笔记,包含CentOS和Ubuntu
linux·后端
何中应2 天前
vi编辑器使用
linux·后端·操作系统
何中应2 天前
Linux进程无法被kill
linux·后端·操作系统
何中应2 天前
rm-rf /命令操作介绍
linux·后端·操作系统