將 UMDF 周邊驅動程式連接到序列埠
SerCx2 受控序列埠上周邊裝置的 UMDF 驅動程式需要特定硬體資源才能操作裝置。 包含在這些資源中,是驅動程式需要開啟與序列埠之邏輯連線的資訊。 其他資源可能包括中斷,以及一或多個 GPIO 輸入或輸出針腳。
此驅動程式會實作 IPnpCallbackHardware2 介面,並在呼叫驅動程式的 IDriverEntry::OnDeviceAdd 方法期間,向 Windows 驅動程式架構註冊此介面。 架構會呼叫 IPnpCallbackHardware2 介面中的方法,以通知驅動程式裝置電源狀態的變更。
在串列連線的周邊裝置進入未初始化的 D0 裝置電源狀態之後,驅動程式架構會呼叫驅動程式的 IPnpCallbackHardware2::OnPrepareHardware 方法,告知驅動程式準備此裝置以供使用。 在此呼叫期間,驅動程式會收到兩份硬體資源清單作為輸入參數。 pWdfResourcesRaw參數會指向原始資源清單,而pWdfResourcesTranslated參數會指向已翻譯的資源清單。 這兩個參數都是 IWDFCmResourceList 物件的指標。 翻譯的資源包括周邊驅動程式建立與串列連線周邊裝置的邏輯連線所需的連線識別碼。
若要讓 UMDF 周邊驅動程式在其資源清單中接收連線識別碼,安裝驅動程式的 INF 檔案必須在其 WDF 特定的 DDInstall 區段中包含下列指示詞:
UmdfDirectHardwareAccess = AllowDirectHardwareAccess 如需此指示詞的詳細資訊,請參閱 在 INF 檔案中指定 WDF 指示詞。 如需用來建置使用這個指示詞之對應 INF 檔案) 的 INX 檔案範例 (範例,請參閱 WDK 驅動程式範例中的 SpbAccelerometer。
下列程式碼範例示範驅動程式的 OnPrepareHardware 方法如何從 pWdfResourcesTranslated 參數取得連線識別碼。
BOOLEAN fConnectIdFound = FALSE;
BOOLEAN fDuplicateFound = FALSE;
LARGE_INTEGER connectionId = 0;
ULONG resourceCount;
resourceCount = pWdfResourcesTranslated->GetCount();
// Loop through the resources and save the relevant ones.
for (ULONG ix = 0; ix < resourceCount; ix++)
{
PCM_PARTIAL_RESOURCE_DESCRIPTOR pDescriptor;
pDescriptor = pWdfResourcesTranslated->GetDescriptor(ix);
if (pDescriptor == NULL)
{
hr = E_POINTER;
break;
}
// Determine the resource type.
switch (pDescriptor->Type)
{
case CmResourceTypeConnection:
{
// Check against the expected connection types.
UCHAR Class = pDescriptor->u.Connection.Class;
UCHAR Type = pDescriptor->u.Connection.Type;
if (Class == CM_RESOURCE_CONNECTION_CLASS_SERIAL)
{
if (Type == CM_RESOURCE_CONNECTION_TYPE_SERIAL_UART)
{
if (fConnIdFound == FALSE)
{
// Save the serial controller's connection ID.
connectionId.LowPart = pDescriptor->u.Connection.IdLowPart;
connectionId.HighPart = pDescriptor->u.Connection.IdHighPart;
fConnectIdFound = TRUE;
}
else
{
fDuplicateFound = TRUE;
}
}
}
if (Class == CM_RESOURCE_CONNECTION_CLASS_GPIO)
{
// Check for GPIO pin resource.
...
}
}
break;
case CmResourceTypeInterrupt:
{
// Check for interrupt resources.
...
}
break;
default:
// Ignore all other resource descriptors.
break;
}
}
上述程式碼範例會將串列連線周邊裝置的連接識別碼複製到名為 的 connectionId
變數中。 下列程式碼範例示範如何將連線識別碼併入裝置路徑名稱,以用來識別周邊裝置所連接的序列控制器。
WCHAR szTargetPath[100];
HRESULT hres;
// Create the device path using the well-known resource hub
// path name and the connection ID.
//
hres = StringCbPrintfW(&szTargetPath[0],
sizeof(DevicePath),
L"\\\\.\\RESOURCE_HUB\\%0*I64x",
(size_t)(sizeof(LARGE_INTEGER) * 2),
connectionId.QuadPart);
if (FAILED(hres))
{
// Error handling
...
}
上述程式碼範例會將序列控制器 szTargetPath
的裝置路徑名稱寫入陣列中。 下列程式碼範例會使用此路徑名稱來開啟序列控制器的檔案控制碼。
UMDF_IO_TARGET_OPEN_PARAMS openParams;
openParams.dwShareMode = 0;
openParams.dwCreationDisposition = OPEN_EXISTING;
openParams.dwFlagsAndAttributes = FILE_FLAG_OVERLAPPED;
hres = pRemoteTarget->OpenFileByName(&szTargetPath[0],
(GENERIC_READ | GENERIC_WRITE),
&openParams);
if (FAILED(hres))
{
// Error handling
...
}
在上述程式碼範例中 pRemoteTarget
,參數是 IWDFRemoteTarget 物件的指標。 如果 呼叫 IWDFRemoteTarget::OpenFileByName 方法成功,則串列連線周邊裝置的驅動程式可以使用 IWDFRemoteTarget 物件將 I/O 要求傳送至序列控制器。
若要將讀取或寫入要求傳送至周邊裝置,驅動程式會先呼叫此物件的 IWDFRemoteTarget::FormatRequestForRead 或 IWDFRemoteTarget::FormatRequestForWrite 方法來格式化要求。 (IWDFRemoteTarget 介面會從 IWDFIoTarget 介面繼承這兩種方法。)
若要將 I/O 控制項要求傳送至序列控制器,驅動程式會先呼叫 IWDFRemoteTarget::FormatRequestForIoctl 方法來格式化要求。 (IWDFRemoteTarget 介面會從 IWDFIoTarget 介面繼承此方法。) 接下來,驅動程式會呼叫 IWDFIoRequest::Send 方法,以將 I/O 控制要求傳送至序列連接的周邊裝置。
在下列程式碼範例中,周邊驅動程式會將 I/O 控制項要求傳送至序列控制器。
HRESULT hres;
IWDFMemory *pInputMemory = NULL;
// Create a new I/O request.
if (SUCCEEDED(hres))
{
hres = pWdfDevice->CreateRequest(NULL,
pWdfDevice,
&pWdfIoRequest);
if (FAILED(hres))
{
// Error handling
...
}
}
// Allocate memory for the input buffer.
if (SUCCEEDED(hres))
{
hres = pWdfDriver->CreatePreallocatedWdfMemory(pInBuffer,
inBufferSize,
NULL,
pWdfIoRequest,
&pInputMemory);
if (FAILED(hres))
{
// Error handling
...
}
}
// Format the request to be an I/O control request.
if (SUCCEEDED(hres))
{
hres = pRemoteTarget->FormatRequestForIoctl(pWdfIoRequest,
ioctlCode,
NULL,
pInputMemory,
NULL,
NULL,
NULL);
if (FAILED(hres))
{
// Error handling
...
}
}
// Send the request to the serial controller.
if (SUCCEEDED(hres))
{
ULONG Flags = fSynchronous ? WDF_REQUEST_SEND_OPTION_SYNCHRONOUS : 0;
if (!fSynchronous)
{
pWdfIoRequest->SetCompletionCallback(pCallback, NULL);
}
hres = pWdfIoRequest->Send(pRemoteTarget, Flags, 0);
if (FAILED(hres))
{
// Error handling
...
}
}
if (fSynchronous || FAILED(hres))
{
pWdfIoRequest->DeleteWdfObject();
SAFE_RELEASE(pWdfIoRequest);
}
上述程式碼範例會執行下列動作:
pWdfDevice
變數是架構裝置物件的IWDFDevice介面指標,代表序列連接的周邊裝置。 IWDFDevice::CreateRequest方法會建立 I/O 要求,並將此要求封裝在 參數所pWdfIoRequest
指向的IWDFIoRequest介面實例中。 稍後刪除 I/O 要求 (請參閱步驟 6) 。 此實作會稍微沒有效率,因為它會針對傳送的每個 I/O 要求建立並刪除要求物件。 更有效率的方法是針對一系列 I/O 要求重複使用相同的要求物件。 如需詳細資訊,請參閱 重複使用 Framework 要求物件。pWdfDriver
變數是代表周邊驅動程式之架構驅動程式物件的IWDFDriver介面指標。pInBuffer
和inBufferSize
變數會指定 I/O 控制項要求的輸入緩衝區位址和大小。 IWDFDriver::CreatePreallocatedWdfMemory方法會建立輸入緩衝區的架構記憶體物件,並將 所pWdfIoRequest
指向的IWDFIoRequest物件指定為記憶體物件的父物件。變數
pWdfRemoteTarget
是從先前程式碼範例中的 OpenFileByName 呼叫取得的遠端目標指標。 IWDFRemoteTarget::FormatRequestForIoctl方法會將 I/O 控制作業的要求格式化。 變數ioctlCode
會設定為 序列 I/O 要求介面中資料表所列的其中一個 I/O 控制項程式碼。fSynchronous
如果要同步傳送 I/O 控制要求,則變數為TRUE,如果是非同步傳送,則為FALSE。 變數pCallback
是先前建立 之 IRequestCallbackRequestCompletion 介面的指標。 如果要以非同步方式傳送要求, 對 IWDFIoRequest::SetCompletionCallback 方法的呼叫會登錄此介面。 稍後會呼叫 IRequestCallbackRequestCompletion::OnCompletion 方法,以在要求非同步完成時通知驅動程式。Send方法會將格式化的寫入要求傳送至串列連接的周邊裝置。 變數
Flags
會指出寫入要求是要以同步或非同步方式傳送。如果同步傳送要求, IWDFIoRequest::D eleteWdfObject 方法會刪除 所
pWdfIoRequest
指向的 I/O 要求物件和 所pInputMemory
指向的子物件。 IWDFIoRequest介面會從IWDFObject介面繼承這個方法。 如果要求是以非同步方式傳送,則稍後應該會在驅動程式的OnCompletion方法中呼叫DeleteWdfObject方法。