CentOS设备管理
时间:2014-05-13 18:44 来源:linux.it.net.cn 作者:it
Linux的设备管理
-
Linux的设备管理的主要任务是控制设备完成输入输出操作,所以又称输入输出(I/O)子系统。
-
它的任务是把各种设备硬件的复杂物理特性的细节屏蔽起来,提供一个对各种不同设备使用统一方式进行操作的接口。
-
Linux把设备看作是特殊的文件,系统通过处理文件的接口—虚拟文件系统VFS来管理和控制各种设备。
§6.1 设备管理概述
-
Linux设备的分类
-
设备被分为三类,块设备、字符设备和网络设备。
-
字符设备是以字符为单位输入输出数据的设备,一般不需要使用缓冲区而直接对它进行读写。
-
块设备是以一定大小的数据块为单位输入输出数据的,一般要使用缓冲区在设备与内存之间传送数据。
-
网络设备是通过通信网络传输数据的设备,一般指与通信网络连接的网络适配器(网卡)等。
Linux使用套接口(socket)以文件I/O方式提供了对网络数据的访问。
-
设备驱动程序
-
系统对设备的控制和操作是由设备驱动程序完成的。
-
设备驱动程序是由设备服务子程序和中断处理程序组成。设备服务子程序包括了对设备进行各种操作的代码,中断处理子程序处理设备中断。
-
设备驱动程序的主要功能是:
-
对设备进行初始化
-
启动或停止设备的运行
-
把设备上的数据传送到内存
-
把数据从内存传送到设备
-
检测设备状态
-
驱动程序是与设备相关的。
-
驱动程序的代码由内核统一管理,
-
驱动程序在具有特权级的内核态下运行。
-
设备驱动程序是输入输出子系统的一部分。
-
驱动程序是为某个进程服务的,其执行过程仍处在进程运行的过程中,即处于进程上下文中。
-
若驱动程序需要等待设备的某种状态,它将阻塞当前进程,把进程加入到该种设备的等待队列中。。
-
Linux的驱动程序分为两个基本类型:字符设备驱动程序和块设备驱动程序。
三.设备的识别
-
对设备的识别使用设备类型、主设备号、次设备号
-
设备类型:字符设备还是块设备。
-
按照设备使用的驱动程序不同而赋予设备不同的主设备号。主设备号是与驱动程序一一对应的,同时还使用次设备号来区分一种设备中的各个具体设备。次设备号用来区分使用同一个驱动程序的个体设备。
-
例如,系统中的块设备IDE硬盘的主设备号是8,
而多个SCSI硬盘及其各个分区分别赋予次设备号1、2、3…。
[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的设备又称为设备文件。
-
设备文件也有文件名,设备文件名一般由两部分组成
-
第一部分2~3个字符,表示设备的种类,如串口设备是cu,并口设备是lp,IDE普通硬盘是hd,SCIS硬盘是sd,软盘是fp等。
-
第二部分通常是字母或数字,用于区分同种设备中的单个设备,如hda、hdb、hdc…分别表示第一块、第二块、第三块IED硬盘。而hda1、hda2…表示第一块硬盘中的第一、第二个磁盘分区。
-
设备文件一般置于/dev目录下,如/dev/hda2、/dev/lp0等。
-
Linux使用虚拟文件系统VFS做为统一的操作接口来处理文件和设备。
-
与普通的目录和文件一样,每个设备也使用一个VFSinode来描述,其中包含着该种设备的主、次设备号。
-
对设备的操作也是通过对文件操作的file_operations结构体来调用驱动程序的设备服务子程序。
-
例如,当进程要求从某个设备上输入数据时,由该设备的
file_operations结构体得到服务子程序的操作函数入口,然后调用其中的read()函数完成数据输入操作。
-
同样,使用file_operations中的open()、close()、write()
分别完成对设备的启动、停止设备运行,向设备输出数据的操作。
§6.2 Linux的I/O控制
Linux的I/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[]数组登记时,它做为数组的下标。
-
把中断号为irq的irqaction结构体的首地址写入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_ops。file_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结构中有一个指向缓冲区信息的指针,
由它决定缓冲区的位置和大小等。
(责任编辑:IT)
Linux的设备管理
§6.1 设备管理概述
Linux使用套接口(socket)以文件I/O方式提供了对网络数据的访问。
三.设备的识别
而多个SCSI硬盘及其各个分区分别赋予次设备号1、2、3…。
[root@localhost /]# ls /dev/sda* -l 四.设备文件
file_operations结构体得到服务子程序的操作函数入口,然后调用其中的read()函数完成数据输入操作。
分别完成对设备的启动、停止设备运行,向设备输出数据的操作。
§6.2 Linux的I/O控制 Linux的I/O控制方式有三种:查询等待方式、 中断方式和DMA(内存直接存取)方式. 一.查询等待方式
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)); /* 向设备输出字符 */ . . 二.中断方式
static struct irqaction *irq_action[NR_IRQS+1];
struct irqaction { void (*handler)(int, void *, struct pt_regs *); /* 指向中断服务例程 */ unsigned long flags; /* 中断标志 */ unsigned long mask; /* 中断掩码 */ void *dev_id; /* struct irqaction *next; /* 指向下一个描述符 */ };
的irqaction结构体,并把它登记到irq_action[]数组中。
int request_irq(unsigned int irq, void (*handler)(int, void *, struct pt_regs *), unsigned long irqflags, const char * devname, void *dev_id);
§6.3 字符设备与块设备管理
一.字符设备管理
#define MAX_CHRDEV 128 static struct device_struct chrdevs[MAX_CHRDEV];
struct device_struct { const char * name; /* 指向设备名字符串 */ struct file_operations * fops; /* 指向文件操作函数的指针 */ };
二.块设备管理
static struct device_struct blkdevs[MAX_BLKDEV]
struct blk_dev_struct blk_dev[MAX_BLKDEV];
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结构中有一个指向缓冲区信息的指针, 由它决定缓冲区的位置和大小等。 (责任编辑:IT) |