一、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()等。
七、设备的轮询和中断处理:
对于不支持中断的设备,读写时需要轮询设备状态,以及是否需要继续进行数据传输。例如,打印机。如果设备支持中断,则可按照中断方式进行。