前一次(http://blog.csdn.net/zj510/article/details/8350184)学习了完成例程返回STATUS_SUCCESS的情况,也就是驱动调用IoCallDriver后不会再获得IRP。这次来看看再次获得IRP控制权的情况。
设置完成例程
先看看设置完成例程那里的代码:
IoCopyCurrentIrpStackLocationToNext(Irp);
KEVENT event;
KeInitializeEvent(&event, NotificationEvent, FALSE);//创建一个event对象。
//通过第三个参数将event对象传递给完成例程
IoSetCompletionRoutine(Irp, MyIoCompletion, &event, TRUE, TRUE, TRUE);
status = IoCallDriver(pdx->NextStackDevice, Irp);
if (status == STATUS_PENDING)
{//如果下层驱动是异步完成irp的话,就等待event被触发(完成例程会触发)
//如果下层驱动是同步完成irp或者其他错误,那么就无需等待了。直接向上返回。
KeWaitForSingleObject(&event,Executive,KernelMode ,FALSE,NULL);//等待完成例程触发这个事件,最后一个参数是NULL,表示无限等待
status = Irp->IoStatus.Status;//获取下层驱动的ir