高手进阶 Linux系统下驱动程序框架概述

来源:岁月联盟 编辑:zhuzhu 时间:2008-03-31
高手进阶 Linux系统下驱动程序框架概述内容简介:一、linux的设备驱动程序与外界的接口可以分为三个部分: 1.驱动程序与操作系统内核的接口。通过file_operations(include/linux/fs.h)数据结构来完成的。 2.驱动程序与系统引导的接口。这部分利用驱动程序对

一、linux的设备驱动程序与外界的接口可以分为三个部分:

 

1.驱动程序与操作系统内核的接口。通过file_operations(include/linux/fs.h)数据结构来完成的。

 

2.驱动程序与系统引导的接口。这部分利用驱动程序对设备进行初始化。

 

3.驱动程序与设备的接口。这部分描述了驱动程序如何与设备进行交互,与具体的设备密切相关。

 

二、根据功能划分,设备驱动程序的代码有以下几部分:

 

1.驱动程序的注册和注销。

 

2.设备的打开和释放。

 

3.设备的读写操作。

 

4.设备的控制操作。

 

5.设备的中断和轮询处理。

 

三、驱动程序的注册和注销:

 

设备驱动程序可以在系统启动的时候初始化,也可以在需要的时候动态加载。字符设备的初始化由chr_dev_init()完成,包括对内存(devfs_register_chrdev(MEM_MAJOR,"mem",&memory_fops)),终端(tty_init()),打印机(lp_init()),鼠标(misc_init())等字符设备的初始化。

 

块设备初始化由blk_dev_init()完成,这包括对IDE硬盘(ide_init()),软盘(floppy_init()),光驱等块设备的初始化。

 

每个字符设备或是块设备的初始化都是通过devfs_register_chrdev()或是devfs_register_blkdev()向内核注册。在关闭字符设备或是块设备时,还需要通过devfs_unregister_chrdev()或是devfs_unregister_blkdev()从内核中注销设备。

 

四、设备的打开和释放:

 

打开设备是由open()来完成的。例如,打印机是用lp_open()打开的,而硬盘是用hd_open()打开的。在大部分设备驱动程序中,open完成如下工作:

 

1.增加设备的是用计数。

 

2.检查设备的相关错误,如设备尚未准备好或是类似硬件的问题。

 

3.检查是首次打开,则初始化设备。

 

4.识别次设备号,如有必要则更新f_op指针。

 

5.如果需要,分配且设置要放在filp->private_data里的数据结构。

 

释放设备由release()来完成,例如释放打印机是用lp_release(),而释放终端设备是用tty_release()。释放设备的一般步骤包括:

 

1.释放在filp->private_data中的open分配的内存。

 

2.如果是最后一次释放,则关闭设备。

 

3.递减设别的使用计数。

 

五、设备的读写操作:

 

字符设备使用各自的read()和write()来进行数据读写。例如,对虚拟终端的读写是通过vcs_read()和vcs_write()来进行数据读写的。

 

块设备使用通用的generic_file_read()和generic_file_write()来进行数据读写。这两个通用函数向请求表添加读写请求,内核可以通过ll_rw_block()优化请求顺序。由于是对内存缓冲区而不是设备进行操作的,因而可以加快读写请求。如果内存缓冲区内没有要读入的数据或是要将写请求写入设备,那么就要真正的执行数据传输。这是通过数据结构request_queue和request_fn()来完成(include/linux/blkdev.h)。

 

六、设备的控制操作:

 

除了读写操作,有时还要控制设备。这可以通过设备驱动程序中的ioctl()来完成。例如IDE硬盘的控制可以通过hd_ioctl(),对光驱的控制可以通过cdrom_ioctl()。

 

与读写操作不同,ioctl()的用法与具体设备密切相关。以软驱的floppy_ioctl()为例(drivers/block/floppy.c):

 

static int fd_ioctl(struct inode *inode,

 

struct file *filp,

 

unsigned int cmd,

 

unsigned long param);

 

其中,cmd的取值及含义都是与软驱有关的,比如,FDEJECT表示弹出软盘。

 

除了ioctl(),设备驱动程序还可能有其他控制函数,比如llseek()等。

 

七、设备的轮询和中断处理:

 

对于不支持中断的设备,读写时需要轮询设备状态,以及是否需要继续进行数据传输。例如,打印机。如果设备支持中断,则可按照中断方式进行。