内核模块编译
实验原理
Linux模块是一些可以作为独立程序来编译的函数和数据类型的集合。之所以提供模块机制,是因为Linux本身是一个单内核。单内核由于所有内容都集成在一起,效率很高,但可扩展性和可维护性相对较差,模块机制可弥补这一缺陷。
Linux模块可以通过静态或动态的方法加载到内核空间,静态加载是指在内核启动过程中加载;动态加载是指在内核运行的过程中随时加载。
一个模块被加载到内核中时,就成为内核代码的一部分。模块加载入系统时,系统修改内核中的符号表,将新加载的模块提供的资源和符号添加到内核符号表中,以便模块间的通信。
编写模块代码
模块构造函数:
执行insmod或modprobe指令加载内核模块时会调用的初始化函数。函数原型必须是module_init(),括号内是函数指针
模块析构函数:
执行rmmod指令卸载模块时调用的函数。函数原型是module_exit()
模块许可声明:
函数原型是MODULE_LICENSE(),告诉内核该程序使用的许可证,不然在加载时它会提示该模块污染内核。一般会写GPL。
编写
代码
#include<linux/init.h>
#include<linux/module.h>
MODULE_LICENSE("Dual BSD/GPL");
static char *name="xiaoziyu";
static int __init name_init(void)
{
printk("==Hello 20199327==
");
printk("==Hello %s==
",name);
return 0;
}
static void __exit name_exit(void)
{
printk(KERN_INFO"Name module exit
");
}
module_init(name_init);
module_exit(name_exit);
module_param(name,charp,S_IRUGO);
Makefile代码
obj-m:=printname.o
CURRENT_PATH:=$(shell pwd)
LINUX_KERNEL_PATH:=/home/huahua/linux-5.0.1
all:
make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) modules
clean:
make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) clean
如果内核模块只有一个源文件,一般模块名称和原文件一样,只是把.c改成.o
obj-m := helloworld.o