⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 intrapi.c

📁 wince下的源代码集合打包
💻 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 + -