Linux内核模块是可动态加载到内核空间的程序,用于扩展内核功能(如驱动、文件系统等)而无需重新编译整个内核,编译内核模块是Linux系统开发的基础技能,以下是详细步骤和注意事项。

环境准备
在开始编译前,需确保系统具备必要的工具和依赖:
- 开发工具包:安装
build-essential(包含gcc、make等)和linux-headers(当前内核的头文件)。- Ubuntu/Debian:
sudo apt install build-essential linux-headers-$(uname -r) - CentOS/RHEL:
sudo yum groupinstall "Development Tools" kernel-devel
- Ubuntu/Debian:
- 内核源码(可选):若需修改内核或调试模块,可下载对应内核源码(
apt install linux-source或从kernel.org获取),并解压到/usr/src/,创建符号链接/usr/src/linux -> 内核源码目录。
步骤1:编写模块源码
以简单“Hello World”模块为例,创建hello.c文件:
#include <linux/init.h> // 模块初始化/清理宏
#include <linux/module.h> // 模块核心宏
#include <linux/kernel.h> // 内核功能(如printk)
// 模块初始化函数(加载时执行)
static int __init hello_init(void) {
printk(KERN_INFO "Hello, Linux module!n");
return 0; // 返回0表示成功
}
// 模块清理函数(卸载时执行)
static void __exit hello_exit(void) {
printk(KERN_INFO "Goodbye, Linux module!n");
}
// 注册初始化和清理函数
module_init(hello_init);
module_exit(hello_exit);
// 模块许可证(必须,否则内核加载时会警告)
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Your Name");
MODULE_DESCRIPTION("A simple Hello World module");
关键点:
__init和__exit:标记函数仅在初始化/清理阶段调用,优化内核空间。printk:内核日志函数,KERN_INFO为日志级别(可通过dmesg查看)。MODULE_LICENSE:必须声明为“GPL”或“GPL v2”,否则内核会标记模块“tainted”(不纯净)。
步骤2:编写Makefile
模块编译需通过Makefile管理,创建Makefile文件(与hello.c同目录):

obj-m += hello.o # 目标模块名(hello.ko),obj-m表示编译为可加载模块
all:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules
clean:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) clean
Makefile关键变量说明:
| 变量/命令 | 说明 |
|———–|——|
| obj-m | 指定模块目标(hello.o会编译为hello.ko) |
| uname -r| 获取当前内核版本(如5.15.0-88-generic) |
| -C | 切换到指定目录(内核源码目录) |
| M=$(PWD) | 返回模块源码目录($(PWD)为当前路径) |
| modules | 内核Makefile的目标,编译模块 |
步骤3:编译模块
在终端执行make命令:
make
编译成功后,当前目录会生成hello.ko(内核模块文件)、.mod.c、.mod.o等临时文件,若报错,检查:
- 是否安装对应内核的
linux-headers; - Makefile语法是否正确(如缩进必须为Tab,而非空格)。
步骤4:加载与测试模块
- 加载模块:
sudo insmod ./hello.ko # 需root权限
- 查看模块状态:
lsmod | grep hello # 检查模块是否加载 dmesg | tail -n 5 # 查看内核日志(应输出“Hello, Linux module!”)
- 卸载模块:
sudo rmmod hello # 模块名(不带.ko) dmesg | tail -n 5 # 确认输出“Goodbye, Linux module!”
常见问题处理
- 加载失败(“Invalid module format”):通常因内核版本与头文件不匹配,确保
linux-headers版本与uname -r一致。 - 权限问题:
insmod/rmmod需root权限,或配置/etc/modules-load.d/实现自动加载。 - 依赖缺失:若模块依赖其他内核功能(如
kmalloc),需在源码中包含对应头文件(linux/slab.h)。
相关问答FAQs
Q1:编译时报错“/lib/modules/$(uname -r)/build: No such file or directory”如何解决?
A:该错误表示系统未安装当前内核的开发头文件,解决方法:

- Ubuntu/Debian:
sudo apt install linux-headers-$(uname -r) - CentOS/RHEL:
sudo yum install kernel-devel-$(uname -r)
安装后重新执行make即可。
Q2:模块加载后,dmesg看不到输出日志,是什么原因?
A:可能原因有两个:
- 日志级别过高:
printk默认输出到/var/log/kern.log,可通过dmesg -w实时查看,或降低日志级别(如printk(KERN_DEBUG "Hello"))。 - 内核配置禁用了console日志:检查
/proc/sys/kernel/printk,确保首行数值不为0(如“4 4 1 7”),表示最低日志级别为4(KERN_WARNING)。
原创文章,发布者:酷番叔,转转请注明出处:https://cloud.kd.cn/ask/21538.html