|簡體中文

比思論壇

 找回密碼
 按這成為會員
搜索



查看: 2220|回復: 0
打印 上一主題 下一主題

linux设备管理

[複製鏈接]

24

主題

0

好友

356

積分

中學生

Rank: 3Rank: 3

  • TA的每日心情
    慵懶
    2024-6-12 08:54
  • 簽到天數: 137 天

    [LV.7]常住居民III

    推廣值
    0
    貢獻值
    0
    金錢
    48
    威望
    356
    主題
    24
    樓主
    發表於 2015-9-29 16:31:24
    Linux的设备管理
    • Linux的设备管理的主要任务是控制设备完成输入输出操作,所以又称输入输出(I/O)子系统

    • 它的任务是把各种设备硬件的复杂物理特性的细节屏蔽起来,提供一个对各种不同设备使用统一方式进行操作的接口

    • Linux把设备看作是特殊的文件,系统通过处理文件的接口—虚拟文件系统VFS来管理和控制各种设备。


    §6.1 设备管理概述
    • Linux设备的分类

    • 设备被分为三类,块设备字符设备网络设备

    • 字符设备是以字符为单位输入输出数据的设备,一般不需要使用缓冲区直接对它进行读写

    • 块设备是以一定大小的数据块为单位输入输出数据的,一般要使用缓冲区在设备与内存之间传送数据。

    • 网络设备是通过通信网络传输数据的设备,一般指与通信网络连接的网络适配器(网卡)等。

    Linux使用套接口(socket)以文件I/O方式提供了对网络数据的访问。
    • 设备驱动程序

    • 系统对设备的控制和操作由设备驱动程序完成的。

    • 设备驱动程序是由设备服务子程序和中断处理程序组成。设备服务子程序包括了对设备进行各种操作的代码,中断处理子程序处理设备中断

    • 设备驱动程序的主要功能是:

    • 对设备进行初始化
    • 启动停止设备的运行
    • 把设备上的数据传送到内存
    • 把数据从内存传送到设备
    • 检测设备状态

    • 驱动程序是与设备相关的。

    • 驱动程序的代码由内核统一管理

    • 驱动程序在具有特权级的内核态下运行

    • 设备驱动程序是输入输出子系统的一部分。

    • 驱动程序是为某个进程服务的,其执行过程仍处在进程运行的过程中,即处于进程上下文中

    • 若驱动程序需要等待设备的某种状态,它将阻塞当前进程,把进程加入到该种设备的等待队列中。。

    • Linux的驱动程序分为两个基本类型:字符设备驱动程序块设备驱动程序

    三.设备的识别
    • 对设备的识别使用设备类型主设备号次设备号
    • 设备类型:字符设备还是块设备

    • 按照设备使用的驱动程序不同而赋予设备不同的主设备号。主设备号是与驱动程序一一对应的

    • 同时还使用次设备号来区分一种设备中的各个具体设备。次设备号用来区分使用同一个驱动程序的个体设备

    • 例如,系统中的块设备IDE硬盘的主设备号是8

    而多个SCSI硬盘及其各个分区分别赋予次设备号123…
    [root@localhost /]# ls /dev/sda* -l
    brw-r----- 1 root disk 8, 0 11-07 12:31
    /dev/sda
    brw-r----- 1 root disk 8, 1 11-07 12:31
    /dev/sda1
    brw-r----- 1 root disk 8, 2 11-07 12:31
    /dev/sda2

    四.设备文件
    • Linux设备管理的基本特点是把物理设备看成文件,采用处理文件的接口和系统调用来管理控制设备。

    • 从抽象的观点出发,Linux设备又称为设备文件

    • 设备文件也有文件名,设备文件名一般由两部分组成

    • 第一部分23个字符,表示设备的种类,如串口设备是cu,并口设备是lpIDE普通硬盘是hdSCIS硬盘是sd,软盘是fp等。

    • 第二部分通常是字母或数字,用于区分同种设备中的单个设备,如hdahdbhdc…分别表示第一块、第二块、第三块IED硬盘。而hda1hda2…表示第一块硬盘中的第一、第二个磁盘分区。

    • 设备文件一般置于/dev目录下,如/dev/hda2/dev/lp0等。

    • Linux使用虚拟文件系统VFS做为统一的操作接口处理文件设备

    • 与普通的目录和文件一样,每个设备也使用一个VFSinode来描述,其中包含着该种设备的主、次设备号

    • 设备的操作也是通过对文件操作的file_operations结构体来调用驱动程序的设备服务子程序

    • 例如,当进程要求从某个设备上输入数据时,由该设备的

    file_operations结构体得到服务子程序的操作函数入口,然后调用其中的read()函数完成数据输入操作。
    • 同样,使用file_operations中的open()close()write()

    分别完成对设备的启动停止设备运行,向设备输出数据的操作。

    §6.2 LinuxI/O控制
    LinuxI/O控制方式有三种:查询等待方式
    中断方式DMA(内存直接存取)方式.
    一.查询等待方式
    • 查询等待方式又称轮询方式polling mode)。

    • 对于不支持中断方式的机器只能采用这种方式来控制I/O过程,所以Linux中也配备了查询等待方式。

    • 例如,并行接口的驱动程序中默认的控制方式就是查询等待方式。

    • 如函数lp_char_polled()就是以查询等待方式向与并口连接的设备输出一个字符。

    static inline int lp_char_polled(char lpchar, int minor)
    {
    int status, wait = 0;
    unsigned long count = 0;
    struct lp_stats *stats;

    do { /* 查询等待循环 */
    status = LP_S(minor);
    count ++;
    if(need_resched)
    schedule();
    } while(!LP_READY(minor,status) && count < LP_CHAR(minor));

    if (count == LP_CHAR(minor)) { /* 超时退出 */
    return 0;
    }
    outb_p(lpchar, LP_B(minor)); /* 向设备输出字符 */
      .
       .
    二.中断方式
    • 硬件支持中断的情况下,驱动程序可以使用中断方式控制I/O过程。

    • I/O过程控制使用的中断是硬件中断,当某个设备需要服务时就向CPU发出一个中断脉冲信号CPU接收到信号后根据中断请求号IRQ启动中断服务例程

    • 在中断方式中,Linux设备管理的一个重要任务就是在CPU接收到中断请求后,能够执行该设备驱动程序的中断服务例程。

    • 为此,Linux设置了名字为irq_action中断例程描述符表

    static struct irqaction *irq_action[NR_IRQS+1];
    • NR_IRQS表示中断源的数目
    • irq_action[]是一个指向irqaction结构的指针数组,它指向的irqaction结构是各个设备中断服务例程的描述符

    struct irqaction {
    void (*handler)(int, void *, struct pt_regs *); /* 指向中断服务例程 */
    unsigned long flags; /* 中断标志 */
    unsigned long mask; /* 中断掩码 */
    void *dev_id; /*
    struct irqaction *next; /* 指向下一个描述符 */
    };
    • 在驱动程序初始化时,调用函数request_irq()建立该驱动程序

    irqaction结构体,并把它登记到irq_action[]数组中。
    • request_irq()函数的原型如下:

    int request_irq(unsigned int irq,
    void (*handler)(int, void *, struct pt_regs *),
    unsigned long irqflags,
    const char * devname,
    void *dev_id);
    • 参数irq是设备中断求号,在向irq_action[]数组登记时,它做为数组的下标

    • 把中断号为irqirqaction结构体的首地址写入irq_action[irq]。这样就把设备的中断请求号与该设备的服务例程联系在一起了。


    • CPU接收到中断请求后,根据中断号就可以通过irq_action[]找到该设备的中断服务例程


    §6.3 字符设备与块设备管理
    • Linux中,一个设备在使用之前必须向系统进行注册,设备注册是在设备初始化时完成的。

    一.字符设备管理
    • 在系统内核保持着一张字符设备注册表每种字符设备占用一个表项

    • 字符设备注册表是结构数组chrdevs[]

    #define MAX_CHRDEV 128
    static struct device_struct chrdevs[MAX_CHRDEV];
    • 注册表的表项是device_struct结构

    struct device_struct {
    const char * name; /* 指向设备名字符串 */
    struct file_operations * fops; /* 指向文件操作函数的指针 */
    };
    • 在字符设备注册表中,每个表项对应一种字符设备的驱动程序。所以字符设备注册表实质上是驱动程序的注册表

    • 使用同一个驱动程序的每种设备有一个唯一的主设备号。所以注册表的每个表项与一个主设备号对应

    • 在Linux中正是使用主设备号来对注册表数组进行索引, 即chrdevs[]数组的下标值就是主设备号

    • device_struct结构中有指向file_operations结构的指针f_opsfile_operations结构中的函数指针指向设备驱动程序的服务例程


    • 打开一个设备文件时,由主设备号就可以找到设备驱动程序


    二.块设备管理
    • 块设备在使用前也要向系统注册

    • 块设备注册在系统的块设备注册表,块设备注册表是结构数组blkdevs[]

    • 它的元素也是device_struct结构

    static struct device_struct blkdevs[MAX_BLKDEV]
    • 在块设备注册表中,每个表项对应一种块设备

    • 注册表blkdevs[]数组的的下标主设备号





    • 块设备是以块为单位传送数据的,设备与内存之间的数据传送必须经过缓冲

    • 当对设备读写时,首先把数据置于缓冲区内,应用程序需要的数据由系统在缓冲区内读写。

    • 只有在缓冲区内已没有要读的数据,或缓冲区已满而无写入的空间时,才启动设备控制器进行设备与缓冲区之间数据交换

    • 设备与缓冲区的数据交换是通过blk_dev[]数组实现的:

    struct blk_dev_struct blk_dev[MAX_BLKDEV];
    • 每个块设备对应数组中的一项,数组的下标值与主设备号对应。

    • 数组元素是blk_dev_struct结构

    struct blk_dev_struct {
    void (*request_fn)(void);
    struct request * current_request;
    struct request plug;
    struct tq_struct plug_tq;
    };
    request_fn :指向设备读写请求函数的指针
    current_request指向request结构的指针。
    当缓冲区需要与设备进行数据交换时
    缓冲机制就blk_dev_struct加入一个request结构
    每个request结构对应一个缓冲区对设备的读写请求
    在request结构中有一个指向缓冲区信息的指针
    由它决定缓冲区的位置和大小等

    重要聲明:本論壇是以即時上載留言的方式運作,比思論壇對所有留言的真實性、完整性及立場等,不負任何法律責任。而一切留言之言論只代表留言者個人意見,並非本網站之立場,讀者及用戶不應信賴內容,並應自行判斷內容之真實性。於有關情形下,讀者及用戶應尋求專業意見(如涉及醫療、法律或投資等問題)。 由於本論壇受到「即時上載留言」運作方式所規限,故不能完全監察所有留言,若讀者及用戶發現有留言出現問題,請聯絡我們比思論壇有權刪除任何留言及拒絕任何人士上載留言 (刪除前或不會作事先警告及通知 ),同時亦有不刪除留言的權利,如有任何爭議,管理員擁有最終的詮釋權。用戶切勿撰寫粗言穢語、誹謗、渲染色情暴力或人身攻擊的言論,敬請自律。本網站保留一切法律權利。

    手機版| 廣告聯繫

    GMT+8, 2024-11-23 17:54 , Processed in 0.019082 second(s), 23 queries , Gzip On.

    Powered by Discuz! X2.5

    © 2001-2012 Comsenz Inc.

    回頂部