《块设备驱动程序介绍.docx》由会员分享,可在线阅读,更多相关《块设备驱动程序介绍.docx(14页珍藏版)》请在三一办公上搜索。
1、 块设备驱动程序 Linux0.11内核包含3个块设备的驱动:ramdisk,硬盘,软盘。首先需要注意的块设备的操作方式:在系统(内核)与硬盘进行IO操作时,需要考虑三个对象之间的交互作用。它们是系统、控制器和驱动器(例如硬盘或软盘驱动器)。 系统可以直接向控制器发送命令或等待控制器发出中断请求:控制器在接受到命令后就会控制驱动器的操作,读/写数据或者进行其他操作。因此我们可以把这里控制器发出的中断信号看作是这三者之间的同步操作信号,所经历的操作步骤为: 首先系统指明控制器在执行命令结束而引发的中断过程中应该调用的C函数,然后向块设备控制器发送读、写、复位或其他操作命令。当控制器完成了指定的命
2、令,会发出中断请求信号,引发系统执行块设备的中断处理过程,并在其中调用指定的C函数对读/写或其他命令进行命令结束后的处理工作。块设备表项和请求项数组构成的数据结构是块设备部分的核心数据结构。请求队列数组与块设备结构为如下关系请求项数组存放所有系统中对块设备的请求,原因估计是块设备一般都是低速外设,所以处理时间较长,来不及的请求先存储起来。每个块设备表项的current_request指向当前需要处理的请求项数组中的请求项,在请求项数组中的请求项之间用指针相链接,这样就构成了对某一设备的请求队列。下面分析ll_rw_blk.c程序,它是块设备(软硬盘,ramdisk)与缓冲管理之间的接口程序,主
3、要功能是为块设备创建块设备读写请求项,并插入到指定块设备请求队列中。实际的读写操作由请求项处理函数request_fn完成。request_fn通过blk.h的宏定义映射到实际设备的操作(do_hd_request,do_fd_request,do_rd_request)ll_rw_block调用序列如上图,ll_rw_block程序为左半边图,作用是往请求项链表中插入请求项,插完后调用request_fn()会引发中断逐个处理某设备的所有请求项。插入请求项是利用电梯算法插入到对应某个设备的请求链表中。 对于硬盘,首先涉及的是硬盘的知识,0.11内核支持的是AT硬盘。硬盘的分区结构如下: 与代
4、码联系较紧密的是主引导记录的分区表,内核中partition结构完全对应了分区表的字段。硬盘参数及类型(Linux0.11只支持最多两个硬盘)hd_info 01hd_i_struct磁头数每磁道扇区数柱面数写前预补偿柱面号磁头着陆柱面号控制字节硬盘分区结构(0,5代表第1,第2个硬盘,14,69为两硬盘的分区)0123456789hd_struct 物理起始扇区号分区扇区数sys_setup函数在系统初始化时被调用,以下为其步骤:从0x90080读入setup.s创建的硬盘参数表设置硬盘起始扇区号和扇区总数通过读取CMOS信息的方法重新确定系统中硬盘数和扇区信息通过bread读取硬盘主引导记
5、录分区表读入每个硬盘分区信息加载(创建)RAMDISK rd_load()安装根文件系统 mount_root()硬盘操作的核心函数是do_hd_request函数,hd.c中其余函数均为其服务。以下为do_hd_request流程检查请求合法性换算硬盘的扇区号、柱面号和磁道号若要执行复位硬盘,则复位硬盘若重新校正标志置位,则重新校正硬盘若当前请求是写扇区命令,循环读取状态寄存器看是否就绪若规定时间(次数)内读取成功,发送1个扇区数据到数据寄存器,否则写盘失败,去处理下一个硬盘请求若当前请求是读扇区命令,发送读扇区命令详见p137解释和p138图6-3下面分析硬盘驱动程序linux/kerne
6、l/blk_drv/hd.c该文件内容分两部分1.硬盘初始化 2. 硬盘驱动以下关注硬盘驱动的实现。其主流程是do_hd_request,由于该流程也在中断中被调用,所以该函数被反复调用,用来处理硬盘复位、重校、读、写的请求。下面分三部分对该函数拆解:1. 复位、重校统称为错误处理(因为复位流程保含了重校流程) 2. 写盘 3. 读盘错误流程处理:1. 若reset置1,置重校标志。2. 诊断复位硬盘控制器3. 向硬盘控制器发送命令块(重建参数)4. 中断指针do_hd指向硬盘重校函数时间监测硬盘执行命令后的状态,1. 若出错看是否超出最多允许出错数,则结束对该请求项的处理,若出错数大于最多允
7、许错误数一半,则置reset=12. 不出错则继续处理硬盘请求do_hd_requestCPU硬盘控制器复位硬盘控制器中断过程对于结束对该请求项的处理见blk.h p136 end_request函数,执行流程为:关闭设备置当前处理中的请求设备对应缓冲区更新标志若更新标志为0,显示出错信息唤醒等待该请求项的进程,唤醒等待请求的进程释放该请求项,指向下一请求项 错误处理流程属于辅助流程,每次处理读、写请求都可能会经历,是保证操作成功的必要步骤。 中断过程是由硬盘控制器触发的中断引起的,中断会传入8259中断控制器再传给CPU,操作系统会调用do_hd指向的函数作为中断过程执行。因此程序中把do_
8、hd赋值给某函数指针,等到硬盘控制器中断被CPU接收后,中断过程就“自动”执行。至于如何“自动”,即中断实现原理,将在进程调度与系统调用一章讨论。读写硬盘的流程大致如下:写盘流程处理1.发送写命令及扇区等命令,中断指令do_hd同时指向写扇区中断函数2. 循环读取状态寄存器信息并判断请求服务标志DRQ_STAT是否置位3. 若循环结束仍未置位,执行错误处理流程。否则进行硬盘第一次写数据时间CPU硬盘控制器1. 若硬盘控制器返回错误信息,执行错误处理流程并返回2. 否则置中断函数指针do_hd为写硬盘函数,并向数据寄存器端口写256字即512B,一个扇区3. 若全部扇区数据已写完,处理请求结束事
9、宜,执行下一硬盘请求硬盘控制器执行写一个扇区数据硬盘控制器执行写一个扇区数据中断中断过程读盘流程处理1.发送读命令及扇区等命令,中断指令do_hd同时指向写扇区中断函数时间CPU硬盘控制器1. 若硬盘控制器返回错误信息,执行错误处理流程并返回2. 将数据从数据寄存器口读到请求结构缓冲3. 若所需读出数据还未读完,则置中断指针do_hd为读硬盘函数并返回,否则处理请求结束事宜,执行下一硬盘请求硬盘控制器执行读一个扇区数据中断中断过程可以看出,写盘、读盘流程差别不大,区别就在于写盘要先读状态寄存器信息并进行第一次写盘。下面分析软盘驱动程序linux/kernel/blk_drv/floppy.c,
10、其主流程同样是do_fd_request,与硬盘驱动最大的不同是用到了一些定时操作,详见p157描述,do_fd_request同样分为1. 复位、重校统称错误处理2. 写盘3. 读盘 三部分错误处理流程1. 关中断2. 置重置软盘操作后的中断调用函数,即do_floppy指向reset_interrupt3. 延迟等待后启动软盘控制器4. 开中断时间CPU硬盘控制器1. 检测中断状态,读取命令执行结果字节2. 发送设定软驱参数命令启动软盘控制器中断中断过程复位1. 当前磁道号归零2. 置校正软盘操作后的中断调用函数,即do_floppy指向recal_interrrupt3. 发送校正命令及
11、参数:磁道号和驱动器号4. 如复位标志被置位,继续执行复位流程1. 检测中断状态,读取命令执行结果字节,若异常结束,复位标志置位2. 返回do_fd_request中断过程校正设定软盘控制器参数校正软盘控制器中断读盘、写盘操作仅有细微差别,并在一起分析:1. 根据请求项,置软盘读写命令码2. 添加定时器,指定驱动器到能正常运行需要的时间(秒),当定时时间到时调用函数floppy_on_interrupt时间CPU硬盘控制器1. 置已选择当前驱动器标志2. 设置数字输出寄存器DOR为当前驱动器3. 执行读写传输函数transfer()定时中断中断过程do_fd_request1. 设置当前驱动器
12、参数,发送命令和参数2. 设置指定驱动器的数据传输速率3. 若复位标志被置位,执行错误处理流程并返回4. 若不需要寻道,执行setup_rw_floppy,否则发送磁头寻道命令及参数并置软盘中断指针do_floppy为seek_interrupt1. 检测中断状态命令,若出错执行错误处理流程否则设置当前磁道为寻道号2. 执行setup_rw_floppy中断过程transfer设置当前驱动器设置驱动器参数中断floppy_on_interruptseek_interrrupt1. 初始化软盘DMA通道2. 置软盘中断指针do_floppy为rw_interrupt3. 发送读写命令及扇区等参数setup_rw_floppy1. 若软盘写保护,结束请求项2. 若为其他错误,执行错误处理流程3. 因DMA的寻址为1MB空间,对读操作作处理4. 释放软盘,结束请求项中断过程rw_interrrupt设置驱动器传输速率执行寻道操作根据读写命令在软盘与内存缓冲区之间传数据中断