目的:建立一个简单的内核模块,并把它挂载到内核中去。这个内核模块没有与外界有任何的数据交互。
/
//hello.c
/
//定义__KERNEL__表示此源文件可以看到内核源文件的所有内容
//因为有的内核的头文件会被用户空间的应用程序包含,但是内
//核中的内容有些是内核专用的,需要对用户隐藏起来
#ifndef __KERNEL__
#define __KERNEL__
#endif
//如果要把代码编译成一个内核模块,必须定义它
#ifndef MODULE
#define MODULE
#endif
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
//一些模块说明宏,这里没有全部列出
MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("A simple hello module");
MODULE_AUTHOR("KCB");
static int year = 2013;
static int __init hello_init(void) {
printk(KERN_DEBUG "Hello kernel, it's %d!\n", year);
return 0;
}
static void __exit hello_exit(void) {
printk(KERN_DEBUG "Bye, kernel!\n");
}
//module_init, module_exit是宏,他们分别表示模块加载和卸载时的
//执行函数,如hello_init, hello_exit.
//当我们注释掉这两个宏的时候,编译时会发出警告,这两个函数被定义了
//但是没有被使用: warning: 'hello_init' defined but not used
module_init(hello_init);
module_exit(hello_exit);
//makefile
obj-m := hello.o
all:
$(MAKE) -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules
clean:
$(MAKE) -C /lib/modules/$(shell uname -r)/build M=$(PWD) clean
本人的机器环境是虚拟机中Ubuntu11.10
在文件保存的路径下执行make
$make
将模块挂载到内核中去
$sudo insmod hello.ko
我们检查系统中是否挂载了我们的模块,前面提到过当模块挂载时,我们的hello.c中被执行的函数hello_init,这个函数的功能就是
打印Hello kernel, it's 2013, 我们现在检查是否打印了这条语句
$dmesg | tail
最后一条消息告诉我们我们的hello模块挂载到了内核中。
卸载模块hello
$sudo rmmod hello
$dmesg | tail
Bye, kernel!
以上这些都是作者学习他人文章后的总结,当然也有自己的理解。由于知识有限,理解会有偏差,欢迎大家指正。
最后,作者对上述的makefile文件还是不够熟悉,没有对其进行注释,希望了解的人可以给个链接,共同学习。
本文是参照Linux程序设计第3版的18章所写,如有疑问可以详细参考该书。