แก้ไข

แชร์ผ่าน


Using __sdv_save_request and __sdv_retrieve_request for Deferred Procedure Calls

Deferred procedure calls (DPCs) present challenges to Static Driver Verifier (SDV) because it is difficult to track the framework request object. One difficulty is that the request must be retrieved from a global pointer, usually from queue context, or from a work item. To overcome this difficulty, Static Driver Verifier provides two functions: __sdv_save_request, and __sdv_retrieve_request. These functions map the deferred request to a request that SDV can track.

The __sdv_save_request, and __sdv_retrieve_request functions have the following syntax:

__sdv_save_request( request ) 
__sdv_retrieve_request( request ) 

Where request can be a handle to any framework request object.

These functions are only used by the static analysis tools. The functions are ignored by the compiler.

The following code example shows how the __sdv_save_request and __sdv_retrieve_request functions are used to guide SDV, so that SDV can map the deferred request. SDV can use this mapping to verify the DeferredRequestCompleted rule. The DeferredRequestCompleted rule requires that __sdv_save_request and __sdv_retrieve_request appear in your code. There are two driver property rules (AliasWithinDispatch, AliasWithinTimerDpc) that look for the existence of the __sdv_save_request and __sdv_retrieve_request functions.

In the following code example, the function EchoEvtIoRead is an EvtIoRead event callback function that saves the handle to the framework request object in the queue context area. The function EchoEvtTimerFunc is an EvtTimerFunc event callback function that retrieves it.

VOID
EchoEvtIoRead(
 )in WDFQUEUE   Queue,
 __in WDFREQUEST Request,
 __in size_t      Length
    )
{
/* ..................... */
    // Mark the request as cancelable
    WdfRequestMarkCancelable(Request, EchoEvtRequestCancel);
 
 
    // Defer the completion to another thread from the timer DPC and save the handle to the framework request object by using the __sdv_save_request function. 
    queueContext->CurrentRequest = Request;    
 __sdv_save_request(Request);

    queueContext->CurrentStatus  = Status;

    return;
}

The following code example demonstrates how the __sdv_retrieve_request function maps an existing request so that SDV can track it for completion.

VOID
EchoEvtTimerFunc(
    IN WDFTIMER     Timer
    )
{...................................................
    queue = WdfTimerGetParentObject(Timer);
    queueContext = QueueGetContext(queue);

    //
    // The DPC is automatically synchronized to the queue lock,
    // so this prevents race conditions from occurring without explicit driver-managed locking. The __sdv_retrieve_request function is used so that SDV can restore the deferred request in the timer DPC. Because we know that this deferred request is valid (because it has been previously saved), the __analysis_assume function is used to suppress false defects that might otherwise result in this code path.

    //
 __sdv_retrieve_request(queueContext->CurrentRequest);
    Request = queueContext->CurrentRequest;
 __analysis_assume(Request != NULL);
    if( Request != NULL ) {

        //
        // Try to remove cancel status from the request.
        //
        // The request is not completed if it is already canceled
        // because the EchoEvtIoCancel function has run, or is about to run,
        // and we are racing with it. 

        Status = WdfRequestUnmarkCancelable(Request);
// Because we know that the request is not NULL in this code path and that the request is no longer marked cancelable, we can use the __analysis_assume function to suppress the reporting of a false defect. 

 __analysis_assume(Status != STATUS_CANCELLED);
        if( Status != STATUS_CANCELLED ) {

            queueContext->CurrentRequest = NULL;
            Status = queueContext->CurrentStatus;

            KdPrint(("CustomTimerDPC Completing request 0x%p, Status 0x%x \n", Request,Status));

            WdfRequestComplete(Request, Status);
        }
        else {
            KdPrint(("CustomTimerDPC Request 0x%p is STATUS_CANCELLED, not completing\n",
                                Request));
        }
    }

    return;
}