다음을 통해 공유


__sdv_save_adapter_context 사용하여 어댑터 컨텍스트 필드 추적

NDIS가 미니포트 드라이버의 MiniportInitializeEx 콜백 함수를 호출하여 미니포트 어댑터를 초기화하는 경우 드라이버는 미니포트 어댑터를 나타내는 자체 내부 데이터 구조를 만듭니다. 드라이버는 미니포트 어댑터 컨텍스트라고 하는 이 구조를 사용하여 미니포트 어댑터를 관리하기 위해 드라이버에 필요한 디바이스별 상태 정보를 유지 관리합니다. 드라이버는 이 구조체에 대한 핸들을 NDIS에 전달합니다.

NDIS가 미니포트 어댑터와 관련된 미니포트 드라이버의 MiniportXxx 함수 중 하나를 호출하면 NDIS는 미니포트 어댑터 컨텍스트를 전달하여 드라이버에 올바른 미니포트 어댑터를 식별합니다. 미니포트 어댑터 컨텍스트는 미니포트 드라이버가 소유하고 유지 관리하며 NDIS 및 프로토콜 드라이버에 불투명합니다.

미니포트 어댑터 컨텍스트는 NDIS 호출 간에 미니포트 드라이버의 상태를 유지하므로 SDV(정적 드라이버 검증 도구)는 이 구조체에 대한 포인터를 식별할 수 있어야 합니다. SDV가 미니포트 드라이버의 상태를 추적할 수 있도록 하려면 __sdv_save_adapter_context 함수를 사용하여 미니포트 어댑터 컨텍스트에 핸들을 저장해야 합니다.

__sdv_save_adapter_context 함수에는 다음 구문이 있습니다.

__sdv_save_adapter_context( &adapter_context ) 

여기서 adapter_context 미니포트 드라이버에 의해 정의된 미니포트 어댑터 컨텍스트에 대한 핸들입니다. 이 함수는 미니포트 드라이버의 컨텍스트에서 한 번만 호출해야 합니다.

__sdv_save_adapter_context 함수는 정적 분석 도구에서만 사용됩니다. 이 함수는 컴파일러에서 무시됩니다.

다음 코드 예제에서는 __sdv_save_adapter_context 호출하는 시기와 SDV가 미니포트 어댑터 컨텍스트에서 정보를 추적하는 방법을 보여 있습니다.

다음 코드 예제에서는 미니포트 어댑터 컨텍스트 샘플 구조의 간소화된 버전을 보여 MP_Adapter.

typedef struct _MP_ADAPTER
{
    NDIS_HANDLE             AdapterHandle;

    BOOLEAN                 InterruptRegistered;
    NDIS_HANDLE             InterruptHandle;

    /* ..................... */

} MP_ADAPTER, *PMP_ADAPTER;

MiniportInitializeEx를 실행하는 동안 MP_ADAPER 구조체에 대한 메모리가 할당됩니다. 메모리 할당 직후 __sdv_save_adapter_context 호출됩니다.

미니포트 어댑터 컨텍스트 구조에 대한 유효한 포인터가 있는 즉시 __sdv_save_adapter_context 함수를 호출해야 합니다.

NDIS_STATUS 
MPInitialize(
    IN  NDIS_HANDLE                        MiniportAdapterHandle,
    IN  NDIS_HANDLE                        MiniportDriverContext,
    IN  PNDIS_MINIPORT_INIT_PARAMETERS     MiniportInitParameters
    )
{
    NDIS_STATUS     Status = NDIS_STATUS_SUCCESS;
    PMP_ADAPTER     pAdapter = NULL;
 
    /* ..................... */
 
    do
    {
   // allocate the memory for the AdapterContext
        pAdapter = NdisAllocateMemoryWithTagPriority(
            MiniportAdapterHandle,
            sizeof(MP_ADAPTER),
            NIC_TAG,
            LowPoolPriority
            );

   // save the adapter context, even before we check whether it is NULL or not 
 __sdv_save_adapter_context(&pAdapter);

        if (pAdapter == NULL)
        {
            Status = NDIS_STATUS_RESOURCES;
            break;
        }
 
        NdisZeroMemory(pAdapter, sizeof(MP_ADAPTER));
 
        pAdapter->AdapterHandle = MiniportAdapterHandle;
        pAdapter->PauseState = NicPaused;

   /* ..................... */

다음 코드 예제에서는 SDV가 미니포트 어댑터 컨텍스트에서 값을 추적하는 방법을 보여줍니다. 이 예제에서 드라이버는 미니포트 제공 함수인 MpRegisterInterrupt를 호출하여 인터럽트를 등록합니다. 호출이 성공하면 드라이버는 결과를 미니포트 어댑터 컨텍스트(pAdapter) 필드인 InterruptRegistered에 저장합니다.

//
// Register the interrupt
   //
        Status = MpRegisterInterrupt(pAdapter);
        if (Status != NDIS_STATUS_SUCCESS)
        {
            break;
        }
 
   // save the value of the result into the AdapterContext
        pAdapter->InterruptRegistered = TRUE;

미니포트 드라이버를 중지해야 하는 경우 NDIS는 핸들을 미니포트 어댑터 컨텍스트에 전달하여 드라이버의 MiniportHaltEx 함수를 호출합니다. SDV에도 이 핸들이 있으므로 이전 호출에서 __sdv_save_adapter_context SDV는 InterruptRegistered 필드의 값을 추적할 수 있습니다.

VOID MPHalt(
    IN  NDIS_HANDLE             MiniportAdapterContext,
    IN  NDIS_HALT_ACTION        HaltAction
    )
{
    PMP_ADAPTER     pAdapter = (PMP_ADAPTER) MiniportAdapterContext;
 
    /* ..................... */

    if (pAdapter->InterruptRegistered)
    {
        NdisMDeregisterInterruptEx(pAdapter->InterruptHandle);
        pAdapter->InterruptRegistered = FALSE;
    }

/* ..................... */