内核模块符号的导出

复制代码
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------ 这就是符号导出的实际作用。

相关推荐
珂玥c20 小时前
windows系统nfs挂载
运维·windows·ssh·remmina·cygwin
菜择贰20 小时前
在linux(wayland)中禁用键盘
linux·运维·chrome
oMcLin20 小时前
如何在 Manjaro Linux 上通过配置systemd服务管理,提升微服务架构的启动速度与资源效率
linux·微服务·架构
Kira Skyler20 小时前
bpftool -S 签名功能实现解析
linux
霸气十足+拼命+追梦少年20 小时前
服务器挂载U盘或硬盘
运维·服务器
慕容雪_20 小时前
运维笔记-笔记本无线网卡连接wifi,有线网卡连接内网
运维·内网·远程
小杰帅气21 小时前
进程优先级与切换调度
linux·运维·服务器
码农学院21 小时前
使用腾讯翻译文本
服务器·数据库·c#
方便面不加香菜21 小时前
Linux基本指令(1)
linux
华纳云IDC服务商21 小时前
DNS负载均衡能自动避开故障服务器吗?
运维·服务器·负载均衡