1.最简单的内核模块hello.c
#include <linux/init.h>
#include <linux/module.h>
MODULE_LICENSE("Dual BSD/GPL");//模块许可证说明,不加此句,会提示内核污染的警告,
static int hello_init(void)
{
printk("<0>" "Hello World enter\n");//中间有空格,不能有逗号
return 0;
}
static int hello_exit(void)
{
printk("<0>" "Hello World exit\n");//使用内核空间的printk(),可定义输出级别
}
module_init(hello_init);
module_exit(hello_exit);
MODULE_AUTHOR("TONG");
MODULE_DESCRIPTION("lt is my first modules");
MODULE_ALIAS("a simplest module");
2.内核模块的Makefile
ifneq ($(KERNELRELEASE),)//如果$(KERNELRELEASE),不为空,执行obj-m:=hello.o
obj-m:=hello.o //以模块的方式将hello.o编译入内核
else
#KDIR := /lib/modules/`uname -r`/build
KDIR := /lib/modules/2.6.27.5-117.fc10.i686/build //-C后指定内核源代码所在目录,M=指定hello.c和Makefile所在目录
all:
make -C $(KDIR) M=$(PWD) modules //编译进内核的方法
clean:
rm -f *.ko *.o *.mod.o *.mod.c *.symvers
endif
3.加载及卸载内核
加载:insmod hello.ko
卸载:rmmod hello
4.查看内核加载信息
1)lsmod可获得系统加载的所有模块及模块间的依赖关系
2)lsmod命令实际上读取了/proc/modules文件
3)内核已加载模块信息也存在于/sys/module目录,每个模块,一个目录。
4)modprobe比insmod命令更强大,加载某模块时会同时加载该模块所依赖的其他模块
加载: modprobe hello.ko
卸载: modprobe -r hello
5)modinfo hello.ko可以获得模块的相关信息
5.导出符号
“/proc/kallsyms”对应着内核符号表,记录了符号以及符号所在的内存地址。
模块使用以下宏名导出符号到内核符号表
EXPORT_SYMBOL(符号名);
EXPORT_SYMBOL_GPL(符号名);
导出的符号可以被其他模块使用,使用前需要声明以下。
6.模块参数
模块参数是模块被加载的时候,可以被传递给它的值,它本身对应于模块内部的全局变量
module_param(num,int, S_IRUGO);
module_param(book_name,charp,S_IRUGO);