一.创建函数文件和Makefile文件
$sudo mkdir qiliguala
$cd qiliguala
$vim helloworld.c
在helloworld.c下输入以下内容
#include <linux/init.h> //模块的初始化的宏定义 以及一些其他函数的初始化函数
#include <linux/kernel.h> //包含了printk函数
#include <linux/module.h> 动态的将模块加载到内核中去
static int __init lkm_init(void)
{
printk("Hello World\n");
return 0;
}
static void __exit lkm_exit(void)
{
printk("Goodbye\n");
}
module_init(lkm_init);
module_exit(lkm_exit);
MODULE_LICENSE("GPL"); //GPL许可证
include包含就不详细说了,很重要,必须要有。
static int __init lkm_init(void)中 __init 比较有意思,百度给的解释是(宏定义__init,用于告诉编译器相关函数或变量仅用于初始化。编译器将标__init的所有代码存在特殊的内存段中,初始化结束后就释放这段内存。)
我的理解就是释放内存
module_init(lkm_init);就是对函数进行声明
MODULE_LICENSE(“GPL”);模块驱动声明
$vim Makefile
输入以下内容
obj-m:=helloworld.o
CURRENT_PATH:=$(shell pwd)
LINUX_KERNEL:=$(shell uname -r)
LINUX_KERNEL_PATH:=/usr/src/linux-headers-$(LINUX_KERNEL)
all:
make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) modules
clean:
make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) clean
第一行obj-m:=helloworld.o,我认为就是生产.o文件
CURRENT_PATH指当前地址
LINUX_KERNEL指内核版本号
LINUX_KERNEL_PATH内核地址
all:
make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) modules
clean:
make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) clean
以上两句就是在内核中生成的文件
二.编译
输入
$make
进行编译
输入ls,发现生成了很多文件
此时并没有在终端显示信息,因为这里内核和终端没有联系,内核不将信息打印到终端上
三.打印
我们输入
lsmod
出现helloworld就说明我们已经将helloworld这个模块插入到内核中去了
再输入
$dmesg
就可以发现打印了
删除,输入
$sudo rmmod helloworld
$lsmod
会发现helloworld模块已经从内核中删除
再输入
$dmesg
此时打印出goodbye