共用方式為


初始化和註冊提供者模組

提供者模組必須先初始化數個數據結構,才能向網路模塊註冊器 (NMR) 註冊本身。 這些結構包括 NPI_MODULEID 結構、NPI_PROVIDER_CHARACTERISTICS 結構、NPI_REGISTRATION_INSTANCE 結構(包含在NPI_PROVIDER_CHARACTERISTICS結構內),以及提供者模組所定義的結構,用於提供者模組的註冊內容。

如果提供者模組向NMR註冊為 網路程式設計介面 (NPI) 的提供者, 定義 NPI 特定提供者特性,提供者模組也必須初始化 NPI 所定義的提供者特性結構的實例。

只要提供者模組向NMR註冊,所有這些數據結構都必須保持有效且保留在記憶體中。

例如,假設「EXNPI」 NPI 在標頭檔 Exnpi.h 中定義了以下內容:

// EXNPI NPI identifier
const NPIID EXNPI_NPIID = { ... };

// EXNPI provider characteristics structure
typedef struct EXNPI_PROVIDER_CHARACTERISTICS_
{
  .
  . // NPI-specific members
  .
} EXNPI_PROVIDER_CHARACTERISTICS, *PEXNPI_PROVIDER_CHARACTERISTICS;

以下顯示將自己註冊為 EXNPI NPI 提供者的提供者模組如何初始化所有這些數據結構:

// Include the NPI specific header file
#include "exnpi.h"

// Structure for the provider module's NPI-specific characteristics
const EXNPI_PROVIDER_CHARACTERISTICS NpiSpecificCharacteristics =
{
  .
  . // The NPI-specific characteristics of the provider module
  .
};

// Structure for the provider module's identification
const NPI_MODULEID ProviderModuleId =
{
  sizeof(NPI_MODULEID),
  MIT_GUID,
  { ... }  // A GUID that uniquely identifies the provider module
};

// Prototypes for the provider module's callback functions
NTSTATUS
  ProviderAttachClient(
    IN HANDLE NmrBindingHandle,
    IN PVOID ProviderContext,
    IN PNPI_REGISTRATION_INSTANCE ClientRegistrationInstance,
    IN PVOID ClientBindingContext,
    IN CONST VOID *ClientDispatch,
    OUT PVOID *ProviderBindingContext,
    OUT PVOID *ProviderDispatch
    );

NTSTATUS
  ProviderDetachClient(
    IN PVOID ProviderBindingContext
    );

VOID
  ProviderCleanupBindingContext(
    IN PVOID ProviderBindingContext
    );

// Structure for the provider module's characteristics
const NPI_PROVIDER_CHARACTERISTICS ProviderCharacteristics =
{
  0,
  sizeof(NPI_PROVIDER_CHARACTERISTICS),
  ProviderAttachClient,
  ProviderDetachClient,
  ProviderCleanupBindingContext,
  {
    0,
    sizeof(NPI_REGISTRATION_INSTANCE),
    &EXNPI_NPIID,
    &ProviderModuleId,
    0,
    &NpiSpecificCharacteristics
  }
};

// Context structure for the provider module's registration
typedef struct PROVIDER_REGISTRATION_CONTEXT_ {
  .
  . // Provider-specific members
  .
} PROVIDER_REGISTRATION_CONTEXT, *PPROVIDER_REGISTRATION_CONTEXT;

// Structure for the provider's registration context
PROVIDER_REGISTRATION_CONTEXT ProviderRegistrationContext =
{
  .
  . // Initial values for the registration context
  .
};

提供者模組通常會在其 DriverEntry 函式內初始化自己。 提供者模組的主要初始化工作如下:

  • 指定 Unload 函式。 當提供者模組從系統卸除時,作系統會呼叫此函式。 如果提供者模組未提供卸除函式,則無法從系統卸載提供者模組。

  • 呼叫 NmrRegisterProvider 函式,向NMR註冊提供者模組。

例如:

// Prototype for the provider module's unload function
VOID
  Unload(
    PDRIVER_OBJECT DriverObject
   );

// Variable to contain the handle for the registration
HANDLE ProviderHandle;

// DriverEntry function
NTSTATUS
  DriverEntry(
    PDRIVER_OBJECT DriverObject,
    PUNICODE_STRING RegistryPath
    )
{
  NTSTATUS Status;

  // Specify the unload function
  DriverObject->DriverUnload = Unload;

  .
  . // Other initialization tasks
  .

  // Register the provider module with the NMR
  Status = NmrRegisterProvider(
    &ProviderCharacteristics,
    &ProviderRegistrationContext,
    &ProviderHandle
    );

  // Return the result of the registration
  return Status;
}

如果提供者模組是多個 NPI 的提供者,它必須初始化一組獨立的數據結構,並針對它所支援的每個 NPI 呼叫 NmrRegisterProvider。 如果網路模組同時是提供者模組和用戶端模組(也就是說,它是一個 NPI 的提供者,另一個 NPI 的用戶端),它必須初始化兩組獨立的數據結構集,一組用於提供者介面,另一組用於用戶端介面,並呼叫兩者 NmrRegisterProviderNmrRegisterClient

提供者模組無需從其 DriverEntry 函式中呼叫 NmrRegisterProvider。 例如,在提供者模組是複雜驅動程式的子元件的情況下,只有在啟用提供者模組子元件時,才會註冊提供者模組。

如需實作提供者模組 Unload 函式的詳細資訊,請參閱 卸除提供者模組