一 EXPORT_SYMBOL
是 Linux 内核编程中的一个宏定义,用于将一个函数、变量或符号(symbol)导出给其他内核模块使用。这样,其他内核模块就可以通过外部引用(extern)来访问和使用这些被导出的符号,无论它们是否位于同一个源文件或源代码树中。
EXPORT_SYMBOL
的主要作用是将被导出的符号添加到内核模块的全局符号表中。这是 Linux 内核提供的一种机制,用于实现内核模块之间的符号共享和交互。通过导出符号,内核模块可以相互调用函数、访问共享变量,从而实现模块间的功能扩展和协同工作。
使用 EXPORT_SYMBOL
的基本语法如下:
c复制代码
EXPORT_SYMBOL(symbol); |
其中 symbol
是要导出的符号的名称。一旦一个符号被导出,其他内核模块就可以使用 extern
关键字来引用它。
值得注意的是,EXPORT_SYMBOL
主要出现在 Linux 2.6 内核及以后的版本中。在 2.4 内核中,非静态(non-static)的函数和变量默认会自动导入到内核空间,因此不需要使用 EXPORT_SYMBOL
进行标记。但在 2.6 内核及以后版本中,由于默认不再导出所有符号,因此需要使用 EXPORT_SYMBOL
来显式地导出所需的符号。
此外,在使用 EXPORT_SYMBOL
时,需要确保符号的导出和引用在模块加载和初始化的正确顺序下进行。通常,先加载定义该符号的模块,然后再加载引用该符号的模块。
通过 EXPORT_SYMBOL
,内核开发者可以更加灵活地组织和管理内核模块,实现模块间的松耦合和可扩展性。这对于构建复杂、可维护的 Linux 内核系统非常重要。
二 EXPORT_SYMBOL
是 Linux 内核编程中用于将一个函数或变量导出到内核符号表,使得其他内核模块能够调用或使用该符号的机制。下面是一个简单的使用示例:
假设我们有一个内核模块 mod1.c
,它定义了一个函数 add_integer
,我们想要让其他模块能够调用这个函数。
c复制代码
// mod1.c | |
#include <linux/init.h> | |
#include <linux/module.h> | |
// 定义一个可以被其他模块调用的函数 | |
int add_integer(int a, int b) { | |
return a + b; | |
} | |
EXPORT_SYMBOL(add_integer); // 导出这个函数到其他模块 | |
static int __init mod1_init(void) { | |
// 模块的初始化代码 | |
return 0; | |
} | |
static void __exit mod1_exit(void) { | |
// 模块的退出清理代码 | |
} | |
module_init(mod1_init); | |
module_exit(mod1_exit); | |
MODULE_LICENSE("GPL"); // 声明模块的许可权 |
在另一个模块 mod2.c
中,我们可以调用 mod1.c
导出的 add_integer
函数:
c复制代码
// mod2.c | |
#include <linux/init.h> | |
#include <linux/module.h> | |
#include <linux/kernel.h> // 用于printk等函数 | |
extern int add_integer(int, int); // 声明在其他模块中定义的函数 | |
static int __init mod2_init(void) { | |
int result = add_integer(2, 3); // 调用mod1中导出的函数 | |
printk(KERN_INFO "Result of add_integer(2, 3): %d\n", result); | |
return 0; | |
} | |
static void __exit mod2_exit(void) { | |
// 模块的退出清理代码 | |
} | |
module_init(mod2_init); | |
module_exit(mod2_exit); | |
MODULE_LICENSE("GPL"); // 声明模块的许可权 |
在这个例子中,mod1.c
通过 EXPORT_SYMBOL(add_integer)
导出了 add_integer
函数,使得 mod2.c
可以通过 extern int add_integer(int, int);
声明并调用它。在编译和加载这些模块时,你需要确保 mod1
先被加载,然后才是 mod2
,因为 mod2
依赖于 mod1
导出的函数。
请注意,使用 EXPORT_SYMBOL
导出的符号需要是全局的(非静态的),并且导出符号的模块和调用它的模块需要有兼容的许可权(例如,都是 GPL)。此外,当编译调用导出函数的模块时,可能会出现警告信息,这通常是因为在编译时还没有和内核符号表进行连接,但在加载模块时,内核会尝试解决这些符号。如果加载时仍然找不到符号,则模块加载会失败。