内核模块编写示例

一、简介

linux支持编写内核模块,通过insmod命令插入,进一步丰富内核功能。

二、C文件

初步写一个c文件 hello_module.c:

// 内核模块相关的头文件
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>

// 加载函数
static int __init hello_init(void)
{
    printk("Hello World.\n");
    return 0;
}

// 卸载函数
static void __exit hello_exit(void)
{
    printk("Goodbye.\n");
}

// 模块调用
module_init(hello_init);
module_exit(hello_exit);
// 许可证
MODULE_LICENSE("GPL");

三、编译文件

内核模块编译,需要使用makefile文件,使用文件名Makefile,make命令会默认查找对应文件:

obj-m:=hello_module.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

makefile有固定的格式,东西较多,后续系统学习。

四、编译过程

1、将上面两个文件hello_module.c和Makefile放在同一个文件夹中,执行命令make:

root@ubuntu:/home/drivet_test# make
make -C /usr/src/linux-headers-4.15.0-47-generic M=/home/drivet_test modules
make[1]: Entering directory '/usr/src/linux-headers-4.15.0-47-generic'
  CC [M]  /home/drivet_test/hello_module.o
  Building modules, stage 2.
  MODPOST 1 modules
  CC      /home/drivet_test/hello_module.mod.o
  LD [M]  /home/drivet_test/hello_module.ko
make[1]: Leaving directory '/usr/src/linux-headers-4.15.0-47-generic'
root@ubuntu:/home/drivet_test# ll
total 160
drwxr-xr-x 3 root root  4096 Dec 24 00:27 ./
drwxr-xr-x 4 root root  4096 Dec 23 23:38 ../
-rw-r--r-- 1 root root 52895 Dec 24 00:27 .cache.mk
-rw-r--r-- 1 root root   339 Dec 24 00:13 hello_module.c
-rw-r--r-- 1 root root  3992 Dec 24 00:27 hello_module.ko
-rw-r--r-- 1 root root   245 Dec 24 00:27 .hello_module.ko.cmd
-rw-r--r-- 1 root root   596 Dec 24 00:27 hello_module.mod.c
-rw-r--r-- 1 root root  2584 Dec 24 00:27 hello_module.mod.o
-rw-r--r-- 1 root root 29776 Dec 24 00:27 .hello_module.mod.o.cmd
-rw-r--r-- 1 root root  3408 Dec 24 00:27 hello_module.o
-rw-r--r-- 1 root root 29668 Dec 24 00:27 .hello_module.o.cmd
-rw-r--r-- 1 root root   271 Dec 24 00:09 Makefile
-rw-r--r-- 1 root root    41 Dec 24 00:27 modules.order
-rw-r--r-- 1 root root     0 Dec 24 00:27 Module.symvers
drwxr-xr-x 2 root root  4096 Dec 24 00:27 .tmp_versions/
root@ubuntu:/home/drivet_test#

编译成功后,会生成hello_module.ko。

五、插入ko

执行命令:insmod hello_module.ko
可以通过lsmod命令查看ko是否插入成功,并在/var/log/kern.log日志中,查看ko加载过程的init函数是否执行:

root@ubuntu:/home/drivet_test# insmod hello_module.ko
root@ubuntu:/home/drivet_test#
root@ubuntu:/home/drivet_test# lsmod | grep hello
hello_module           16384  0
root@ubuntu:/home/drivet_test#
root@ubuntu:/home/drivet_test# tail /var/log/kern.log
Dec 24 00:32:53 ubuntu kernel: [ 3631.352417] Hello World. Life is short, we need passion.

六、删除ko

执行命令:rmmod hello_module.ko
同样可以通过lsmod查询ko是否存在,以及在/var/log/kern.log日志中,查看ko卸载过程的exit函数是否执行:

root@ubuntu:/home/drivet_test# rmmod hello_module.ko
root@ubuntu:/home/drivet_test#
root@ubuntu:/home/drivet_test# lsmod | grep hello
root@ubuntu:/home/drivet_test#
root@ubuntu:/home/drivet_test# tail /var/log/kern.log
Dec 24 00:32:53 ubuntu kernel: [ 3631.352417] Hello World. Life is short, we need passion.
Dec 24 00:36:52 ubuntu kernel: [ 3815.830910] Goodbye, honey.

posted @ 2021-12-24 00:39  Pangolin2  阅读(323)  评论(0编辑  收藏  举报