1 /*++ 2 3 Copyright (c) Microsoft Corporation 4 5 Module Name: 6 7 FxDriver.hpp 8 9 Abstract: 10 11 This is the definition of the FxDriver object. 12 13 Author: 14 15 16 17 Environment: 18 19 Both kernel and user mode 20 21 Revision History: 22 23 --*/ 24 25 #ifndef _FXDRIVER_H_ 26 #define _FXDRIVER_H_ 27 28 #include "fxdrivercallbacks.hpp" 29 30 31 // 32 // Unique value to retrieve the FxDriver* from the PDEVICE_OBJECT. 33 // 34 #define FX_TRACE_INFO_ID (FxDriver::_GetTraceInfoExtension) 35 36 // 37 // Structure to hold WMI Callback info 38 // 39 struct FxTraceInfo { 40 MdDriverObject DriverObject; 41 PFN_WDF_TRACE_CALLBACK Callback; 42 PVOID Context; 43 }; 44 45 // 46 // Unique value to retrieve the FxDriver* from the MdDriverObject. Use a value 47 // that is not exposed to the driver writer through the dispatch table or WDM. 48 // 49 #define FX_DRIVER_ID ((PVOID)FxDriver::GetFxDriver) 50 51 // 52 // The following are support classes for FxDriver 53 // 54 class FxDriver : public FxNonPagedObject, public IFxHasCallbacks 55 { 56 friend class FxDevice; 57 friend class FxPackage; 58 friend class FxWmiIrpHandler; 59 60 private: 61 62 MxDriverObject m_DriverObject; 63 UNICODE_STRING m_RegistryPath; 64 65 BOOLEAN m_DebuggerConnected; 66 67 // 68 // Callbacks to device driver 69 // 70 FxDriverDeviceAdd m_DriverDeviceAdd; 71 72 // 73 // This represents any constraints on callbacks 74 // to the device driver that may be inherited 75 // by child devices and their objects 76 // 77 WDF_EXECUTION_LEVEL m_ExecutionLevel; 78 WDF_SYNCHRONIZATION_SCOPE m_SynchronizationScope; 79 80 // 81 // Frameworks objects that raise event callbacks into the device 82 // driver provide spinlock and mutex based callback locks 83 // to allow proper synchronization between the driver and 84 // these callbacks. 85 // 86 // Some events must be passive level, while others at dispatch 87 // level, thus the need for two locks. 88 // 89 // The objects internal state is protected by the FxNonPagedObject 90 // lock inherited by the object, and is different from the callback 91 // locks. 92 // 93 FxCallbackMutexLock m_CallbackMutexLock; 94 95 // 96 // These pointers allow the proper lock to be acquired 97 // based on the configuration with a minimal of runtime 98 // checks. This is configured by ConfigureConstraints() 99 // 100 FxCallbackLock* m_CallbackLockPtr; 101 FxObject* m_CallbackLockObjectPtr; 102 103 // 104 // This is the Driver-wide configuration 105 // 106 WDF_DRIVER_CONFIG m_Config; 107 108 // 109 // Deferred Disposal List 110 // 111 FxDisposeList* m_DisposeList; 112 113 #if FX_IS_USER_MODE 114 // 115 // A handle to the driver service parameters key. 116 // The framework does not have permission to open it with 117 // write access from user mode, so we keep a pre-opened one. 118 // 119 HKEY m_DriverParametersKey; 120 #endif 121 122 private: 123 124 static 125 MdDriverAddDeviceType AddDevice; 126 127 public: 128 129 // This is public to allow the C function FxCoreDriverUnload to call it 130 FxDriverUnload m_DriverUnload; 131 132 FxDriver( 133 __in MdDriverObject DriverObject, 134 __in PWDF_DRIVER_CONFIG DriverConfig, 135 __in PFX_DRIVER_GLOBALS FxDriverGlobals 136 ); 137 138 ~FxDriver(); 139 140 static 141 VOID 142 _InitializeDriverName( 143 __in PFX_DRIVER_GLOBALS Globals, 144 __in PCUNICODE_STRING RegistryPath 145 ); 146 147 static 148 VOID 149 _InitializeTag( 150 __in PFX_DRIVER_GLOBALS Globals, 151 __in PWDF_DRIVER_CONFIG Config 152 ); 153 154 static 155 FxDriver* 156 GetFxDriver( 157 __in MdDriverObject DriverObject 158 ); 159 160 _Must_inspect_result_ 161 NTSTATUS 162 AllocateDriverObjectExtensionAndStoreFxDriver( 163 VOID 164 ); 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 __inline 201 WDFDRIVER 202 GetHandle( 203 VOID 204 ) 205 { 206 return (WDFDRIVER) GetObjectHandle(); 207 } 208 209 #if (FX_CORE_MODE == FX_CORE_KERNEL_MODE) 210 211 _Must_inspect_result_ 212 NTSTATUS 213 AddDevice( 214 __in MdDeviceObject PhysicalDeviceObject 215 ); 216 217 #else 218 219 _Must_inspect_result_ 220 NTSTATUS 221 FxDriver::AddDevice( 222 _In_ IWudfDeviceStack * DevStack, 223 _In_ LPCWSTR KernelDeviceName, 224 _In_opt_ HKEY PdoKey, 225 _In_ LPCWSTR ServiceName, 226 _In_ LPCWSTR DevInstanceID, 227 _In_ ULONG DriverID 228 ); 229 #endif 230 231 VOID 232 InitializeInternal( 233 VOID 234 ); 235 236 _Must_inspect_result_ 237 FxString * 238 GetRegistryPath( 239 VOID 240 ); 241 242 PUNICODE_STRING 243 GetRegistryPathUnicodeString( 244 VOID 245 ) 246 { 247 return &m_RegistryPath; 248 } 249 250 __inline 251 MdDriverObject 252 GetDriverObject( 253 VOID 254 ) 255 { 256 return m_DriverObject.GetObject(); 257 } 258 259 _Must_inspect_result_ 260 NTSTATUS 261 Initialize( 262 __in PCUNICODE_STRING RegistryPath, 263 __in PWDF_DRIVER_CONFIG Config, 264 __in_opt PWDF_OBJECT_ATTRIBUTES DriverAttributes 265 ); 266 267 // 268 // The following methods support the callback constraints 269 // and handle locking and deferral 270 // 271 VOID 272 ConfigureConstraints( 273 __in_opt PWDF_OBJECT_ATTRIBUTES DriverAttributes 274 ); 275 276 // 277 // IFxHasCallbacks Support 278 // 279 280 virtual 281 VOID 282 GetConstraints( 283 __out WDF_EXECUTION_LEVEL* ExecutionLevel, 284 __out WDF_SYNCHRONIZATION_SCOPE* SynchronizationScope 285 ) { 286 287 if (ExecutionLevel != NULL) { 288 *ExecutionLevel = m_ExecutionLevel; 289 } 290 291 if (SynchronizationScope != NULL) { 292 *SynchronizationScope = m_SynchronizationScope; 293 } 294 } 295 296 virtual 297 FxCallbackLock* 298 GetCallbackLockPtr( 299 __deref_out FxObject** LockObject 300 ) { 301 302 if (LockObject != NULL) { 303 *LockObject = m_CallbackLockObjectPtr; 304 } 305 306 return m_CallbackLockPtr; 307 } 308 309 // 310 // IFxAssociation Support 311 // 312 virtual 313 NTSTATUS 314 QueryInterface( 315 __inout FxQueryInterfaceParams* Params 316 ) 317 { 318 switch (Params->Type) { 319 case FX_TYPE_DRIVER: 320 *Params->Object = (FxDriver*) this; 321 break; 322 323 default: 324 return FxNonPagedObject::QueryInterface(Params); // __super call 325 } 326 327 return STATUS_SUCCESS; 328 } 329 330 virtual 331 VOID 332 DeleteObject( 333 VOID 334 ) 335 { 336 // 337 // If diposed at > PASSIVE, we will cause a deadlock in FxDriver::Dispose 338 // when we call into the dispose list to wait for empty when we are in 339 // the context of the dispose list's work item. 340 // 341 ASSERT(Mx::MxGetCurrentIrql() == PASSIVE_LEVEL); 342 343 FxNonPagedObject::DeleteObject(); // __super call 344 } 345 346 virtual 347 BOOLEAN 348 Dispose( 349 VOID 350 ); 351 352 __inline 353 FxDisposeList* 354 GetDisposeList( 355 ) 356 { 357 return m_DisposeList; 358 } 359 360 __inline 361 PFN_WDF_DRIVER_DEVICE_ADD 362 GetDriverDeviceAddMethod( 363 ) 364 { 365 return m_DriverDeviceAdd.Method; 366 } 367 368 static 369 MdDriverUnloadType Unload; 370 371 #if FX_IS_USER_MODE 372 private: 373 374 // 375 // Open the handle to the driver service parameters key 376 // that we keep opened for use from user mode. 377 // 378 NTSTATUS 379 OpenParametersKey( 380 VOID 381 ); 382 383 VOID 384 ClearDriverObjectFxDriver( 385 VOID 386 ); 387 388 public: 389 390 __inline 391 HKEY 392 GetDriverParametersKey( 393 VOID 394 ) 395 { 396 return m_DriverParametersKey; 397 } 398 #endif 399 400 #if (FX_CORE_MODE == FX_CORE_USER_MODE) 401 402 VOID 403 SetDriverObjectFlag( 404 _In_ FxDriverObjectUmFlags Flag 405 ) 406 { 407 m_DriverObject.SetDriverObjectFlag(Flag); 408 } 409 410 BOOLEAN 411 IsDriverObjectFlagSet( 412 _In_ FxDriverObjectUmFlags Flag 413 ) 414 { 415 return m_DriverObject.IsDriverObjectFlagSet(Flag); 416 } 417 418 #endif 419 420 }; 421 422 #endif // _FXDRIVER_H_ 423