本文介绍一种,一个模块调动另一个模块中定义的函数的方法
在内核中,函数又叫做符号,为了便于理解,此处说成我们在C语言中熟悉的函数调用,回想一下C语言中的函数调用,如果我们需要在一个.C文件中调用另一个.C文件中的函数,很简单,我们只需要在头文件中声明一下即可。但是,这件事,在内核中模块间却要麻烦一些,下面听我细细道来。
在这之前,我先假定一个背景,以此来更形象得阐述这件事,现下有两个不同的模块,以及一个函数,模块2(hehe)要调用模块1(hello)中定义的mystrlen函数。
(1)在编写模块1的.c文件时,将所定义函数,用EXPORT_SYMBOL导出
EXPORT_SYMBOL(mystrlen);
(2)在编写模块2的.c文件时,在顶部声明一下是调用的外部函数
extern int mystrlen(char *);
(3)将模块1make一下, 模块1的makefile文件放在文章末尾,此处我们继续。make完成之后,加载一下模块1,然后使用下面这条命令查看一下
cat /proc/kallsyms | grep mystrlen
会发现
这说明在我们内核里的符号表里已经有这个函数了(/proc/kallsyms这个文件夹包含的是所有内核知道的符号)
(4)拷贝make模块1时所生成的Module.symvers文件到模块2所在的位置。
插一句,Module.symvers这个是什么东西?我们可以vim打开看一下,这个文件包含了内存位置、符号名称、模块所在位置。有了这三条信息,模块2就知道了去哪调用了,不然模块2怎么知道要调用谁,要去哪调用。
(5)把模块2也make一下
(6)在两个模块所在的目录下,分别执行sudo make modules_install,其中modules_install的具体实现在Makefile中有写,见文末。此举主要是将模块安装到根目录下的对应文件夹下,安装成功后我们可以查看一下有没有
(7)sudo depmod -a 建立起模块之间的依赖关系,因为,模块2需要调动模块1中的函数,所以需要这样来建立起依赖关系。
(8)sudo modprobe hehe 直接加载这个模块2,因为上一条命令已经将模块间的依赖建立,此处直接用modprobe就可以直接加载,不用再去用insmod一个个地去加载,那样不是很方便。
文末给出两个.c文件和Makefile文件
模块1
#include <linux/module.h>
#include <linux/init.h>
MODULE_LICENSE("GPL");
int mystrlen(char *p)
{
int i=0;
while(*p)
{
i++;
p++;
}
return i;
}
EXPORT_SYMBOL(mystrlen);
//int init_module()
static int __init hello_init(void)
{
printk("hello, init .....\n");
printk("lcd = %d\n",lcd);
return 0;
}
static void __exit hello_exit(void)
{
printk("hello,exit...\n");
}
module_init(hello_init); //指出当前模块的入口
module_exit(hello_exit);
模块2
#include <linux/module.h>
#include <linux/init.h>
extern int mystrlen(char *);
MODULE_LICENSE("GPL");
char *p="abcdefg";
//int init_module()
static int __init hello_init(void)
{
printk("hello, init .....\n");
printk("length of p is %d\n",mystrlen(p));
return 0;
}
static void __exit hello_exit(void)
{
printk("hello,exit...\n");
}
module_init(hello_init); //指出当前模块的入口
module_exit(hello_exit);
Makefile
ifeq ($(KERNELRELEASE),)
KERNELDIR ?= /lib/modules/$(shell uname -r)/build
#KERNELDIR ?=/home/farsight/linux-add-usb/linux-3.14
PWD := $(shell pwd)
modules:
$(MAKE) -C $(KERNELDIR) M=$(PWD) modules
modules_install:
$(MAKE) -C $(KERNELDIR) M=$(PWD) INSTALL_MOD_PATH= modules_install
else
obj-m := hello.o
endif
clean:
rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions Module* modules*