WDF驱动中,当我们写下WDF_DECLARE_CONTEXT_TYPE_WITH_NAME(XXX, YYY)到底发生了什么?

本文详细探讨了在Windows Driver Frameworks(WDF)中使用WDF_DECLARE_CONTEXT_TYPE_WITH_NAME宏的作用。通过分析宏定义,解释了该宏如何声明上下文类型和全局变量,以及如何通过WdfObjectGetTypedContextWorker获取对象上下文。同时,介绍了相关的编译器指令和结构体定义,揭示了驱动开发中的对象上下文处理机制。
摘要由CSDN通过智能技术生成
WDF驱动中,当我们写下WDF_DECLARE_CONTEXT_TYPE_WITH_NAME(FDO_DATA, FdoGetData)
到底发生了什么?
看下wdfobject.h文件中的WDF_DECLARE_CONTEXT_TYPE_WITH_NAME的宏定义:
#define WDF_DECLARE_CONTEXT_TYPE_WITH_NAME(_contexttype, _castingfunction) \
WDF_DECLARE_TYPE_AND_GLOBALS(                                              \
    _contexttype,                                                          \
    WDF_GET_CONTEXT_TYPE_INFO(_contexttype),                               \
    NULL,                                                                  \
    WDF_TYPE_DEFAULT_SECTION_NAME)                                         \
WDF_DECLARE_CASTING_FUNCTION(_contexttype, _castingfunction)
有点眼花,其中又调用了2个宏,所以继续看这2个宏定义:
#define WDF_DECLARE_TYPE_AND_GLOBALS(_contexttype, _UniqueType, _GetUniqueType, _section)\
typedef _contexttype* WDF_TYPE_NAME_POINTER_TYPE(_contexttype);         \
WDF_E
WDF驱动向底层磁盘驱动发送IRP_MJ_READ请求读取原始数据的示例代码如下: ```cpp NTSTATUS ReadFromDisk(WDFDEVICE Device, PVOID Buffer, ULONG Length) { NTSTATUS status = STATUS_SUCCESS; WDFREQUEST request = NULL; PIRP irp = NULL; KEVENT event; IO_STATUS_BLOCK ioStatusBlock; // 初始化事件 KeInitializeEvent(&event, NotificationEvent, FALSE); // 创建请求对象 status = WdfRequestCreate(WDF_NO_OBJECT_ATTRIBUTES, WdfDeviceGetIoTarget(Device), &request); if (!NT_SUCCESS(status)) { return status; } // 获取IRP对象 irp = WdfRequestWdmGetIrp(request); // 设置IRP参数 IoSetCompletionRoutine(irp, ReadCompletionRoutine, &event, TRUE, TRUE, TRUE); irp->Flags |= IRP_BUFFERED_IO | IRP_DEALLOCATE_BUFFER; irp->AssociatedIrp.SystemBuffer = Buffer; irp->MdlAddress = NULL; irp->IoStatus.Status = STATUS_NOT_SUPPORTED; irp->IoStatus.Information = 0; irp->Tail.Overlay.Thread = PsGetCurrentThread(); irp->Tail.Overlay.OriginalFileObject = NULL; irp->Tail.Overlay.AuxiliaryBuffer = NULL; irp->Flags &= ~(IRP_PAGING_IO | IRP_NOCACHE | IRP_SYNCHRONOUS_API); // 设置IO栈位置 PIO_STACK_LOCATION irpStack = IoGetNextIrpStackLocation(irp); irpStack->MajorFunction = IRP_MJ_READ; irpStack->Parameters.Read.Length = Length; irpStack->Parameters.Read.ByteOffset.QuadPart = 0; // 发送IRP请求 status = WdfRequestSend(request, WdfDeviceGetIoTarget(Device), WDF_NO_SEND_OPTIONS); if (!NT_SUCCESS(status)) { WdfRequestComplete(request, status); return status; } // 等待请求完成 KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL); // 获取IO状态 status = irp->IoStatus.Status; // 完成请求 WdfRequestComplete(request, status); return status; } ``` 在以上代码,我们创建了一个WDF请求对象,并获取了其对应的IRP对象,然后设置了IRP的各项参数,最终发送请求并等待请求完成。一旦请求完成,我们就可以通过IRP的IoStatus成员获取IO操作的状态,然后使用WdfRequestComplete函数完成请求。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值