Функция SendDownStreamIrp
В примере кода для функции, приведенной SendDownStreamIrp
в этом разделе, показано, как реализовать предоставленную драйвером функцию, которая отправляет синхронные запросы IOCTL драйверу ACPI. Функцию SendDownStreamIrp
можно использовать для отправки запроса IOCTL_ACPI_EVAL_METHOD , запроса IOCTL_ACPI_EVAL_METHOD_EX или запроса IOCTL_ACPI_ENUM_CHILDREN .
Пример кода для SendDownStreamIrp
функции, включенной в этот раздел, выполняет следующую последовательность операций:
Создает объект события.
Вызывает IoBuildDeviceIoControlRequest для создания запроса IOCTL.
Вызывает IoCallDriver для отправки запроса IOCTL.
Ожидает, пока драйвер ACPI не сообщит объекту события, который указывает, что запрос завершен.
Возвращает состояние запроса вызывающей объекту.
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;
}