一个模块可以使用另一个模块导出的函数,可以通过函数EXPORT_SYMBOL(func_name)来导出,导出后的函数位于/proc/kallsyms文件中。
1. 导出代码:
#include <linux/init.h>
#include <linux/module.h>
MODULE_LICENSE("Dual BSD/GPL");
static int hello_init(void)
{
printk(KERN_INFO "Hello,world");
return 0;
}
static int hello_export(void) {
printk(KERN_INFO "Hello from another module");
return 0;
}
static void hello_exit(void)
{
printk(KERN_INFO "Goodbye!");
}
EXPORT_SYMBOL(hello_export);
module_init(hello_init);
module_exit(hello_exit);
创建Makefile文件如下:
obj-m := hello.o
KDIR := /lib/modules/`uname -r`/build
PWD := $(shell pwd)
default:
make -C $(KDIR) M=$(PWD) modules
clean:
rm -rf *.o .* .cmd *.ko *.mod.c .tmp_versions
查看导出的函数如下:
# cat /proc/kallsyms | grep hello
ffffffffc0725000 T hello_export [hello]
ffffffffc0725020 t hello_init [hello]
ffffffffc0725040 t hello_exit [hello]
可以看到函数被内核识别并导出,第一列表示地址,最后一列表示函数属于哪个模块。
也可以查看源码编译路径中的Module.symvers文件如下:
# cat Module.symvers
0x00000000 hello_export /root//module/hello EXPORT_SYMBOL
2. 导入使用
接下去我们看下如何使用这个hello模块导出的函数。使用过程需要通过extern关键字来声明函数是外部导入。
#include <linux/init.h>
#include <linux/module.h>
MODULE_LICENSE("Dual BSD/GPL");
extern int hello_export(void);
static int usehello_init(void)
{
printk(KERN_INFO "Hello usehello");
hello_export();
return 0;
}
static void usehello_exit(void)
{
printk(KERN_INFO "Goodbye usehello");
}
module_init(usehello_init);
module_exit(usehello_exit);
Makefile同上,不过模块名字需要修改一下。这个新模块就依赖于上一个提供函数的模块了,导入时候必须先导入上一个模块,不然这个新模块无法使用。