linux字符设备驱动

Linux字符设备驱动

 

Linux设备驱动有两种加载方式,动态加载和静态加载。动态加载就是在运行的时候去加载,静态加载就是在编译的时候去加载。

这里主要记录学习动态加载的方式,在学习动态加载方式之前,要先熟悉驱动动态加载的流程,动态加载需要做哪些工作。

 

模块加载:

 

Linux内核模块主要由以下几个部分组成:

模块加载函数,模块卸载函数,模块许可证明,这三项是必须的。

模块参数,模块导出符号,模块作者信息等声明,这三项是可选项,可有可无。

 

 

为了清晰的了解模块加载和卸载的流程,这里以大家常用的hello word来说明。

 

 

#include <linux/init.h>

#inlcude <linux/module.h>

 

static int hello_init(void)

{

         printk("hello_init\n");

         return0;

}

 

static int hello_exit(void)

{

         printk("hello_exit\n");

         return0;

}

 

module_init(hello_init);

module_exit(hello_exit);

 

这个简单的函数只包含模块加载函数和模块卸载函数。编译完成之后会在相应的目录下面产生hello.ko文件,通过执行insmod ./hello.ko命令来加载这个模块,通过rmmod hello命令来卸载这个模块,注意卸载的时候没有.ko。在执行加载和卸载命令时,会有相应的打印信息出现,来直观的看到模块加载和卸载的过程。Lsmod命令可以查看当前已经加载的模块的信息。实际上lsmod命令解析的是/proc/modules文件,

 

有的教材上说加载和卸载函数一定要用__init__  和__exit__两个宏,我在2.6内核上面用了这里两个宏,没出现什么问题,也有的教材上说部需要,待验证。

 

内核中已经加载的信息模块存在sys/module 目录下,加载hello.ko之后,在这个目录下会自动生成一个hello目录。

 

Linux内核模块的组成和动态加载流程搞清楚之后,就以一个简单的字符设备驱动为例进行进一步的学习。

 

Linux字符设备驱动的结构

2.6内核中使用cdev结构体描述字符设备,cdev结构体如下所示:

struct cdev {

         structkobject kobj; //内嵌的kobject对象

         structmodule *owner; //所属模块

         conststruct file_operations *ops; //文件操作结构体,这个是关键

         structlist_head list;

         dev_tdev; //设备号

         unsignedint count;//引用次数

};

dev_t: 设备号,32位宽,高12位为主设备号,低20位为次设备号。

file_operations:提供了字符设备驱动提供给虚拟文件系统的接口函数。

关于cdev的操作函数主要由三个:

Cdev_init,cdev_add,cdev_del。

Cdev_init:初始化cdev的成员,并建立cdev与file_operations之间的连接。

cdev_add,cdev_del分别向系统添加和删除一个cdev设备。cdev_add一般放在字符设备驱动加载的函数中,cdev_del一般放在字符设备模块卸载的函数中。

 

设备号的分配和释放:

 

在调用cdev_add()函数向系统注册字符设备之前,应首先申请设备号。申请设备号有两种方法:

动态申请设备号,调用alloc_chrdev_region()。

当已知起始设备的设备号的时候,可以调用register_chrdev_region()。

不过,为了避免冲突,一般是动态申请,由系统来自动分配设备号。

在调用cdev_del()函数注销字符设备之后,要调用unregister_chrdev_region()释放原先申请的设备号。

File_operations结构体:

File_operations结构体的成员函数是字符设备驱动设计的主体内容。

 

Globalmem虚拟设备实例描述


参考资料:
《linux设备驱动开发详解》宋宝华 编著

你可能感兴趣的:(linux,list,struct,File,Module,linux内核)