一、实现ISR
1. ISR的概念
ISR(interrupt service routine)是处理IRQs(interrupt request line)的程序。Windows CE用一个ISR来处理所有的IRQ请求。当一个中断发生时,内核的异常处理程序先调用内核ISR,内核ISR禁用所有具有相同优先级和较低优先级的中断,然后调用已经注册的OAL ISR程序,一般ISR有下列特征:
1) 执行最小的中断处理,最小的中断处理指能够检验、答复产生中断的硬件,而把更多的处理工作留给IST(interrupt service thread)。
2) 当ISR完成时返回中断ID(中断ID大部分是预定义的)。
2. X86平台的ISR结构
X86平台的ISR保存在%_WINCEROOT%\PUBLIC\COMMON\OAK\CSP\I486 \OAL\fwpc.c中,函数名为PeRPISR。下面分析一下此函数的主要代码:
ULONG PeRPISR(void)
{
ULONG ulRet = SYSINTR_NOP; ///返回值,既中断ID(以SYSINTR_为前缀)
UCHAR ucCurrentInterrupt; ///当前中断号
if (fIntrTime) ////// fIntrTime 用于测试SR和IST的延时时间,测试工具为ILTiming.exe。
......
ucCurrentInterrupt = PICGetCurrentInterrupt(); ////返回当前中断IRQ
if (ucCurrentInterrupt == INTR_TIMER0) ///IRQ0,IRQ0为系统时钟(system tick)中断,具体见“二、实现系统时钟”
......
if (dwRebootAddress) ////是否需要重启动
RebootHandler();
......
if(ucCurrentInterrupt == INTR_RTC) ////IRQ8,real-time clock的中断
......
else if (ucCurrentInterrupt <= INTR_MAXIMUM) ///如果中断小于 INTR_MAXIMUM
{
ulRet = NKCallIntChain(ucCurrentInterrupt); ////调用中断链
if (ulRet == SYSINTR_CHAIN) ///如果中断链未包含中断
ulRet = OEMTranslateIrq(ucCurrentInterrupt); ////在IRQ 和SYSINTR之间转换,此函数返回IRQ对应的SYSINTR
......
PICEnableInterrupt(ucCurrentInterrupt, FALSE); ///启用除当前中断以外的所有中断
} ///else if
OEMIndicateIntSource(ulRet); ///通知内核已经发生SYSINTR中断
}