1 /*++
2 
3 Copyright (c) Microsoft Corporation
4 
5 Module Name:
6 
7     FxRequest.hpp
8 
9 Abstract:
10 
11     This is the request object for the driver frameworks.
12 
13     The request object wraps the IRP, containing persistent
14     information required by the driver frameworks.
15 
16 Author:
17 
18 
19 
20 
21 
22 Environment:
23 
24     Both kernel and user mode
25 
26 Revision History:
27 
28 --*/
29 
30 #ifndef _FXREQUEST_H_
31 #define _FXREQUEST_H_
32 
33 //
34 // Magic number to differentiate between default value and caller provided one
35 //
36 #define USE_DEFAULT_PRIORITY_BOOST ((CHAR) 0x7F)
37 
38 //
39 // This tag is used to set and clear the completion callback state as the
40 // ownership of the request transfers from I/O pkg-to-queue or queue-to-queue.
41 //
42 #define FXREQUEST_STATE_TAG (PVOID) 'tatS'
43 
44 //
45 // This tag is used when the request is added and removed from FxIrpQueue.
46 //
47 #define FXREQUEST_QUEUE_TAG (PVOID) 'ueuQ'
48 
49 //
50 // This tag is used to take a reference in the completion path.
51 //
52 #define FXREQUEST_COMPLETE_TAG (PVOID) 'pmoC'
53 
54 //
55 // Use this tag when you want to temporarily hold the object from
56 // disappearing between unlock and lock operation.
57 //
58 #define FXREQUEST_HOLD_TAG (PVOID) 'dloH'
59 
60 //
61 // This tag is used to take a reference in the completion path.
62 //
63 #define FXREQUEST_FWDPRG_TAG (PVOID) 'PdwF'
64 
65 //
66 // This tag is used to take a reference in the completion path for driver created
67 // requests that support completion operations.
68 //
69 #define FXREQUEST_DCRC_TAG (PVOID) 'CRCD'
70 
71 extern "C" {
72 #if defined(EVENT_TRACING)
73 #include "FxRequest.hpp.tmh"
74 #endif
75 }
76 
77 #if (FX_CORE_MODE == FX_CORE_USER_MODE)
78 #define WDF_REQUEST_SEND_OPTION_IMPERSONATION_FLAGS                   \
79         (WDF_REQUEST_SEND_OPTION_IMPERSONATE_CLIENT |                 \
80          WDF_REQUEST_SEND_OPTION_IMPERSONATION_IGNORE_FAILURE)
81 
82 #define FxIrpStackFlagsFromSendFlags(sendFlags)                       \
83                ((WUDFX_IRP_STACK_FLAGS)                               \
84                     ((sendFlags) & WDF_REQUEST_SEND_OPTION_IMPERSONATION_FLAGS))
85 #endif
86 
87 typedef
88 NTSTATUS
89 (*PFN_FX_QUEUE_REQUEST_COMPLETE) (
90     __in FxRequest* Request,
91     __in FxIoQueue* Queue,
92     __in_opt WDFCONTEXT Context
93     );
94 
95 struct FxRequestSystemBuffer : public IFxMemory {
96     friend FxRequest;
97 
98 public:
99     FxRequestSystemBuffer(
100         VOID
101         )
102     {
103         m_Buffer = NULL;
104     }
105 
106     _Must_inspect_result_
107     virtual
108     PVOID
109     GetBuffer(
110         VOID
111         );
112 
113     virtual
114     size_t
115     GetBufferSize(
116         VOID
117         );
118 
119     _Must_inspect_result_
120     virtual
121     PMDL
122     GetMdl(
123         VOID
124         );
125 
126     virtual
127     WDFMEMORY
128     GetHandle(
129         VOID
130         );
131 
132     virtual
133     USHORT
134     GetFlags(
135         VOID
136         );
137 
138     virtual
139     PFX_DRIVER_GLOBALS
140     GetDriverGlobals(
141         VOID
142         );
143 
144     virtual
145     ULONG
146     AddRef(
147         __in PVOID Tag,
148         __in LONG Line,
149         __in_opt PSTR File
150         );
151 
152     virtual
153     ULONG
154     Release(
155         __in PVOID Tag,
156         __in LONG Line,
157         __in_opt PSTR File
158         );
159 
160     virtual
161     VOID
162     Delete(
163         VOID
164         );
165 
166     __inline
167     BOOLEAN
168     IsBufferSet(
169         VOID
170         )
171     {
172         return m_Buffer != NULL ? TRUE : FALSE;
173     }
174 
175     __inline
176     VOID
177     ClearBufferMdl(
178         VOID
179         )
180     {
181         m_Buffer = NULL;
182         m_Mdl = NULL;
183     }
184 
185 protected:
186     __inline
187     VOID
188     SetBuffer(
189         PVOID Buffer
190         )
191     {
192         ASSERT(m_Buffer == NULL);
193         m_Buffer = Buffer;
194     }
195 
196     __inline
197     VOID
198     SetMdl(
199         PMDL Mdl
200         )
201     {
202         ASSERT(m_Mdl == NULL);
203         m_Mdl = Mdl;
204     }
205 
206     FxRequest*
207     GetRequest(
208         VOID
209         );
210 
211 protected:
212     //
213     // The current irp stack location indicates which type to use
214     //
215     // The buffer / MDL is cached away as a copy instead of using the PIRP values
216     // directly because we want to capture the current state of the irp when
217     // returning the WDFMEMORY.  For instance, if used the PIRP value directly
218     // when implementing GetBuffer(), we are subject to the PIRP being formatted
219     // for the next stack location and changing the buffer pointer, or worse,
220     // changing the MDL value and have the resulting MDL not be mapped, and then
221     // a call to MmGetSystemAddressForMdlSafe can return NULL, and thus GetBuffer(),
222     // return NULL, which would violate the contract for GetBuffer().
223     //
224     // As an example, let's
225     // 1) the WDFREQUEST/PIRP comes in as a read on a direct io device object,
226     //    so Irp->MdlAddress = <read MDL>
227     // 2) This WDFMEMORY will return <read MDL VA> in GetBuffer()
228     // 3) the WDFREQUEST is formatted for an IOCTL which is METHOD_OUT_DIRECT
229     //    with a new output buffer.  Irp->MdlAddres = <IOCTL MDL> now.
230     // 4) This same WDFMEMORY will now return <IOCTL MDL VA> in GetBuffer()
231     //
232     // Essentialy, formatting the WDFREQUEST causes unintended side affects for
233     // the WDFMEMORYs it returns.  To eliminate those side affects, we capture
234     // the original buffer.
235     //
236     union {
237         PVOID m_Buffer;
238         PMDL m_Mdl;
239     };
240 };
241 
242 struct FxRequestOutputBuffer : public IFxMemory {
243     friend FxRequest;
244 
245 public:
246     FxRequestOutputBuffer(
247         VOID
248         )
249     {
250         m_Buffer = NULL;
251     }
252 
253     virtual
254     PVOID
255     GetBuffer(
256         VOID
257         );
258 
259     virtual
260     size_t
261     GetBufferSize(
262         VOID
263         );
264 
265     _Must_inspect_result_
266     virtual
267     PMDL
268     GetMdl(
269         VOID
270         );
271 
272     virtual
273     WDFMEMORY
274     GetHandle(
275         VOID
276         );
277 
278     virtual
279     USHORT
280     GetFlags(
281         VOID
282         );
283 
284     virtual
285     PFX_DRIVER_GLOBALS
286     GetDriverGlobals(
287         VOID
288         );
289 
290     virtual
291     ULONG
292     AddRef(
293         __in PVOID Tag,
294         __in LONG Line,
295         __in_opt PSTR File
296         );
297 
298     virtual
299     ULONG
300     Release(
301         __in PVOID Tag,
302         __in LONG Line,
303         __in_opt PSTR File
304         );
305 
306     virtual
307     VOID
308     Delete(
309         VOID
310         );
311 
312     __inline
313     BOOLEAN
314     IsBufferSet(
315         VOID
316         )
317     {
318         return m_Buffer != NULL ? TRUE : FALSE;
319     }
320 
321     __inline
322     VOID
323     ClearBufferMdl(
324         VOID
325         )
326     {
327         m_Buffer = NULL;
328         m_Mdl = NULL;
329     }
330 
331 protected:
332     __inline
333     VOID
334     SetBuffer(
335         __in PVOID Buffer
336         )
337     {
338         ASSERT(m_Buffer == NULL);
339         m_Buffer = Buffer;
340     }
341 
342     __inline
343     VOID
344     SetMdl(
345         __in PMDL Mdl
346         )
347     {
348         ASSERT(m_Mdl == NULL);
349         m_Mdl = Mdl;
350     }
351 
352     FxRequest*
353     GetRequest(
354         VOID
355         );
356 
357 protected:
358     //
359     // The current irp stack location indicates which type to use
360     //
361     // See comments in FxRequestSystemBuffer's union for why we capture the
362     // values vs using them directly from the PIRP.
363     //
364     union {
365         PVOID m_Buffer;
366         PMDL m_Mdl;
367     };
368 };
369 
370 // begin_wpp enum
371 enum FxListEntryNames {
372     FxListEntryNameCleanup = 0,
373 
374     // this entry is used when the request is owned by the framework
375     FxListEntryQueueOwned,
376 
377     // this entry is used when the request is presented to the driver
378     FxListEntryDriverOwned,
379 
380     // this entry is used for forward progress
381     FxListEntryForwardProgress
382 };
383 
384 enum  FxRequestPowerStopState {
385     FxRequestPowerStopUnknown = 0, // Initial state
386 
387     // Set when the driver calls WdfRequestStopAcknowledge
388     FxRequestPowerStopAcknowledged = 0x1,
389 
390     // Set when the driver WdfRequestStopAcknowledge with requeue option
391     FxRequestPowerStopAcknowledgedWithRequeue = 0x2,
392 };
393 
394 // end_wpp
395 
396 class FxRequest : public FxRequestBase {
397 
398     friend FxIoTarget;
399     friend FxIoQueue;
400 
401     friend FxRequestMemory;
402     friend FxRequestOutputBuffer;
403     friend FxRequestSystemBuffer;
404     friend VOID GetTriageInfo(VOID);
405 
406 protected:
407     //
408     // This field points to the queue that the request is currently
409     // associated with.
410     //
411     FxIoQueue* m_IoQueue;
412 
413     FxRequestSystemBuffer m_SystemBuffer;
414 
415     FxRequestOutputBuffer m_OutputBuffer;
416 
417     //
418     // This is for use by the owner of the FxRequest which is FxIoQueue OR FxIoTarget
419     //
420     LIST_ENTRY m_OwnerListEntry;
421 
422     LIST_ENTRY m_OwnerListEntry2;
423 
424     //
425     // This is used by the queue to keep track of all forward progress requests
426     //
427     LIST_ENTRY m_ForwardProgressList;
428 
429     //
430     // Used when the request is a reserved request to track the queue it was
431     // originally allocated for, so that it can be returned to the forward
432     // progress queue for reuse when the request is completed.
433     //
434     FxIoQueue *m_ForwardProgressQueue;
435 
436     //
437     // Generic context exposed to other modules.
438     //
439     PVOID m_InternalContext;
440 
441     //
442     // If TRUE, the client driver has been presented with this WDFREQUEST at
443     // least once.
444     //
445     BOOLEAN m_Presented;
446 
447     //
448     // For tracking whether the driver has acknowledged power stop/purge notifications.
449     //
450     BYTE m_PowerStopState;
451 
452     //
453     // If TRUE, this is a reserved request
454     //
455     BOOLEAN   m_Reserved;
456 
457     //
458     //  If TRUE, this is used to determine how to free the request -
459     //  either to the lookaside list or using ExFreePool
460     //
461     BOOLEAN   m_ForwardRequestToParent;
462 
463 public:
464 
465     //
466     // Normally, this is available by the object implementing
467     // IFxListEntry, but currently all callers of this know they
468     // are dealing with an FxRequest*.
469     //
470     // If FxRequests must go on a general typeless list, then
471     // the IFxListEntry interface should be added to FxRequest.
472     //
473     __inline
474     PLIST_ENTRY
475     GetListEntry(
476         __in FxListEntryNames Index
477         )
478     {
479         switch (Index) {
480         case FxListEntryQueueOwned:  return &m_OwnerListEntry;
481         case FxListEntryDriverOwned:  return &m_OwnerListEntry2;
482         case FxListEntryForwardProgress: return &m_ForwardProgressList;
483         default: ASSERT(FALSE); return NULL;
484         }
485     }
486 
487     static
488     FxRequest*
489     _FromOwnerListEntry(
490         __in FxListEntryNames Index,
491         __in PLIST_ENTRY OwnerListEntry
492         )
493     {
494         switch (Index) {
495         case FxListEntryQueueOwned:
496             return CONTAINING_RECORD(OwnerListEntry, FxRequest, m_OwnerListEntry);
497         case FxListEntryDriverOwned:
498             return CONTAINING_RECORD(OwnerListEntry, FxRequest, m_OwnerListEntry2);
499         case FxListEntryForwardProgress:
500             return CONTAINING_RECORD(OwnerListEntry, FxRequest, m_ForwardProgressList);
501         default:
502             ASSERT(FALSE);
503             return NULL;
504         }
505     }
506 
507     __inline
508     VOID
509     CopyCurrentIrpStackLocationToNext(
510         VOID
511         )
512     {
513         FxIrp* irp = GetSubmitFxIrp();
514         irp->CopyCurrentIrpStackLocationToNext();
515     }
516 
517     _Must_inspect_result_
518     NTSTATUS
519     Reuse(
520         __in PWDF_REQUEST_REUSE_PARAMS ReuseParams
521         );
522 
523     __inline
524     BOOLEAN
525     IsCancelled(
526         VOID
527         )
528     {
529         return m_Irp.IsCanceled() || m_Canceled;
530     }
531 
532     __inline
533     VOID
534     CopyCompletionParams(
535         __in PWDF_REQUEST_COMPLETION_PARAMS Params
536         )
537     {
538         if (m_RequestContext != NULL) {
539             RtlCopyMemory(Params,
540                           &m_RequestContext->m_CompletionParams,
541                           sizeof(WDF_REQUEST_COMPLETION_PARAMS));
542         }
543         else {
544             DoTraceLevelMessage(
545                 GetDriverGlobals(), TRACE_LEVEL_ERROR, TRACINGREQUEST,
546                 "WdfRequestGetCompletionParams will not return valid information if the"
547                 " request is not formatted using WdfIoTargetFormatxxx DDIs"
548                 );
549 
550             FxVerifierDbgBreakPoint(GetDriverGlobals());
551 
552             WDF_REQUEST_COMPLETION_PARAMS_INIT(Params);
553         }
554     }
555 
556     VOID
557     __inline
558     SetPresented(
559         VOID
560         )
561     {
562         //
563         // No need to synchronize setting this value with checking it because
564         // we check it in the complete path.  We will not be about to present
565         // and completing the request in 2 simultaneous contexts.
566         //
567         m_Presented = TRUE;
568     }
569 
570     VOID
571     AddIrpReference(
572         VOID
573         );
574 
575     VOID
576     ReleaseIrpReference(
577         VOID
578         );
579 
580     virtual
581     ULONG
582     AddRefOverride(
583         __in WDFOBJECT_OFFSET Offset,
584         __in PVOID Tag = NULL,
585         __in LONG Line = 0,
586         __in_opt PSTR File = NULL
587         );
588 
589     virtual
590     ULONG
591     ReleaseOverride(
592         __in WDFOBJECT_OFFSET Offset,
593         __in PVOID Tag = NULL,
594         __in LONG Line = 0,
595         __in_opt PSTR File = NULL
596         );
597 
598     __inline
599     CfxDevice*
600     GetDevice(
601         VOID
602     )
603     {
604         return m_Device;
605     }
606 
607     __inline
608     BOOLEAN
609     IsReserved(
610         )
611     {
612         return m_Reserved;
613     }
614 
615     __inline
616     VOID
617     SetReserved(
618         )
619     {
620         m_Reserved = TRUE;
621     }
622 
623     __inline
624     VOID
625     SetForwardProgressQueue(
626         __in FxIoQueue *Queue
627         )
628     {
629         m_ForwardProgressQueue = Queue;
630     }
631 
632 protected:
633     FxRequest(
634         __in PFX_DRIVER_GLOBALS FxDriverGlobals,
635         __in MdIrp Irp,
636         __in FxRequestIrpOwnership Ownership,
637         __in FxRequestConstructorCaller Caller,
638         __in USHORT ObjectSize
639         );
640 
641     #if DBG
642     ~FxRequest(
643         VOID
644         );
645     #endif // DBG
646 
647     __inline
648     VOID
649     SetCurrentQueue(
650         __in FxIoQueue *Queue
651         )
652     {
653         m_IoQueue = Queue;
654     }
655 
656 
657     WDFMEMORY
658     GetMemoryHandle(
659         __in USHORT Offset
660         )
661     {
662         ULONG_PTR handle;
663 
664         //
665         // The offset into this object must be self relative.
666         //
667         ASSERT(*((PUSHORT) WDF_PTR_ADD_OFFSET(this, Offset)) == Offset);
668 
669         handle = (ULONG_PTR) WDF_PTR_ADD_OFFSET(this, Offset);
670 
671         //
672         // Make sure initial value has the flag bits cleared.
673         //
674         ASSERT((handle  & FxHandleFlagMask) == 0);
675 
676         //
677         // We always apply the mask.
678         //
679         handle = handle ^ FxHandleValueMask;
680 
681         //
682         // Make sure final value (before we set the flag) has the flag bits
683         // cleared.
684         //
685         ASSERT((handle  & FxHandleFlagMask) == 0);
686 
687         //
688         // This handle is an offset
689         handle |= FxHandleFlagIsOffset;
690 
691         return (WDFMEMORY) handle;
692     }
693 
694     _Must_inspect_result_
695     virtual
696     NTSTATUS
697     QueryInterface(
698         __in FxQueryInterfaceParams* Params
699         );
700 
701 public:
702     // Factory functions to create FxRequest* objects
703     _Must_inspect_result_
704     static
705     NTSTATUS
706     _CreateForPackage(
707         __in CfxDevice* Device,
708         __in PWDF_OBJECT_ATTRIBUTES RequestAttributes,
709         __in MdIrp Irp,
710         __deref_out FxRequest** Request
711         );
712 
713     _Must_inspect_result_
714     static
715     NTSTATUS
716     _Create(
717         __in PFX_DRIVER_GLOBALS FxDriverGlobals,
718         __in_opt PWDF_OBJECT_ATTRIBUTES RequestAttributes,
719         __in_opt MdIrp Irp,
720         __in_opt FxIoTarget* Target,
721         __in FxRequestIrpOwnership Ownership,
722         __in FxRequestConstructorCaller Caller,
723         __deref_out FxRequest** Request
724         );
725 
726     _Must_inspect_result_
727     FxIoQueue*
728     GetCurrentQueue(
729         VOID
730         )
731     {
732         if(m_Completed) {
733             return NULL;
734         }
735 
736         return m_IoQueue;
737     }
738 
739     FxRequestCompletionState
740     SetCompletionState(
741         __in FxRequestCompletionState NewState
742         );
743 
744     VOID
745     __inline
746     SetStatus(
747         __in NTSTATUS Status
748         )
749     {
750         m_Irp.SetStatus(Status);
751     }
752 
753     NTSTATUS
754     SetInformation(
755         __in ULONG_PTR Information
756         );
757 
758     ULONG_PTR
759     GetInformation(
760         VOID
761         );
762 
763     KPROCESSOR_MODE
764     GetRequestorMode(
765         VOID
766         );
767 
768     __inline
769     NTSTATUS
770     Complete(
771         __in NTSTATUS Status
772     )
773     {
774         CfxDevice* const fxDevice = GetDevice();
775 
776         //
777         // Complete the current request object. Can be called directly
778         // by the FxIoQueue to complete a request.
779         //
780         // When an FxRequest is completed, it is marked as completed,
781         // removed from any CSQ it may be a member of, and any registered
782         // callback functions are called. Then the NT IRP is completed,
783         // and the reference count on the object due to the callback routine
784         // is released if a callback routine was specified.
785         //
786         // Completing a request object can cause its reference
787         // count to go to zero, thus deleting it. So the caller
788         // must either reference it explicitly, or not touch it
789         // any more after calling complete.
790         //
791 
792         DoTraceLevelMessage(
793             GetDriverGlobals(), TRACE_LEVEL_VERBOSE, TRACINGREQUEST,
794             "Completing WDFREQUEST 0x%p for IRP 0x%p with "
795             "Information 0x%I64x, %!STATUS!",
796             GetHandle(), m_Irp.GetIrp(), m_Irp.GetInformation(), Status);
797 
798         if (fxDevice != NULL) {
799             SetPriorityBoost(fxDevice->GetDefaultPriorityBoost());
800         }
801         else {
802             SetPriorityBoost(0);
803         }
804 
805         return CompleteInternal(Status);
806     }
807 
808     __inline
809     NTSTATUS
810     CompleteWithInformation(
811         __in NTSTATUS Status,
812         __in ULONG_PTR Information
813         )
814     {
815         //
816         // Complete the request object. If the status is success, get the
817         // priority boost for the owning device type, and complete the request.
818         //
819         m_Irp.SetInformation(Information);
820         return Complete(Status);
821     }
822 
823     __inline
824     NTSTATUS
825     CompleteWithPriority(
826         __in NTSTATUS Status,
827         __in CCHAR PriorityBoost
828         )
829     {
830         //
831         // Complete the current request object. Can be called directly
832         // by the FxIoQueue to complete a request.
833         //
834         // When an FxRequest is completed, it is marked as completed,
835         // removed from any CSQ it may be a member of, and any registered
836         // callback functions are called. Then the NT IRP is completed,
837         // and the reference count on the object due to the callback routine
838         // is released if a callback routine was specified.
839         //
840         // Completing a request object can cause its reference
841         // count to go to zero, thus deleting it. So the caller
842         // must either reference it explicitly, or not touch it
843         // any more after calling complete.
844         //
845 
846         DoTraceLevelMessage(
847             GetDriverGlobals(), TRACE_LEVEL_VERBOSE, TRACINGREQUEST,
848             "Completing WDFREQUEST 0x%p for IRP 0x%p with "
849             "Information 0x%I64x, %!STATUS!",
850             GetHandle(), m_Irp.GetIrp(), m_Irp.GetInformation(), Status);
851 
852         SetPriorityBoost(PriorityBoost);
853         return CompleteInternal(Status);
854     }
855 
856     // Do not specify argument names
857     FX_DECLARE_VF_FUNCTION(
858     VOID,
859     VerifyPreProcessSendAndForget
860     );
861 
862     VOID
863     PreProcessSendAndForget(
864         VOID
865         );
866 
867     VOID
868     PostProcessSendAndForget(
869         VOID
870         );
871 
872     NTSTATUS
873     GetStatus(
874         VOID
875         );
876 
877     _Must_inspect_result_
878     NTSTATUS
879     GetParameters(
880         __out PWDF_REQUEST_PARAMETERS Parameters
881         );
882 
883     _Must_inspect_result_
884     NTSTATUS
885     GetMemoryObject(
886         __deref_out IFxMemory** Memory,
887         __out PVOID* Buffer,
888         __out size_t* Length
889         );
890 
891     _Must_inspect_result_
892     NTSTATUS
893     GetMdl(
894         __out PMDL *pMdl
895         );
896 
897     _Must_inspect_result_
898     NTSTATUS
899     GetDeviceControlOutputMemoryObject(
900         __deref_out IFxMemory** MemoryObject,
901         __out PVOID* Buffer,
902         __out size_t* Length
903         );
904 
905     _Must_inspect_result_
906     NTSTATUS
907     GetDeviceControlOutputMdl(
908         __out PMDL *pMdl
909         );
910 
911     // Do not specify argument names
912     FX_DECLARE_VF_FUNCTION(
913     NTSTATUS,
914     VerifyProbeAndLock
915     );
916 
917     _Must_inspect_result_
918     NTSTATUS
919     ProbeAndLockForRead(
920         __in  PVOID Buffer,
921         __in  ULONG Length,
922         __deref_out FxRequestMemory** pMemoryObject
923         );
924 
925     _Must_inspect_result_
926     NTSTATUS
927     ProbeAndLockForWrite(
928         __in  PVOID Buffer,
929         __in  ULONG Length,
930         __deref_out FxRequestMemory** pMemoryObject
931         );
932 
933 #if (FX_CORE_MODE == FX_CORE_USER_MODE)
934     _Must_inspect_result_
935     NTSTATUS
936     Impersonate(
937         _In_ SECURITY_IMPERSONATION_LEVEL ImpersonationLevel,
938         _In_ PFN_WDF_REQUEST_IMPERSONATE EvtRequestImpersonate,
939         _In_opt_ PVOID Context
940         );
941 #endif
942 
943     VOID
944     SetImpersonationFlags(
945         _In_ ULONG Flags
946         )
947     {
948 #if (FX_CORE_MODE == FX_CORE_USER_MODE)
949         GetSubmitFxIrp()->GetIoIrp()->SetFlagsForNextStackLocation(
950                 FxIrpStackFlagsFromSendFlags(Flags));
951 #else
952         UNREFERENCED_PARAMETER(Flags);
953 #endif
954     }
955 
956     FxIrp*
957     GetFxIrp(
958         VOID
959         )
960     {
961         return &m_Irp;
962     }
963 
964     __inline
965     FxIoQueue*
966     GetIoQueue(
967         VOID
968         )
969     {
970         return m_IoQueue;
971     }
972 
973     _Must_inspect_result_
974     NTSTATUS
975     GetIrp(
976         __deref_out MdIrp* ppIrp
977         )
978     {
979         if (GetDriverGlobals()->FxVerifierIO) {
980             NTSTATUS status;
981             KIRQL irql;
982 
983             Lock(&irql);
984 
985             status = VerifyRequestIsNotCompleted(GetDriverGlobals());
986             if (!NT_SUCCESS(status)) {
987                 *ppIrp = NULL;
988                 status = STATUS_INVALID_DEVICE_REQUEST;
989             }
990             else {
991                 *ppIrp = m_Irp.GetIrp();
992             }
993 
994             Unlock(irql);
995 
996             return status;
997         }
998         else {
999             *ppIrp = m_Irp.GetIrp();
1000             return STATUS_SUCCESS;
1001         }
1002     }
1003 
1004     //
1005     // Return the FxFileObject if associated with this request
1006     //
1007     _Must_inspect_result_
1008     NTSTATUS
1009     GetFileObject(
1010         __deref_out_opt FxFileObject** pFileObject
1011         );
1012 
1013     //
1014     // Get the IoStack location of the request.
1015     //
1016     // Since this returns the pointer to the underlying IRP
1017     // IO_STACK_LOCATION, it can not be called in a situation
1018     // which the request is completed out from underneath us.
1019     //
1020     // Note: Must implemention a version for the drivers use.
1021     //       Must interact with completion events from the
1022     //       driver due to cancel as well.
1023     //
1024     //
1025     PIO_STACK_LOCATION
1026     GetCurrentIrpStackLocation(
1027         VOID
1028         )
1029     {
1030         if (GetDriverGlobals()->FxVerifierIO) {
1031             PIO_STACK_LOCATION ios;
1032             KIRQL irql;
1033             NTSTATUS status;
1034 
1035             Lock(&irql);
1036             status = VerifyRequestIsNotCompleted(GetDriverGlobals());
1037             if (!NT_SUCCESS(status)) {
1038                 ios = NULL;
1039             }
1040             else {
1041                 ios = m_Irp.GetCurrentIrpStackLocation();
1042             }
1043             Unlock(irql);
1044 
1045             return ios;
1046         }
1047         else {
1048             return m_Irp.GetCurrentIrpStackLocation();
1049         }
1050     }
1051 
1052     //
1053     // The following functions are to support use of
1054     // the Cancel Safe FxIrpQueue.
1055     //
1056 
1057     // Do not specify argument names
1058     FX_DECLARE_VF_FUNCTION_P1(
1059     NTSTATUS,
1060     VerifyInsertIrpQueue,
1061         _In_ FxIrpQueue*
1062         );
1063 
1064     //
1065     // Insert the request in the supplied FxIrpQueue
1066     // and associated it with it.
1067     //
1068     _Must_inspect_result_
1069     NTSTATUS
1070     InsertTailIrpQueue(
1071         __in FxIrpQueue* IrpQueue,
1072         __out_opt ULONG*      pRequestCount
1073         );
1074 
1075     _Must_inspect_result_
1076     NTSTATUS
1077     InsertHeadIrpQueue(
1078         __in FxIrpQueue* IrpQueue,
1079         __out_opt ULONG*      pRequestCount
1080         );
1081 
1082     //
1083     // Remove it from the FxIrpQueue it is associated with.
1084     //
1085     // Returns STATUS_CANCELLED if the cancel routine has
1086     // fired and removed it from the queue first.
1087     //
1088     //
1089     _Must_inspect_result_
1090     NTSTATUS
1091     RemoveFromIrpQueue(
1092         __in FxIrpQueue* IrpQueue
1093         );
1094 
1095     //
1096     // Mark that this request is no longer on the IrpQueue
1097     //
1098     __inline
1099     VOID
1100     MarkRemovedFromIrpQueue(
1101         VOID
1102         )
1103     {
1104         m_IrpQueue = NULL;
1105         return;
1106     }
1107 
1108     //
1109     // Return the FxRequest's CsqContext address
1110     //
1111     __inline
1112     PMdIoCsqIrpContext
1113     GetCsqContext(
1114         VOID
1115         )
1116     {
1117         return &m_CsqContext;
1118     }
1119 
1120 
1121     //
1122     // Function to return an FxRequest from an FxIrpQueue
1123     //
1124     _Must_inspect_result_
1125     static
1126     FxRequest*
1127     GetNextRequest(
1128         __in FxIrpQueue*  IrpQueue
1129         );
1130 
1131     _Must_inspect_result_
1132     static
1133     NTSTATUS
1134     GetNextRequest(
1135         __in FxIrpQueue*        IrpQueue,
1136         __in_opt MdFileObject   FileObject,
1137         __in_opt FxRequest*     TagRequest,
1138         __deref_out FxRequest** ppOutRequest
1139         );
1140 
1141     //
1142     // Allow peeking at requests in the IrpQueue
1143     //
1144     _Must_inspect_result_
1145     static
1146     NTSTATUS
1147     PeekRequest(
1148         __in FxIrpQueue*         IrpQueue,
1149         __in_opt FxRequest*      TagRequest,
1150         __in_opt MdFileObject    FileObject,
1151         __out_opt PWDF_REQUEST_PARAMETERS Parameters,
1152         __deref_out FxRequest**  ppOutRequest
1153         );
1154 
1155     //
1156     // Internal function to retrieve the FxRequest
1157     // structure from a pointer to its CsqContext
1158     // member.
1159     //
1160     __inline
1161     static
1162     FxRequest*
1163     RetrieveFromCsqContext(
1164         __in PMdIoCsqIrpContext pCsqContext
1165         )
1166     {
1167         return CONTAINING_RECORD(pCsqContext, FxRequest, m_CsqContext);
1168     }
1169 
1170 
1171     __inline
1172     BOOLEAN
1173     IsInIrpQueue(
1174         __in FxIrpQueue*          pIrpQueue
1175         )
1176     {
1177         return pIrpQueue->IsIrpInQueue(GetCsqContext());
1178     }
1179 
1180 
1181     // Do not specify argument names
1182     FX_DECLARE_VF_FUNCTION_P1(
1183     NTSTATUS,
1184     VerifyStopAcknowledge,
1185         _In_ BOOLEAN
1186         );
1187 
1188     VOID
1189     StopAcknowledge(
1190         __in BOOLEAN Requeue
1191         );
1192 
1193     __inline
1194     BOOLEAN
1195     IsPowerStopAcknowledged(
1196         VOID
1197         )
1198     {
1199         return ((m_PowerStopState == FxRequestPowerStopAcknowledged)
1200                 ||
1201                 (m_PowerStopState == FxRequestPowerStopAcknowledgedWithRequeue))? TRUE : FALSE;
1202     }
1203 
1204     __inline
1205     BOOLEAN
1206     IsPowerStopAcknowledgedWithRequeue(
1207         VOID
1208         )
1209     {
1210         return (m_PowerStopState == FxRequestPowerStopAcknowledgedWithRequeue) ? TRUE : FALSE;
1211     }
1212 
1213     VOID
1214     ClearPowerStopState(
1215         VOID
1216         )
1217     {
1218         m_PowerStopState = FxRequestPowerStopUnknown;
1219     }
1220 
1221     // Do not specify argument names
1222     FX_DECLARE_VF_FUNCTION(
1223     VOID,
1224     VerifierBreakpoint_RequestEarlyDisposeDeferred
1225     );
1226 
1227     // Do not specify argument names
1228     FX_DECLARE_VF_FUNCTION(
1229     NTSTATUS,
1230     VerifyRequestIsDriverOwned
1231     );
1232 
1233     // Do not specify argument names
1234     FX_DECLARE_VF_FUNCTION(
1235     NTSTATUS,
1236     VerifyRequestIsCancelable
1237     );
1238 
1239     // Do not specify argument names
1240     FX_DECLARE_VF_FUNCTION(
1241     NTSTATUS,
1242     VerifyRequestIsNotCancelable
1243     );
1244 
1245     // Do not specify argument names
1246     FX_DECLARE_VF_FUNCTION(
1247     NTSTATUS,
1248     VerifyRequestIsInCallerContext
1249     );
1250 
1251     // Do not specify argument names
1252     FX_DECLARE_VF_FUNCTION(
1253     NTSTATUS,
1254     VerifyRequestIsInEvtIoStopContext
1255     );
1256 
1257     // Do not specify argument names
1258     FX_DECLARE_VF_FUNCTION(
1259     NTSTATUS,
1260     VerifyRequestIsNotCompleted
1261     );
1262 
1263     // Do not specify argument names
1264     FX_DECLARE_VF_FUNCTION(
1265     NTSTATUS,
1266     VerifyRequestIsTagRequest
1267     );
1268 
1269     // Do not specify argument names
1270     FX_DECLARE_VF_FUNCTION(
1271     NTSTATUS,
1272     VerifyRequestIsAllocatedFromIo
1273     );
1274 
1275     // Do not specify argument names
1276     FX_DECLARE_VF_FUNCTION(
1277     NTSTATUS,
1278     VerifyRequestIsCurrentStackValid
1279     );
1280 
1281     // Do not specify argument names
1282     FX_DECLARE_VF_FUNCTION(
1283     NTSTATUS,
1284     VerifyRequestCanBeCompleted
1285     );
1286 
1287     VOID
1288     FreeRequest(
1289         VOID
1290         );
1291 
1292     __inline
1293     VOID
1294     ClearFieldsForReuse(
1295         VOID
1296         )
1297     {
1298         m_SystemBuffer.ClearBufferMdl();
1299         m_OutputBuffer.ClearBufferMdl();
1300 
1301         ASSERT(m_ForwardRequestToParent == FALSE);
1302 
1303         FxRequestBase::ClearFieldsForReuse(); // __super call
1304     }
1305 
1306     virtual
1307     ULONG
1308     Release(
1309         __in PVOID Tag,
1310         __in LONG Line,
1311         __in_opt PSTR File
1312     );
1313 
1314     __inline
1315     BOOLEAN
1316     IsRequestForwardedToParent(
1317         VOID
1318         )
1319     {
1320         return m_ForwardRequestToParent;
1321     }
1322 
1323 private:
1324     FX_DECLARE_VF_FUNCTION_P1(
1325         VOID,
1326         VerifyCompleteInternal,
1327             _In_ NTSTATUS
1328             );
1329 
1330     NTSTATUS
1331     CompleteInternalReserved(
1332         __in NTSTATUS Status,
1333         __in CCHAR PriorityBoost
1334         );
1335 
1336     NTSTATUS
1337     CompleteInternal(
1338         __in NTSTATUS Status
1339         );
1340 
1341     VOID
1342     PostProcessCompletion(
1343         __in FxRequestCompletionState State,
1344         __in FxIoQueue* Queue
1345         );
1346 
1347     VOID
1348     PostProcessCompletionForReserved(
1349         __in FxRequestCompletionState State,
1350         __in FxIoQueue* Queue
1351         );
1352 
1353     VOID
1354     PreProcessCompletionForDriverRequest(
1355         __in FxRequestCompletionState State,
1356         __in FxIoQueue* Queue
1357         );
1358 
1359     VOID
1360     PostProcessCompletionForDriverRequest(
1361         __in FxRequestCompletionState State,
1362         __in FxIoQueue* Queue
1363         );
1364 
1365     static
1366     VOID
1367     CheckAssumptions(
1368         VOID
1369         );
1370 
1371     VOID
1372     AssignMemoryBuffers(
1373         __in WDF_DEVICE_IO_TYPE IoType
1374     )
1375 {
1376 
1377     switch (m_Irp.GetMajorFunction()) {
1378     case IRP_MJ_DEVICE_CONTROL:
1379     case IRP_MJ_INTERNAL_DEVICE_CONTROL:
1380 
1381 
1382 
1383 
1384 
1385 
1386 
1387 
1388 
1389       switch (m_Irp.GetParameterIoctlCodeBufferMethod()) {
1390       case METHOD_BUFFERED:
1391           //
1392           // Set the buffer in the memory interface. For kernel mode,
1393           // GetOutputBuffer is same as GetSystemBuffer, but for user-mode,
1394           // host provides separate buffers, so that input buffer can only be
1395           // read, and output buffer can only be written to.
1396           //
1397           m_SystemBuffer.SetBuffer(m_Irp.GetSystemBuffer());
1398           m_OutputBuffer.SetBuffer(m_Irp.GetOutputBuffer());
1399           break;
1400 
1401       case METHOD_IN_DIRECT:
1402           //
1403           // InputBuffer is in SystemBuffer
1404           // OutputBuffer is in MdlAddress with read access
1405           //
1406           KMDF_ONLY_CODE_PATH_ASSERT();
1407           m_SystemBuffer.SetBuffer(m_Irp.GetSystemBuffer());
1408           break;
1409 
1410       case METHOD_OUT_DIRECT:
1411           //
1412           // InputBuffer is in SystemBuffer
1413           // OutputBuffer is in MdlAddress with write access
1414           //
1415           KMDF_ONLY_CODE_PATH_ASSERT();
1416           m_SystemBuffer.SetBuffer(m_Irp.GetSystemBuffer());
1417           break;
1418 
1419       case METHOD_NEITHER:
1420           //
1421           // Internal device controls are kernel mode to kernel mode, and deal
1422           // with direct unmapped pointers.
1423           //
1424           // In addition, a normal device control with
1425           // RequestorMode == KernelMode is also treated as kernel mode
1426           // to kernel mode since the I/O Manager will not generate requests
1427           // with this setting from a user mode request.
1428           //
1429           KMDF_ONLY_CODE_PATH_ASSERT();
1430           if (m_Irp.GetMajorFunction() == IRP_MJ_INTERNAL_DEVICE_CONTROL ||
1431               (m_Irp.GetRequestorMode() == KernelMode)) {
1432               m_SystemBuffer.SetBuffer(
1433                   m_Irp.GetParameterIoctlType3InputBuffer()
1434                   );
1435               m_OutputBuffer.SetBuffer(m_Irp.GetUserBuffer());
1436           }
1437           else {
1438               return;
1439           }
1440           break;
1441       }
1442       break;
1443 
1444       case IRP_MJ_READ:
1445       case IRP_MJ_WRITE:
1446           switch (IoType) {
1447           case WdfDeviceIoBuffered:
1448               m_SystemBuffer.SetBuffer(m_Irp.GetSystemBuffer());
1449               break;
1450 
1451           case WdfDeviceIoNeither:
1452               KMDF_ONLY_CODE_PATH_ASSERT();
1453               if (m_Irp.GetRequestorMode() == KernelMode) {
1454                    m_SystemBuffer.SetBuffer(m_Irp.GetUserBuffer());
1455               }
1456               else {
1457                   return;
1458               }
1459           break;
1460 
1461           default:
1462               return;
1463           }
1464           break;
1465 
1466       default:
1467           return;
1468       }
1469 
1470       if (m_SystemBuffer.IsBufferSet()) {
1471           m_RequestBaseStaticFlags |= FxRequestBaseStaticSystemBufferValid;
1472       }
1473       if (m_OutputBuffer.IsBufferSet()) {
1474           m_RequestBaseStaticFlags |= FxRequestBaseStaticOutputBufferValid;
1475       }
1476 }
1477 
1478 
1479 public:
1480     __inline
1481     VOID
1482     SetInternalContext(
1483         PVOID Context
1484         )
1485     {
1486         ASSERT(NULL == m_InternalContext);
1487         m_InternalContext = Context;
1488     }
1489 
1490     __inline
1491     PVOID
1492     GetInternalContext(
1493         VOID
1494         )
1495     {
1496         PVOID context;
1497 
1498         context = m_InternalContext;
1499         m_InternalContext = NULL;
1500 
1501         return context;
1502     }
1503 };
1504 
1505 class FxRequestFromLookaside : public FxRequest {
1506 
1507 public:
1508     FxRequestFromLookaside(
1509         __in CfxDevice* Device,
1510         __in MdIrp Irp
1511         );
1512 
1513     PVOID
1514     operator new(
1515         __in size_t Size,
1516         __in CfxDevice* Device,
1517         __in_opt PWDF_OBJECT_ATTRIBUTES Attributes
1518         );
1519 
1520 protected:
1521     //
1522     // FxObject override
1523     //
1524     VOID
1525     SelfDestruct(
1526         VOID
1527         );
1528 };
1529 
1530 #endif // _FXREQUEST_H_
1531