内核模块符号的导出

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

相关推荐
liebe1*12 小时前
第十一章 密码学
服务器·密码学·php
^乘风破浪^2 小时前
Ubuntu部署Xingrin(星环)企业级漏洞扫描与资产管理平台
linux·运维·ubuntu
LSL666_2 小时前
12 MyBatis的连接池
java·服务器·mybatis
2301_800256112 小时前
第十一章 PostgreSQL 服务器编程知识点梳理(1)
服务器·数据库·postgresql
C语言魔术师2 小时前
【linux】linux进程概念(四)(环境变量)
linux·运维·服务器
松涛和鸣2 小时前
DAY32 Linux Thread Programming
linux·运维·数据库·算法·list
eggrall2 小时前
《gdb 与 cgdb 深度解析:命令行调试的效率革命》
linux
源代码•宸2 小时前
分布式缓存-GO(简历写法、常见面试题)
服务器·开发语言·经验分享·分布式·后端·缓存·golang
秦jh_2 小时前
【Qt】常用控件(上)
服务器·数据库·qt