Linux设备驱动编程编程总结(一)-------模块编程
2019-07-13 08:15发布
生成海报
笔者搞设备驱动有一个多月了,也看了一些程序,前段时间一直很乱,没有办法总结,所以一直没有写文章,昨日热的睡不着,脑袋中却分外清晰,于是整理思路,将一个多月的学习心得总结出来,一方面供广大嵌入式Linux开发者参考,一方面稳固自己的知识。(PS:昨天还落枕了,现在脖子还疼)
我将分为5篇文章去总结,这是第一篇,因为Linux中的驱动都是以模块的方式加载到内核中的,所以学习模块编程必须成为第一步。
各种官方的介绍此处略过,我们先来看一个最简单的模块实例:helloworld!
#include
#include
#include
MODULE_LICENSE("GPL"); //申明LICENSE,不写编译会有警告,所以还是写了吧
/*-----------------------------------------------------------------------------
函数名: hello_init
参数: void
返回值: int
描述: 模块初始化函数,在安装模块时候执行
*-----------------------------------------------------------------------------*/
static int __init hello_init(void)
{
printk("hello world!--It is kernel speaking
"); //类似于printf,是在内核中使用的打印函数
return 0;
}
/*-----------------------------------------------------------------------------
函数名: hello_exit
参数: void
返回值: void
描述: 模块退出函数,在安装卸载时候执行
*-----------------------------------------------------------------------------*/
static void __exit hello_exit(void)
{
printk("Goodbye!Kernel
");
}
module_init(hello_init);
module_exit(hello_exit);
这个函数已经简单的和helloworld一样简单清楚和可爱了,我只做几点说明:
1.大家可以看到模块函数没有main函数,只有init函数,一般在模块编译好之后,会执行insmod命令,这是就会调用module_init函数中注册的初始化函数,也就是hello_init函数!同样在模块用完之后,我们通常会执行rmmod以移除模块,这时候就会调用module_exit中注册的函数,也就是 hello_exit 函数。
2.__init和__exit,(注意是两个下划线)这个是两个关键字,目的是告诉内核,在模块init的时候开辟内存,在模块退出的时候释放内存,如果不加这两个关键字,模块卸载的时候就不会释放内存,这就造成了内存的浪费。
Makefile:
有了程序,下面就要编译程序了,模块的编译方法和编译程序可不一样,下面提供一个Makefile模板,这个时代已经不是从无到有的用手敲代码了,而是在模板的基础上改动,这样才能提高效率:
obj-m:=hello.o
CURRENT_PATH :=$(shell pwd)
#VERSION_NUM :=$(shell uname -r)
#LINUX_PATH :=/usr/src/linux-headers-$(VERSION_NUM)
LINUX_PATH :=~/by700/linux-2.6.30-atmel9260
all :
make -C $(LINUX_PATH) M=$(CURRENT_PATH) modules
.PHONY :clean
clean:
rm -rf *.o *ko
我也做几点说明:
1.object -m :=hello.o,是要编译的模块的目标文件对应于hello.c,所以需要根据具体情况修改
2.对于这个模板我们第二个要修改的地方就是内核目录,大家看到我用#把两句话隐藏掉了,这个方便开启PC和目标 板之间的切换:
对于PC端,一般编译好的内核源代码就放在我#号隐掉的目录下,只要#号去掉,同时把第二个LINUX_PATH隐掉,就可以直接使用这个Makefile文件,这个不难,主要是目标板上的模块,我调试了两天才调试通
对于目标板(笔者的是at91)的芯片,LINUX_PATH这个目录就需要调整了,因为我们在PC机上编译目标板的内核代码由读者自己选定,我这个暂时就选定在如代码中的目录下。
特别注意,这个目录一定要和目标板烧写的镜像是一致的,否则会出现内核和模块不匹配的错误,笔者就是被这个错误改了内核代码,结果越来越多错误,最后请教高手才解决的
我在这里用的交叉编译环境是:arm-angstrom-linux-gnueabi-
所以在执行的make的时候,要执行如下:make ARCH=arm CROSS_COMPILE=arm-angstrom-linux-gnueabi-
结果如下:
root@at91sam9260ek:/mnt/hello# insmod hello.ko
hello world!--It is kernel speaking
root@at91sam9260ek:/mnt/hello# lsmod
Module Size Used by Not tainted
hello 1120 0
root@at91sam9260ek:/mnt/hello# rmmod hello
Goodbye!Kernel
root@at91sam9260ek:/mnt/hello# lsmod
Module Size Used by Not tainted
root@at91sam9260ek:/mnt/hello#
内核模块可以加载的文件是.ko后缀名的!
打开微信“扫一扫”,打开网页后点击屏幕右上角分享按钮