日期:2014-05-16  浏览次数:20736 次

linux内核--中断和中断处理(一)

    让硬件在需要的时候再向内核发出信号。这就是中断机制,先讨论中断,进而讨论内核如何使用所谓的中断处理函数处理对应的中断。

一、中断

  1)中断

    中断使得硬件得以发出通知给处理器。例如,在前几键盘的时候,键盘控制器会发出一个中断,通知操作系统有键按下。中断本质上时一种特殊的电信号,由硬件设备发向处理器。处理器收到中断后,会马上向操作系统反应此信号的到来,然后就由操作系统负责这些新到来的数据。硬件设备生产中断的时候并不考虑与处理器的时钟同步也就是说中断随时可以产生。因此,内核随时可能因为新到来的中断而被打断。

    当接收到一个中断后,中断控制器会给处理器发送一个电信号。处理器一经检测到此信号,便中断自己的当前工作转而处理中断。此后,处理器会通知操作系统一经产生中断,这样,操作系统就可以对这个中断进行适当的处理了。

    不同的设备对用不同的中断,每个中断通过一个唯一的数字标识。这些中断值通常称为中断请求(IRQ)。每个IRQ线都会被关联一个数值量。例如:IRQ 0是时钟中断,而IRQ 1是键盘中断。但并非所有的中断号都是这样严格定义的。对于链接在PCI总线上的设备而言,中断是动态分配的。而且其他费PC的体系结构也具有动态分配可用中断的特性。实际上,硬件发出中断就是为了引起内核的关注。

  2)异常

    异常与中断不同,它在产生时必须考虑与处理器时钟同步。实际上,异常页常常成为同步中断。比如在除数为0时,处理器就会产生一个异常。因为许多处理器体系结构处理异常与处理中断的方式类似。

二、中断处理程序

    相应一个中断的时候,内核会执行一个函数,该函数叫做中断处理程序或中断服务例程。产生中断的每个设备都有一个相应的中断处理程序。例如,由一个函数专门处理来自系统时钟的中断,而另外一个函数专门处理由键盘产生的中断。一个设备的中断处理程序是它设备驱动程序的一部分--设备驱动程序时用于对设备进行管理的内核代码。

    中断处理程序与其他内核函数的真正区别在于,中断处理程序是被内核调用来响应中断的,而它们运行于我们称之为中断上下文的特殊上下文中。中断可能随时发生,因此中断处理程序也就随时可能执行。所以必须保证中断处理程序能够快速执行,这样才能保证尽可能快地恢复中断代码的执行。因此,尽管对硬件而言,操作系统能迅速对其中断进程服务非常重要;当然对系统的其他部分而言,让中断处理程序在尽可能短的时间内完成运行也同样重要。

三、上半部与下半部的对比

    又想中断处理程序运行得快,又想中断处理程序完成的工作量多,两者不能兼得。为了达到更好的目的,一般把中断处理切为两个部分,中断处理程序时上半部,接收到一个中断,它就立即开始执行,但只做有严格时限的工作,例如对接收的中断进行应答或复位硬件,这些工作都是在所有中断被禁止的情况下完成的。能够被允许稍后完成的工作会推迟到下半部去。

    例如,当网卡接收来自网络的数据包时,需要通知内核数据包到了。网卡需要立即完成这件事,从而优化网络的吞吐量和传输周期,以避免超时。因此,网卡立即发出中断。内核通过执行网卡已注册的中断处理程序来做出应答。

    中断开始执行,通知硬件,拷贝最新的网络数据包到内存,然后读取网卡更多的数据报。这些都是重要、紧迫而又与硬件相关的工作。内核通常需要快速的拷贝网络数据报到系统内存,因为网卡上接收网络数据报的缓存大小固定,而且相比系统内存也要小得多。所以上述拷贝动作一旦被延迟,必然造成缓存溢出--进入的网络包占满了网卡的缓存,后续的入包只能丢弃。当网络数据报拷贝到系统内存后,中断的任务结束。这时它将控制权交换给系统被中断前原先运行的程序。处理和操作数据报的其他工作在随后的下半部中进行。

四、注册中断处理程序

    每一设备都有相关的驱动程序,如果设备使用中断(大部分设备如此),那么相应的驱动程序就注册一个中断处理程序。

    驱动程序可以通过request_irq()函数注册一个中断处理程序(它被声明在文件<linux/interrupt.h>中),并且激活给定的中断线,以处理中断:

/*   request_irq:分配一条给定的中断线 */

int request_irq(unsigned int irq,irq_handler_t handler,unsigned long flags,const char *name,void *dev);