1 /*++
2 
3 Copyright (c) Microsoft Corporation
4 
5 Module Name:
6 
7     FxWorkItem.hpp
8 
9 Abstract:
10 
11     This module implements a frameworks managed WorkItem that
12     can synchrononize with driver frameworks object locks.
13 
14 Author:
15 
16 
17 
18 
19 Environment:
20 
21     Both kernel and user mode
22 
23 Revision History:
24 
25 
26 --*/
27 
28 #ifndef _FXWORKITEM_H_
29 #define _FXWORKITEM_H_
30 
31 //
32 // Driver Frameworks WorkItem Design:
33 //
34 // The driver frameworks provides an optional WorkItem wrapper object that allows
35 // a reference counted WorkItem object to be created that can synchronize
36 // automatically with certain frameworks objects.
37 //
38 // This provides automatic synchronization between the WorkItems execution, and the
39 // frameworks objects event callbacks into the device driver.
40 //
41 // The WDFWORKITEM object is designed to be re-useable, in which it can be re-linked
42 // into the WorkItem queue after firing.
43 //
44 // Calling GetWorkItemPtr returns the WDM IoWorkItem.
45 //
46 
47 class FxWorkItem : public FxNonPagedObject {
48 
49 private:
50     //
51     // WDM work item.
52     //
53     MxWorkItem         m_WorkItem;
54 
55     // Ensures only one of either Delete or Cleanup runsdown the object
56     BOOLEAN            m_RunningDown;
57 
58     //
59     // If this is set, a WorkItem has been enqueued
60     //
61     BOOLEAN            m_Enqueued;
62 
63     //
64     // This count is used to prevent the object from being deleted if
65     // one worker thread is preempted right after we drop the lock to call
66     // the client callback and another workitem gets queued and runs
67     // to completion and signals the event.
68     //
69     ULONG           m_WorkItemRunningCount;
70 
71     //
72     // This is the Framework object who is associated with the work
73     // item if supplied
74     //
75     FxObject*          m_Object;
76 
77     //
78     // This is the callback lock for the object this WorkItem will
79     // synchronize with
80     //
81     FxCallbackLock*    m_CallbackLock;
82 
83     //
84     // This is the object whose reference count actually controls
85     // the lifetime of the m_CallbackLock
86     //
87     FxObject*          m_CallbackLockObject;
88 
89     //
90     // This is the user supplied callback function and context
91     //
92     PFN_WDF_WORKITEM   m_Callback;
93 
94     //
95     // This event is signaled when the workitem is done processing
96     // an Enqueue request.
97     //
98     FxCREvent          m_WorkItemCompleted;
99 
100     //
101     // This is a pointer to thread object that invoked our workitem
102     // callback. This value will be used to avoid deadlock when we try
103     // to flush the workitem.
104     //
105     MxThread          m_WorkItemThread;
106 
107 public:
108 
109     static
110     _Must_inspect_result_
111     NTSTATUS
112     _Create(
113         __in PFX_DRIVER_GLOBALS FxDriverGlobals,
114         __in PWDF_WORKITEM_CONFIG Config,
115         __in PWDF_OBJECT_ATTRIBUTES Attributes,
116         __in FxObject* ParentObject,
117         __out WDFWORKITEM* WorkItem
118         );
119 
120 /*++
121 
122 Routine Description:
123 
124     Construct an FxWorkItem
125 
126 Arguments:
127 
128 Returns:
129 
130     NTSTATUS
131 
132 --*/
133 
134     FxWorkItem(
135         __in PFX_DRIVER_GLOBALS FxDriverGlobals
136         );
137 
138     virtual
139     ~FxWorkItem(
140        );
141 
142     MdWorkItem
GetWorkItemPtr()143     GetWorkItemPtr() {
144         return m_WorkItem.GetWorkItem();
145     }
146 
147 /*++
148 
149 Routine Description:
150 
151     Initialize the WorkItem using either the caller supplied WorkItem
152     struct, or if NULL, our own internally allocated one
153 
154 Arguments:
155 
156 Returns:
157 
158     NTSTATUS
159 
160 --*/
161     _Must_inspect_result_
162     NTSTATUS
163     Initialize(
164         __in PWDF_OBJECT_ATTRIBUTES Attributes,
165         __in PWDF_WORKITEM_CONFIG Config,
166         __in FxObject* ParentObject,
167         __out WDFWORKITEM* WorkItem
168         );
169 
170     virtual
171     BOOLEAN
172     Dispose(
173         VOID
174         );
175 
176     VOID
177     Enqueue(
178         );
179 
180     WDFOBJECT
GetAssociatedObject()181     GetAssociatedObject(
182         )
183     {
184         if( m_Object != NULL ) {
185             return m_Object->GetObjectHandle();
186         }
187         else {
188             return NULL;
189         }
190     }
191 
192     WDFWORKITEM
GetHandle(VOID)193     GetHandle(
194         VOID
195         )
196     {
197         return (WDFWORKITEM) GetObjectHandle();
198     }
199 
200 private:
201 
202     //
203     // Called from Dispose, or cleanup list to perform final flushing of any
204     // outstanding DPC's and dereferencing of objects.
205     //
206     VOID
207     FlushAndRundown(
208         );
209 
210     VOID
211     WorkItemHandler(
212         );
213 
214     static
215     MX_WORKITEM_ROUTINE
216     WorkItemThunk;
217 
218     VOID
219     WaitForSignal(
220         VOID
221         );
222 
223 public:
224     VOID
225     FlushAndWait(
226         VOID
227         );
228 };
229 
230 #endif // _FXWORKITEM_H_
231 
232