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

📄 intrapi.c

📁 WinCE5.0部分核心源码
💻 C
字号:
//
// Copyright (c) Microsoft Corporation.  All rights reserved.
//
//
// This source code is licensed under Microsoft Shared Source License
// Version 1.0 for Windows CE.
// For a copy of the license visit http://go.microsoft.com/fwlink/?LinkId=3223.
//
/**     TITLE("Interrupt APIs")
 *++
 *
 *
 * 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);


BOOL IsValidIntrEvent(HANDLE hEvent);

DWORD (* pfnOEMTranslateSysIntr) (DWORD idInt);

#define MAX_IRQ     256
static BYTE iMaskRefs[MAX_IRQ];         // ref count for masked IRQ
static BYTE iEnableRefs[MAX_IRQ];       // ref count for initialized IRQ

static BOOL DoInterruptEnable (DWORD idInt, LPVOID pvData, DWORD cbData)
{
    DWORD irq = MAX_IRQ;
    BOOL  fEnable, fRet = TRUE;
    if (pfnOEMTranslateSysIntr
        && ((irq = pfnOEMTranslateSysIntr (idInt)) >= MAX_IRQ)) {
        // invalid irq, can't do anything about it
        return FALSE;
    }
    fEnable = INTERRUPTS_ENABLE (FALSE);
    DEBUGCHK ((irq >= MAX_IRQ) || (iEnableRefs[irq] < 255));
    if (irq >= MAX_IRQ) {
        fRet = OEMInterruptEnable (idInt, pvData, cbData);
    } else if (iEnableRefs[irq]
        || (fRet = OEMInterruptEnable (idInt, pvData, cbData))) {
        iEnableRefs[irq] ++;
    }
    INTERRUPTS_ENABLE (fEnable);
    return fRet;
}


static void DoInterruptDisable (DWORD idInt)
{
    DWORD irq = MAX_IRQ;
    BOOL  fEnable;
    if (pfnOEMTranslateSysIntr
        && ((irq = pfnOEMTranslateSysIntr (idInt)) >= MAX_IRQ)) {
        // invalid irq, can't do anything about it
        return;
    }
    fEnable = INTERRUPTS_ENABLE (FALSE);
    DEBUGCHK ((irq >= MAX_IRQ) || iEnableRefs[irq]);
    if ((irq >= MAX_IRQ)
        || (iEnableRefs[irq] && !--iEnableRefs[irq]))
        OEMInterruptDisable (idInt);
    INTERRUPTS_ENABLE (fEnable);
}


//------------------------------------------------------------------------------
//
//  @func   BOOL | InterruptMask | Mask Hardware Interrupt
//  @rdesc  Mask a hardware interrupt
//  @parm   DWORD | idInt | Interrupt ID to be associated with this IST.
//          See <l Interrupt ID's.Interrupt ID's> for possible values.
//  @parm   BOOL | fDisable | Either to disable (mask) or enable (unmask) the interrupt
//  @comm   A device driver can call InterrupMask to temporary mask/unmask an interrupt
//          so that it can safely access the hardware without worring about being
//          interrupted by the same interrupt.
//  @xref   <f InterruptDone> <f InterruptDisable> <f OEMInterruptEnable>
//          <l Overview.Kernel Interrupt Support>
//
//------------------------------------------------------------------------------
void SC_InterruptMask (DWORD idInt, BOOL fDisable)
{
    BOOL fEnable;
    DWORD irq = MAX_IRQ;
    
    TRUSTED_API_VOID (L"SC_InterruptMask");

    if (pfnOEMTranslateSysIntr
        && ((irq = pfnOEMTranslateSysIntr (idInt)) >= MAX_IRQ)) {
        // invalid irq, can't do anything about it
        return;
    }

    fEnable = INTERRUPTS_ENABLE (FALSE);
    if (irq < MAX_IRQ) {
        if (fDisable) {
            DEBUGCHK (iMaskRefs[irq] < 255);
            if (! iMaskRefs[irq]++)
                OEMInterruptDisable (idInt);
            
        } else {
            DEBUGCHK (iMaskRefs[irq]);
            if (iMaskRefs[irq] && ! --iMaskRefs[irq]) {
                OEMInterruptEnable (idInt, 0, 0);
            }
        }
    } else if (fDisable) {
        OEMInterruptDisable (idInt);
    } else {
        OEMInterruptEnable (idInt, 0, 0);
    }
    INTERRUPTS_ENABLE (fEnable);
}



//------------------------------------------------------------------------------
//
//  @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 
SC_InterruptInitialize(
    DWORD idInt,
    HANDLE hEvent,
    LPVOID pvData,
    DWORD cbData
    )
{
    TRUSTED_API (L"SC_InterruptInitialize", FALSE);
    
    /* 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 (DoInterruptEnable(idInt, pvData, cbData))
            return TRUE;
    } else {
        if (IntrEvents[idInt-SYSINTR_DEVICES]->pIntrProxy = AllocMem(HEAP_PROXY)) {
            if (DoInterruptEnable(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
    )
{
    TRUSTED_API_VOID (L"SC_InterruptDone");
    if (idInt >= SYSINTR_DEVICES && idInt < SYSINTR_MAXIMUM)
        OEMInterruptDone(idInt);
}



//------------------------------------------------------------------------------
//
//  @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;
    TRUSTED_API_VOID (L"SC_InterruptDisable");
    if ((idInt >= SYSINTR_DEVICES) && (idInt < SYSINTR_MAXIMUM) && (IntrEvents[idInt-SYSINTR_DEVICES])) {
        DoInterruptDisable(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);
}

//------------------------------------------------------------------------------
// NKSetInterruptEvent
//------------------------------------------------------------------------------
BOOL NKSetInterruptEvent (DWORD idInt)
{
    long mask;
    long pend;
    long *ptrPend;

    if ((idInt < SYSINTR_DEVICES) || (idInt >= SYSINTR_MAXIMUM))
        return FALSE;
    idInt -= SYSINTR_DEVICES;
    ptrPend = (long *) KInfoTable[KINX_PENDEVENTS];
    
    // calculate which DWORD based on idInt
    ptrPend += idInt >> 5;  // idInt / 32
    idInt   &= 0x1f;        // idInt % 32

    mask = 1 << idInt;
    do {
        pend = *ptrPend;
        if (pend & mask)
            return TRUE;    // The bit is already set, so all done.
    } while (InterlockedTestExchange(ptrPend, pend, pend|mask) != pend);
    return TRUE;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -