📄 intrapi.c
字号:
/** TITLE("Interrupt APIs") *++ * * Copyright (c) 1995-2000 Microsoft Corporation. All rights reserved. * * Module Name: * * intrapi.c * * Abstract: * * This file contains the interrupt API functions. These functions allow * user mode threads to receive interrupt notifications and to interact with * the firmware interrupt management routines. *-- */#include "kernel.h"/* @doc EXTERNAL KERNEL HAL @topic Kernel Interrupt Support | The interrupt model for the kernel is that all interrupts get vectored into the kernel. To handle an interrupt, the OEM would typically install both an ISR (the actual interrupt service routine), and an IST(Interrupt service thread). All the work is mostly done in the IST. To install an ISR which should be invoked on a particular hardware interrupt, the OEM code should call the function <f HookInterrupt>. This will cause the kernel to jump into the registered function on that interrupt - with as small an overhead as possible - essentially just handfuls of assembly instructions. These ISR's however, are a bit different from traditions ISR's in systems like DOS or Windows. They are expected to be *extremely* light weight and fast - not executing more than a handful of assembly instructions. The reason is that these routines have lots if limitations - they execute with all interrupts turned off, they cannot call any kernel functions at all, they cannot use any stack, and they cant even use all of the registers. Above all they cannot cause any nested exception. What they can do is return back to the kernel with a return value which either tells the kernel that the interrupt has been handled (NOP), or asks it to schedule a specific IST by returning the interrupt code corresponding to it. See <l Interrupt ID's.Interrupt ID's> for details of the return codes. If the ISR returns a NOP, the kernel gets out of the exception handler right away. This means the ISR should have done everything required - for instance, signal an end of interrupt if necessary. If an IST is to be scheduled, it calls <l SetEvent.SetEvent> on the registered event and then gets out of the handler. To register an IST with the kernel, the OEM device driver should call the <f InterruptInitialize> function and register the event which should be signalled when an interrupt needs to be handled. On waking up, the thread should do everything required - eg talk to the hardware, read any data, process it etc. It should then call the <f InterruptDone> function inside the kernel - which simply calls the firmware routine <l OEMInterruptDone.OEMInterruptDone>. The OEM function should do the appropriate unmasking or end of interrupt signalling which is required. Other functions are used to initialize the system, and enable/disable specific interrupts. @xref <l OEMInit.OEMInit> <l OEMInterruptEnable.OEMInterruptEnable> <l OEMInterruptDisable.OEMInterruptDisable> <l OEMIdle.OEMIdle> <l OEMInterruptDone.OEMInterruptDone> <l HookInterrupt.HookInterrupt> <l InterruptInitialize.InterruptInitialize> <l InterruptDone.InterruptDone> <l InterruptDisable.InterruptDisable> <l Interrupt ID's.Interrupt ID's> @module intrapi.c - Interrupt Support APIs | Kernel API Functions for device driver interrupt handling. @xref <f InterruptInitialize> <f InterruptDone> <f InterruptDisable> <l Interrupt ID's.Interrupt ID's> <f HookInterrupt> <l Overview.Kernel Interrupt Support> *//* @func BOOL | HookInterrupt | Allows the firware to hook a specific hardware interrupt. @rdesc return FALSE if hook failed. @parm int | hwIntNumber | This is the hardware interrupt line which went off. Note - This is NOT an interrupt ID as defined by the Windows CE system. @parm FARPROC | pfnHandler | The interrupt handler to be registered @comm This is a kernel exposed function which is typically used by the OEM firmware to register it's ISR's during the <f OEMInit> call. @xref <l Overview.Kernel Interrupt Support> <f OEMInit> */ extern BOOL HookInterrupt(int hwIntNumber, FARPROC pfnHandler);/* @func BOOL | InterruptInitialize | Initialize Hardware Interrupt @rdesc returns FALSE if unable to initialize the interrupt @parm DWORD | idInt | Interrupt ID to be associated with this IST. See <l Interrupt ID's.Interrupt ID's> for possible values. @parm HANDLE | hEvent | Event to be signalled when the interrupt goes off. @parm LPVOID | pvData | Data to be passed to the firmware <f OEMInterruptEnable> call. This should be used for init data, or for allocating scratch space for the firmware ISR to use. The kernel will lock this data down and pass the physical address to the firmware routine so it can access it without being in danger of getting a TLB miss. @parm DWORD | cbData | size of data passed in. @comm A device driver calls InterruptInitialize to register an event to be signalled when the interrupt occurs and to enable the interrupt. @xref <f InterruptDone> <f InterruptDisable> <f OEMInterruptEnable> <l Overview.Kernel Interrupt Support>*/BOOL IsValidIntrEvent(HANDLE hEvent);BOOL SC_InterruptInitialize(DWORD idInt, HANDLE hEvent, LPVOID pvData, DWORD cbData){ /* Verify that the interrupt id is valid and that it is not already * initialized. */ if (idInt < SYSINTR_DEVICES || idInt >= SYSINTR_MAXIMUM) goto invParm; if (IntrEvents[idInt-SYSINTR_DEVICES]) goto invParm; if (!IsValidIntrEvent(hEvent)) { DEBUGCHK(0); goto invParm; } IntrEvents[idInt-SYSINTR_DEVICES] = HandleToEvent(hEvent); if (IntrEvents[idInt-SYSINTR_DEVICES]->pIntrProxy) { if (OEMInterruptEnable(idInt, pvData, cbData)) return TRUE; } else { if (IntrEvents[idInt-SYSINTR_DEVICES]->pIntrProxy = AllocMem(HEAP_PROXY)) { if (OEMInterruptEnable(idInt, pvData, cbData)) { IntrEvents[idInt-SYSINTR_DEVICES]->onequeue = 1; return TRUE; } FreeMem(IntrEvents[idInt-SYSINTR_DEVICES]->pIntrProxy,HEAP_PROXY); IntrEvents[idInt-SYSINTR_DEVICES]->pIntrProxy = 0; } } /* The firmware refused to enable the interrupt. */ IntrEvents[idInt-SYSINTR_DEVICES] = 0;invParm: KSetLastError(pCurThread, ERROR_INVALID_PARAMETER); return FALSE;}/* @func VOID | InterruptDone | Signal completion of interrupt processing @parm DWORD | idInt | Interrupt ID. See <l Interrupt ID's.Interrupt ID's> for a list of possible ID's. @rdesc none @comm A device driver calls InterruptDone when it has completed the processing for an interrupt and is ready for another interrupt. InterruptDone must be called to unmask the interrupt before the driver waits for the registered event to be signaled again. The kernel essentially calls through to the <f OEMInterruptDone> function. @xref <f InterruptInitialize> <f OEMInterruptDone> <l Overview.Kernel Interrupt Support>*/void SC_InterruptDone(DWORD idInt){ if (idInt >= SYSINTR_DEVICES && idInt < SYSINTR_MAXIMUM) OEMInterruptDone(idInt); return;}/* @func VOID | InterruptDisable | Disable Hardware Interrupt @parm DWORD | idInt | Interrupt ID. See <l Interrupt ID's.Interrupt ID's> for a list of possible ID's. @rdesc none @comm A device driver calls InterruptDisable to disable the hardware interrupt and to de-register the event registered via InterruptInitialize. The driver must call InterruptDisable before closing the event handle. The kernel calls through to the <f OEMInterruptDisable> routine as part of this. @xref <f InterruptInitialize> <f OEMInterruptDisable> <l Overview.Kernel Interrupt Support>*/ void SC_InterruptDisable(DWORD idInt){ DWORD loop; HANDLE hWake; LPPROXY pProx = NULL; LPEVENT pEvent; if ((idInt >= SYSINTR_DEVICES) && (idInt < SYSINTR_MAXIMUM) && (IntrEvents[idInt-SYSINTR_DEVICES])) { OEMInterruptDisable(idInt); pEvent = IntrEvents[idInt-SYSINTR_DEVICES]; IntrEvents[idInt-SYSINTR_DEVICES] = 0; for (loop = SYSINTR_DEVICES; loop < SYSINTR_MAXIMUM; loop++) if ((loop != idInt) && (pEvent == IntrEvents[loop-SYSINTR_DEVICES])) break; if (loop == SYSINTR_MAXIMUM) { pProx = pEvent->pIntrProxy; pEvent->pIntrProxy = 0; pEvent->onequeue = 0; if (hWake = (HANDLE)KCall((PKFN)EventModIntr,pEvent,EVENT_SET)) KCall((PKFN)MakeRunIfNeeded,hWake); FreeMem(pProx,HEAP_PROXY); } }}void FreeIntrFromEvent(LPEVENT lpe) { DWORD idInt; for (idInt = SYSINTR_DEVICES; idInt < SYSINTR_MAXIMUM; idInt++) if (IntrEvents[idInt-SYSINTR_DEVICES] == lpe) SC_InterruptDisable(idInt);}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -