linux驱动开发(一):一个最简单的内核驱动程序

一、编写.c程序

编写DriverFramework.c。该模块的功能很简单,就是在被内核加载时打印“hello init”,被内核卸载时打印“hello exit”。

#include <linux/init.h>
#include <linux/module.h>

MODULE_LICENSE("GPL");
MODULE_AUTHOR("zz"); //作者名称,可以随意命名

static int hello_init(void)
{
	printk("hello init\n");
	return 0;
}

static void hello_exit(void)
{
	printk("hello exit\n");
	return;
}

module_init(hello_init);
module_exit(hello_exit);

我们通常接触的经典的c程序都是应用程序,运行在用户空间下,然而本例是驱动程序,运行在内核空间下。

当我们在终端输入insmod指令时,会调用module_init,它和module_exit都是linux/init.h中定义的宏。module_init会将内核模块加载函数声明为hello_init(函数名可以随意定义)。hello_init函数的功能是在内核的log中打印“hello init”。

同理,在终端输入rmmod指令时,会调用module_exit宏,module_exit会将内核模块卸载函数声明为hello_exit函数,该函数的功能是在内核的log中打印“hello exit”。

宏MODULE_LICENSE用来声明此模块的许可证,此模块符合GPL协议。否则在加载此模块时,会收到内核被污染 “kernel tainted” 的警告。

宏MODULE_AUTHOR用来声明作者。

printk函数是运行在内核态下的打印函数,而printf函数运行在用户态。

二、编写Makefile

ifneq ($(KERNELRELEASE),)
	obj-m:=DriverFramework.o
else
	KDIR	:= /lib/modules/$(shell uname -r)/build
	PWD		:= $(shell pwd)
all:
	make -C $(KDIR) M=$(PWD) modules
clean:
	make -C $(KDIR) M=$(PWD) clean
endif

解释一下Makefile文件。KERNELRELEASE是在内核源码的上层Makefile中定义的一个变量。在该模块未被动到内核源代码中时,这个变量不会被定义。该模块移动到内核源代码后,这个变量就有定义了。

KDIR变量表示的这个目录下存放该版本linux内核源码,其中调用的shell指令uname -r用来打印该内核的版本号。PWD是当前目录所在的路径。

在终端输入make指令后,对.c文件编译进行编译。这个过程比较复杂。首先,初次编译前变量KERNELRELEASE为空,因此执行else后面的程序,即

make -C $(KDIR) M=$(PWD) modules

-C参数的作用是指定跳转目录,-C $(KDIR)指明跳转到内核源码所在的目录并读取那里的Makefile,启动kbuild机制。M=$(PWD)再返回到当前目录继续执行当前的Makefile。

kbuild即kernel build,用于编译Linux内核文件,对Makefile进行功能上的扩展。大部分内核中的Makefile都使用kbuild进行组织,它能使原本的Makefile代码变得更简洁、高效。kbuild中会预定义一些变量,如obj-y、obj-m,用来指定要生成的.o目标文件。只需要对该变量进行赋值,kbuild就会自动把代码编译到内核或编译成模块。

通常,内核驱动有两种编译和加载方式:第一种是直接把驱动程序编译进内核中,对应obj-y变量;第二种是将驱动程序作为模块单独编译成.ko文件,而不编译进内核中,然后手动加载,即obj-m变量。本例中将DriverFramework.o赋值给obj-m变量,就是采用第二种模式,单独生成一个独立的DriverFramework.ko文件。

三、模块的编译和加载

首先使用make编译模块。

接下来,我们介绍在终端需要输入的指令。首先加载驱动模块,指令为

sudo insmod DriverFramework.ko

注意是只有在root权限下才可以执行insmod指令。

打印内核信息。

dmesg

如果觉得内核打印的信息太多,可以使用dmesg -c清空一下内核log再加载模块。

可以看到打印了hello init。

使用lsmod可以查看所有被加载的模块。当然,也可以使用下面的指令只查看我们关注的模块。

lsmod | grep DriverFramework

移除内核模块的指令

sudo rmmod DriverFramework

查看内核打印信息。可以看到打印了hello exit。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值