【驱动】——驱动入门用例
1 #include <linux/init.h> 2 #include <linux/module.h> 3 4 static int __init hello_init(void){ 5 printk("hello kernel!\n"); 6 return 0; 7 } 8 9 static void __exit hello_exit(void){ 10 printk("bye kernel!\n"); 11 } 12 module_init(hello_init); 13 module_exit(hello_exit); 14 15 MODULE_AUTHOR("ngnetboy"); 16 MODULE_DESCRIPTION("say hello to kernel!"); 17 MODULE_LICENSE("GPL"); 18 MODULE_VERSION("0.0.0.1");
实例代码
注:
__init 对内核来说是一种暗示,表明该函数仅在初始化期间使用。在模块被装载之后,模块装载器就会将初始化函数扔掉,这样可将该函数占用的内存释放出来,以作他用。
__exit 修饰词标记该代码仅用于模块卸载(编译器将把该函数放在特殊的 ELF 段中)。如果一个模块未定义清除函数,则内核不允许卸载该模块。
ifeq ($(KERNELRELEASE), ) KERNELDIR ?= /lib/modules/$(shell uname -r)/build PWD := $(shell pwd) all: clean $(MAKE) -C $(KERNELDIR) M=$(PWD) modules -rm -rf *.o *~ core .depend .*.cmd *.mod.c .tmp_versions Module.* Makefile.xen modules.order clean: -rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions Module.* Makefile.xen modules.order else obj-m := hello.o endif
Makefile 文件。
注:
-C 改变目录到指定的位置(内核源代码目录),其中保存有内核的顶层 makefile 文件。
M= 选项让该 makefile 在构造 modules 目标之前返回到模块源代码目录。
按照顺序分析以下make的执行步骤:
在模块的源代码目录下执行make,此时,宏“KERNELRELEASE”没有定义,因此进入else。由于make 后面没有目标,所以make会在Makefile中的第一个不是以.开头的目标作为默认的目标执行。于是 all 成为 make 的目标。 make 会执行 $(MAKE) -C $(KERNELDIR) M=$(PWD) modules 。 这个目标会第二次运行 make 命令。
最简单的驱动示例程序;