Freigeben über


SendDownStreamIrp-Funktion

Das Codebeispiel für die Funktion, die SendDownStreamIrp in diesem Thema bereitgestellt wird, zeigt, wie eine vom Treiber bereitgestellte Funktion implementiert wird, die synchrone IOCTL-Anforderungen an den ACPI-Treiber sendet. Die SendDownStreamIrp Funktion kann verwendet werden, um eine IOCTL_ACPI_EVAL_METHOD-Anforderung , eine IOCTL_ACPI_EVAL_METHOD_EX-Anforderung oder eine IOCTL_ACPI_ENUM_CHILDREN-Anforderung zu senden.

Der Beispielcode für die Funktion, die SendDownStreamIrp in diesem Abschnitt enthalten ist, führt die folgende Vorgangssequenz aus:

  • Erstellt ein Ereignisobjekt.

  • Ruft IoBuildDeviceIoControlRequest auf, um die IOCTL-Anforderung zu erstellen.

  • Ruft IoCallDriver auf, um die IOCTL-Anforderung zu senden.

  • Wartet, bis der ACPI-Treiber das Ereignisobjekt signalisiert, was angibt, dass die Anforderung abgeschlossen wurde.

  • Gibt die status der Anforderung an den Aufrufer zurück.

NTSTATUS
SendDownStreamIrp(
    IN PDEVICE_OBJECT   Pdo,
    IN ULONG            Ioctl,
    IN PVOID            InputBuffer,
    IN ULONG            InputSize,
    IN PVOID            OutputBuffer,
    IN ULONG            OutputSize
)
/*
Routine Description:
    General-purpose function called to send a request to the PDO. 
    The IOCTL argument accepts the control method being passed down
    by the calling function

    This subroutine is only valid for the IOCTLS other than ASYNC EVAL. 

Parameters:
    Pdo             - the request is sent to this device object
    Ioctl           - the request - specified by the calling function
    InputBuffer     - incoming request
    InputSize       - size of the incoming request
    OutputBuffer    - the answer
    OutputSize      - size of the answer buffer

Return Value:
    NT Status of the operation
*/
{
    IO_STATUS_BLOCK     ioBlock;
    KEVENT              myIoctlEvent;
    NTSTATUS            status;
    PIRP                irp;

    // Initialize an event to wait on
    KeInitializeEvent(&myIoctlEvent, SynchronizationEvent, FALSE);

    // Build the request
    irp = IoBuildDeviceIoControlRequest(
        Ioctl, 
        Pdo,
        InputBuffer,
        InputSize,
        OutputBuffer,
        OutputSize,
        FALSE,
        &myIoctlEvent,
        &ioBlock);

    if (!irp) {
        return STATUS_INSUFFICIENT_RESOURCES;
    }

    // Pass request to Pdo, always wait for completion routine
    status = IoCallDriver(Pdo, irp);

    if (status == STATUS_PENDING) {
        // Wait for the IRP to be completed, and then return the status code
        KeWaitForSingleObject(
            &myIoctlEvent,
            Executive,
            KernelMode,
            FALSE,
            NULL);

        status = ioBlock.Status;
    }

    return status;
}