1 // 2 // Copyright (C) Microsoft. All rights reserved. 3 // 4 #ifndef _FXSELFMANAGEDIOSTATEMACHINE_H_ 5 #define _FXSELFMANAGEDIOSTATEMACHINE_H_ 6 7 // 8 // This is a magical number based on inspection. If the queue overflows, 9 // it is OK to increase these numbers without fear of either dependencies or 10 // weird side affects. 11 // 12 const UCHAR FxSelfManagedIoEventQueueDepth = 8; 13 14 enum FxSelfManagedIoEvents { 15 SelfManagedIoEventInvalid = 0x00, 16 SelfManagedIoEventStart = 0x01, 17 SelfManagedIoEventCleanup = 0x02, 18 SelfManagedIoEventSuspend = 0x04, 19 SelfManagedIoEventFlush = 0x08, 20 SelfManagedIoEventNull = 0xFF, 21 }; 22 23 enum FxSelfManagedIoStates { 24 FxSelfManagedIoInvalid = 0, 25 FxSelfManagedIoCreated, 26 FxSelfManagedIoInit, 27 FxSelfManagedIoInitFailed, 28 FxSelfManagedIoStarted, 29 FxSelfManagedIoSuspending, 30 FxSelfManagedIoStopped, 31 FxSelfManagedIoRestarting, 32 FxSelfManagedIoFailed, 33 FxSelfManagedIoFlushing, 34 FxSelfManagedIoFlushed, 35 FxSelfManagedIoCleanup, 36 FxSelfManagedIoFinal, 37 FxSelfManagedIoMax, 38 }; 39 40 typedef 41 _Must_inspect_result_ 42 FxSelfManagedIoStates 43 (*PFN_SELF_MANAGED_IO_STATE_ENTRY_FUNCTION)( 44 __in FxSelfManagedIoMachine*, 45 __out PNTSTATUS Status 46 ); 47 48 struct FxSelfManagedIoTargetState { 49 FxSelfManagedIoEvents SelfManagedIoEvent; 50 51 FxSelfManagedIoStates SelfManagedIoState; 52 53 #if FX_SUPER_DBG 54 BOOLEAN EventDebugged; 55 #endif 56 }; 57 58 // 59 // This type of union is done so that we can 60 // 1) shrink the array element to the smallest size possible 61 // 2) keep types within the structure so we can dump it in the debugger 62 // 63 union FxSelfManagedIoMachineEventHistory { 64 struct { 65 FxSelfManagedIoEvents Event1 : 8; 66 FxSelfManagedIoEvents Event2 : 8; 67 FxSelfManagedIoEvents Event3 : 8; 68 FxSelfManagedIoEvents Event4 : 8; 69 FxSelfManagedIoEvents Event5 : 8; 70 FxSelfManagedIoEvents Event6 : 8; 71 FxSelfManagedIoEvents Event7 : 8; 72 FxSelfManagedIoEvents Event8 : 8; 73 } E; 74 75 UCHAR History[FxSelfManagedIoEventQueueDepth]; 76 }; 77 78 // 79 // This type of union is done so that we can 80 // 1) shrink the array element to the smallest size possible 81 // 2) keep types within the structure so we can dump it in the debugger 82 // 83 union FxSelfManagedIoMachineStateHistory { 84 struct { 85 FxSelfManagedIoStates State1 : 8; 86 FxSelfManagedIoStates State2 : 8; 87 FxSelfManagedIoStates State3 : 8; 88 FxSelfManagedIoStates State4 : 8; 89 FxSelfManagedIoStates State5 : 8; 90 FxSelfManagedIoStates State6 : 8; 91 FxSelfManagedIoStates State7 : 8; 92 FxSelfManagedIoStates State8 : 8; 93 } S; 94 95 UCHAR History[FxSelfManagedIoEventQueueDepth]; 96 }; 97 98 struct FxSelfManagedIoStateTable { 99 PFN_SELF_MANAGED_IO_STATE_ENTRY_FUNCTION StateFunc; 100 101 const FxSelfManagedIoTargetState* TargetStates; 102 103 ULONG TargetStatesCount; 104 }; 105 106 class FxSelfManagedIoMachine : public FxStump { 107 108 public: 109 FxSelfManagedIoMachine( 110 __in FxPkgPnp* PkgPnp 111 ); 112 113 static 114 NTSTATUS 115 _CreateAndInit( 116 __deref_out FxSelfManagedIoMachine** SelfManagedIoMachine, 117 __in FxPkgPnp* PkgPnp 118 ); 119 120 121 // 122 // Sets event callbacks 123 // 124 VOID 125 InitializeMachine( 126 __in PWDF_PNPPOWER_EVENT_CALLBACKS Callbacks 127 ); 128 129 _Must_inspect_result_ 130 NTSTATUS 131 Start( 132 VOID 133 ) 134 { 135 return ProcessEvent(SelfManagedIoEventStart); 136 } 137 138 _Must_inspect_result_ 139 NTSTATUS 140 Suspend( 141 VOID 142 ) 143 { 144 return ProcessEvent(SelfManagedIoEventSuspend); 145 } 146 147 VOID 148 Flush( 149 VOID 150 ) 151 { 152 (void) ProcessEvent(SelfManagedIoEventFlush); 153 } 154 155 VOID 156 Cleanup( 157 VOID 158 ) 159 { 160 (void) ProcessEvent(SelfManagedIoEventCleanup); 161 } 162 163 protected: 164 _Must_inspect_result_ 165 NTSTATUS 166 ProcessEvent( 167 __in FxSelfManagedIoEvents Event 168 ); 169 170 static 171 FxSelfManagedIoStates 172 Init( 173 __in FxSelfManagedIoMachine* This, 174 __out PNTSTATUS Status 175 ); 176 177 static 178 FxSelfManagedIoStates 179 Suspending( 180 __in FxSelfManagedIoMachine* This, 181 __out PNTSTATUS Status 182 ); 183 184 static 185 FxSelfManagedIoStates 186 Restarting( 187 __in FxSelfManagedIoMachine* This, 188 __out PNTSTATUS Status 189 ); 190 191 static 192 FxSelfManagedIoStates 193 Flushing( 194 __in FxSelfManagedIoMachine* This, 195 __out PNTSTATUS Status 196 ); 197 198 static 199 FxSelfManagedIoStates 200 Cleanup( 201 __in FxSelfManagedIoMachine* This, 202 __out PNTSTATUS Status 203 ); 204 205 WDFDEVICE 206 GetDeviceHandle( 207 VOID 208 ); 209 210 public: 211 FxPnpDeviceSelfManagedIoCleanup m_DeviceSelfManagedIoCleanup; 212 FxPnpDeviceSelfManagedIoFlush m_DeviceSelfManagedIoFlush; 213 FxPnpDeviceSelfManagedIoInit m_DeviceSelfManagedIoInit; 214 FxPnpDeviceSelfManagedIoSuspend m_DeviceSelfManagedIoSuspend; 215 FxPnpDeviceSelfManagedIoRestart m_DeviceSelfManagedIoRestart; 216 217 protected: 218 FxWaitLockInternal m_StateMachineLock; 219 220 FxPkgPnp* m_PkgPnp; 221 222 // uses FxSelfManagedIoStates values 223 BYTE m_CurrentState; 224 225 UCHAR m_EventHistoryIndex; 226 227 UCHAR m_StateHistoryIndex; 228 229 // extra padded byte is put in here by the compiler 230 231 FxSelfManagedIoMachineEventHistory m_Events; 232 233 FxSelfManagedIoMachineStateHistory m_States; 234 235 static const FxSelfManagedIoStateTable m_StateTable[]; 236 237 static const FxSelfManagedIoTargetState m_CreatedStates[]; 238 static const FxSelfManagedIoTargetState m_InitFailedStates[]; 239 static const FxSelfManagedIoTargetState m_StartedStates[]; 240 static const FxSelfManagedIoTargetState m_StoppedStates[]; 241 static const FxSelfManagedIoTargetState m_FailedStates[]; 242 static const FxSelfManagedIoTargetState m_FlushedStates[]; 243 }; 244 245 #endif // _FXSELFMANAGEDIOSTATEMACHINE_H_ 246