背景: 我正在写一个虚拟USB转RS232驱动程序。但由于我的硬件是USB芯片(PDIUSBD12)而不是纯UART芯片,因此驱动程序需要进行一些特殊的调整。使用类似Hyperterminal之类的PC应该相信它正在与常规RS232芯片通信。无论如何,问题不在于此问题,而是更多的是理解-WDF问题,呵呵:)
问题: 我想要完成的是创建一个“读取请求”(无中生有)并将其传递给硬件。遗憾的是, WdfRequestRetrieveOutputMemory 导致“访问声音”/崩溃。我如何创建新请求有任何根本问题吗? WdfRequestRetrieveOutputMemory 的输入变量都不是NULL,但我猜想maskRequest变量在某种程度上是错误的?!
case IOCTL_SERIAL_WAIT_ON_MASK: // *** Wait on Mask ***
if (m_WaitMask == 0) { // Can only set if mask is not zero
status = STATUS_UNSUCCESSFUL;
DbgPrint("IOCTL_SERIAL_WAIT_ON_MASK failed, no wait mask\n");
bytesTransferred = 0;
break;
}
else {
// Registers completion routine for the mask-request
WdfRequestSetCompletionRoutine(Request, WaitOnMaskCompletionRoutine, pDevContext->BulkReadPipe); // pDevContext->BulkReadPipe??
// Forward the mask-request to the mask wait queue
status = WdfRequestForwardToIoQueue(Request, mask_queue);
if (!NT_SUCCESS(status)) {
DbgPrint("IOCTL_SERIAL_WAIT_ON_MASK, WdfRequestForwardToIoQueue failed\n");
bytesTransferred = 0;
break;
}
status = STATUS_PENDING;
// Create a brand new read request and pass it down to the hardware
mask_status = WdfRequestCreate(WDF_NO_OBJECT_ATTRIBUTES, NULL, &maskRequest);
if(!NT_SUCCESS(mask_status)) {
goto MaskExit;
}
mask_status = WdfRequestRetrieveOutputMemory(maskRequest, &maskMemory);
if(!NT_SUCCESS(mask_status)) {
goto MaskExit;
}
mask_status = WdfUsbTargetPipeFormatRequestForRead(pDevContext->BulkReadPipe, maskRequest, maskMemory, NULL);
if (!NT_SUCCESS(mask_status)) {
goto MaskExit;
}
WdfRequestSetCompletionRoutine(maskRequest, EvtRequestMaskReadCompletionRoutine, pDevContext->BulkReadPipe);
ret = WdfRequestSend(maskRequest, WdfUsbTargetPipeGetIoTarget(pDevContext->BulkReadPipe), WDF_NO_SEND_OPTIONS);
if (ret == FALSE) {
mask_status = WdfRequestGetStatus(maskRequest);
goto MaskExit;
}
else {
break;
}
MaskExit:
WdfRequestCompleteWithInformation(maskRequest, mask_status, 0);
}
答案 0 :(得分:1)
您不能以这种方式访问请求的输出缓冲区。使用WdfRequestCreate,您只需创建请求对象。因此,当您调用WdfRequestRetrieveOutputMemory时,此请求doenst会附加任何缓冲区。之后,您需要使用WdfUsbTargetPipeFormatRequestForRead进行初始化。在此调用的第三个参数中,您可以为此请求指定读取缓冲区。