Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1756 lines
48 KiB

  1. /*++
  2. Copyright (c) 1997 Microsoft Corporation
  3. Module Name:
  4. cxpnp.c
  5. Abstract:
  6. PnP handling code for the Cluster Network Driver.
  7. Author:
  8. Mike Massa (mikemas) March 21, 1998
  9. Revision History:
  10. Who When What
  11. -------- -------- ----------------------------------------------
  12. mikemas 03-22-98 created
  13. Notes:
  14. --*/
  15. #include "precomp.h"
  16. #include <ntddk.h>
  17. #include <wmistr.h>
  18. #include <ndisguid.h>
  19. #include <ntddndis.h>
  20. #include <ntpnpapi.h>
  21. #include <zwapi.h>
  22. #pragma hdrstop
  23. #include "cxpnp.tmh"
  24. //
  25. // Local data structures
  26. //
  27. typedef struct _CNP_WMI_RECONNECT_WORKER_CONTEXT {
  28. PIO_WORKITEM WorkItem;
  29. CL_NETWORK_ID NetworkId;
  30. } CNP_WMI_RECONNECT_WORKER_CONTEXT, *PCNP_WMI_RECONNECT_WORKER_CONTEXT;
  31. //
  32. // WMI Data
  33. //
  34. PERESOURCE CnpWmiNdisMediaStatusResource = NULL;
  35. PVOID CnpWmiNdisMediaStatusConnectObject = NULL;
  36. PVOID CnpWmiNdisMediaStatusDisconnectObject = NULL;
  37. HANDLE CnpIpMediaSenseFileHandle = NULL;
  38. PIRP CnpIpDisableMediaSenseIrp = NULL;
  39. PKEVENT CnpIpDisableMediaSenseEvent = NULL;
  40. //
  41. // Local prototypes
  42. //
  43. NTSTATUS
  44. CnpWmiPnpDisableMediaSenseCompletion(
  45. IN PDEVICE_OBJECT DeviceObject,
  46. IN PIRP Irp,
  47. IN PVOID Context
  48. );
  49. VOID
  50. CnpWmiNdisMediaStatusConnectCallback(
  51. IN PVOID Wnode,
  52. IN PVOID Context
  53. );
  54. VOID
  55. CnpWmiNdisMediaStatusDisconnectCallback(
  56. IN PVOID Wnode,
  57. IN PVOID Context
  58. );
  59. VOID
  60. CnpReconnectLocalInterfaceWrapper(
  61. IN PDEVICE_OBJECT DeviceObject,
  62. IN PVOID Context
  63. );
  64. VOID
  65. CnpDisconnectLocalInterface(
  66. PCNP_INTERFACE Interface,
  67. PCNP_NETWORK Network
  68. );
  69. #ifdef ALLOC_PRAGMA
  70. #pragma alloc_text(INIT, CxWmiPnpLoad)
  71. #pragma alloc_text(PAGE, CxWmiPnpUnload)
  72. #pragma alloc_text(PAGE, CxWmiPnpInitialize)
  73. #pragma alloc_text(PAGE, CxWmiPnpShutdown)
  74. #endif // ALLOC_PRAGMA
  75. //
  76. // Exported Routines
  77. //
  78. VOID
  79. CxTdiAddAddressHandler(
  80. IN PTA_ADDRESS TaAddress,
  81. IN PUNICODE_STRING DeviceName,
  82. IN PTDI_PNP_CONTEXT Context
  83. )
  84. {
  85. if (TaAddress->AddressType == TDI_ADDRESS_TYPE_IP) {
  86. NTSTATUS status;
  87. PTDI_ADDRESS_IP tdiAddressIp = (PTDI_ADDRESS_IP)
  88. &(TaAddress->Address[0]);
  89. IF_CNDBG(CN_DEBUG_CONFIG) {
  90. CNPRINT((
  91. "[CX] Processing PnP add event for IP address %lx\n",
  92. tdiAddressIp->in_addr
  93. ));
  94. }
  95. //
  96. // Ensure that this is a valid address, and that it is not one
  97. // that we brought online for a cluster ip address resource.
  98. //
  99. if (tdiAddressIp->in_addr != 0) {
  100. if (!IpaIsAddressRegistered(tdiAddressIp->in_addr)) {
  101. IF_CNDBG(CN_DEBUG_CONFIG) {
  102. CNPRINT((
  103. "[CX] Issuing address add event to cluster svc for IP address %lx\n",
  104. tdiAddressIp->in_addr
  105. ));
  106. }
  107. CnIssueEvent(
  108. ClusnetEventAddAddress,
  109. 0,
  110. (CL_NETWORK_ID) tdiAddressIp->in_addr
  111. );
  112. }
  113. else {
  114. IF_CNDBG(CN_DEBUG_CONFIG) {
  115. CNPRINT((
  116. "[CX] PnP add event is for an IP address resource, skip.\n"
  117. ));
  118. }
  119. }
  120. }
  121. }
  122. return;
  123. } // CxTdiAddAddressHandler
  124. VOID
  125. CxTdiDelAddressHandler(
  126. IN PTA_ADDRESS TaAddress,
  127. IN PUNICODE_STRING DeviceName,
  128. IN PTDI_PNP_CONTEXT Context
  129. )
  130. {
  131. if (TaAddress->AddressType == TDI_ADDRESS_TYPE_IP) {
  132. NTSTATUS status;
  133. PCNP_INTERFACE interface;
  134. PCNP_NETWORK network;
  135. PLIST_ENTRY entry;
  136. PTA_IP_ADDRESS taIpAddress;
  137. CN_IRQL nodeTableIrql;
  138. CL_NODE_ID i;
  139. PTDI_ADDRESS_IP tdiAddressIp = (PTDI_ADDRESS_IP)
  140. &(TaAddress->Address[0]);
  141. IF_CNDBG(CN_DEBUG_CONFIG) {
  142. CNPRINT((
  143. "[CX] Processing PnP delete event for IP address %lx.\n",
  144. tdiAddressIp->in_addr
  145. ));
  146. }
  147. if (tdiAddressIp->in_addr != 0) {
  148. //
  149. // Figure out if this is the address for one of this node's
  150. // registered interfaces.
  151. //
  152. CnAcquireLock(&CnpNodeTableLock, &nodeTableIrql);
  153. if (CnpLocalNode != NULL) {
  154. CnAcquireLockAtDpc(&(CnpLocalNode->Lock));
  155. CnReleaseLockFromDpc(&CnpNodeTableLock);
  156. CnpLocalNode->Irql = nodeTableIrql;
  157. network = NULL;
  158. for (entry = CnpLocalNode->InterfaceList.Flink;
  159. entry != &(CnpLocalNode->InterfaceList);
  160. entry = entry->Flink
  161. )
  162. {
  163. interface = CONTAINING_RECORD(
  164. entry,
  165. CNP_INTERFACE,
  166. NodeLinkage
  167. );
  168. taIpAddress = (PTA_IP_ADDRESS) &(interface->TdiAddress);
  169. if (taIpAddress->Address[0].Address[0].in_addr ==
  170. tdiAddressIp->in_addr
  171. )
  172. {
  173. //
  174. // Found the local interface corresponding to this
  175. // address. Be proactive - destroy the corresponding
  176. // network now.
  177. //
  178. network = interface->Network;
  179. CnAcquireLockAtDpc(&CnpNetworkListLock);
  180. CnAcquireLockAtDpc(&(network->Lock));
  181. CnReleaseLockFromDpc(&(CnpLocalNode->Lock));
  182. network->Irql = DISPATCH_LEVEL;
  183. IF_CNDBG(CN_DEBUG_CONFIG) {
  184. CNPRINT((
  185. "[CX] Deleting network ID %u after PnP "
  186. "delete event for IP address %lx.\n",
  187. network->Id, tdiAddressIp->in_addr
  188. ));
  189. }
  190. CnpDeleteNetwork(network, nodeTableIrql);
  191. //
  192. // Both locks were released.
  193. //
  194. break;
  195. }
  196. }
  197. if (network == NULL) {
  198. CnReleaseLock(&(CnpLocalNode->Lock), CnpLocalNode->Irql);
  199. }
  200. //
  201. // Post an event to the service.
  202. //
  203. CnIssueEvent(
  204. ClusnetEventDelAddress,
  205. 0,
  206. (CL_NETWORK_ID) tdiAddressIp->in_addr
  207. );
  208. }
  209. else {
  210. CnReleaseLock(&CnpNodeTableLock, nodeTableIrql);
  211. }
  212. }
  213. }
  214. return;
  215. } // CxTdiDelAddressHandler
  216. NTSTATUS
  217. CxWmiPnpLoad(
  218. VOID
  219. )
  220. /*++
  221. Notes:
  222. Called when clusnet driver is loaded.
  223. --*/
  224. {
  225. PDEVICE_OBJECT ipDeviceObject = NULL;
  226. PFILE_OBJECT ipFileObject = NULL;
  227. PIO_STACK_LOCATION irpSp;
  228. NTSTATUS status;
  229. //
  230. // Allocate a synchronization resource.
  231. //
  232. CnpWmiNdisMediaStatusResource = CnAllocatePool(sizeof(ERESOURCE));
  233. if (CnpWmiNdisMediaStatusResource == NULL) {
  234. return(STATUS_INSUFFICIENT_RESOURCES);
  235. }
  236. status = ExInitializeResourceLite(CnpWmiNdisMediaStatusResource);
  237. if (!NT_SUCCESS(status)) {
  238. return(status);
  239. }
  240. //
  241. // Get a handle to the IP device object to disable media sense
  242. //
  243. status = CnpOpenDevice(
  244. DD_IP_DEVICE_NAME,
  245. &CnpIpMediaSenseFileHandle
  246. );
  247. if (!NT_SUCCESS(status)) {
  248. IF_CNDBG(CN_DEBUG_INIT) {
  249. CNPRINT(("[CX] Failed to open IP device to "
  250. "disable media sense, status %lx\n",
  251. status));
  252. }
  253. return(status);
  254. }
  255. //
  256. // Disable IP media sense. This works by submitting an
  257. // IOCTL_IP_DISABLE_MEDIA_SENSE_REQUEST IRP. The IRP
  258. // will pend until we cancel it (re-enabling media sense).
  259. //
  260. CnpIpDisableMediaSenseEvent = CnAllocatePool(sizeof(KEVENT));
  261. if (CnpIpDisableMediaSenseEvent != NULL) {
  262. KeInitializeEvent(
  263. CnpIpDisableMediaSenseEvent,
  264. SynchronizationEvent,
  265. FALSE
  266. );
  267. //
  268. // Reference the IP file object and get the device object
  269. //
  270. status = ObReferenceObjectByHandle(
  271. CnpIpMediaSenseFileHandle,
  272. 0,
  273. NULL,
  274. KernelMode,
  275. &ipFileObject,
  276. NULL
  277. );
  278. if (NT_SUCCESS(status)) {
  279. ipDeviceObject = IoGetRelatedDeviceObject(ipFileObject);
  280. //
  281. // File object reference is no longer needed
  282. // because the handle is still open.
  283. //
  284. ObDereferenceObject(ipFileObject);
  285. CnpIpDisableMediaSenseIrp = IoAllocateIrp(
  286. ipDeviceObject->StackSize,
  287. FALSE
  288. );
  289. if (CnpIpDisableMediaSenseIrp != NULL) {
  290. irpSp = IoGetNextIrpStackLocation(CnpIpDisableMediaSenseIrp);
  291. irpSp->MajorFunction = IRP_MJ_DEVICE_CONTROL;
  292. irpSp->Parameters.DeviceIoControl.IoControlCode
  293. = IOCTL_IP_DISABLE_MEDIA_SENSE_REQUEST;
  294. irpSp->DeviceObject = ipDeviceObject;
  295. irpSp->FileObject = ipFileObject;
  296. IoSetCompletionRoutine(
  297. CnpIpDisableMediaSenseIrp,
  298. CnpWmiPnpDisableMediaSenseCompletion,
  299. NULL,
  300. TRUE,
  301. TRUE,
  302. TRUE
  303. );
  304. status = IoCallDriver(
  305. ipDeviceObject,
  306. CnpIpDisableMediaSenseIrp
  307. );
  308. if (status != STATUS_PENDING) {
  309. IF_CNDBG(CN_DEBUG_INIT) {
  310. CNPRINT(("[CX] Failed to disable IP media "
  311. "sense, status %lx\n", status));
  312. }
  313. KeWaitForSingleObject(
  314. CnpIpDisableMediaSenseEvent,
  315. Executive,
  316. KernelMode,
  317. FALSE,
  318. NULL
  319. );
  320. CnFreePool(CnpIpDisableMediaSenseEvent);
  321. CnpIpDisableMediaSenseEvent = NULL;
  322. IoFreeIrp(CnpIpDisableMediaSenseIrp);
  323. CnpIpDisableMediaSenseIrp = NULL;
  324. //
  325. // Cannot risk simply returning status
  326. // because we need the driver load to
  327. // fail.
  328. //
  329. if (NT_SUCCESS(status)) {
  330. status = STATUS_UNSUCCESSFUL;
  331. }
  332. } else {
  333. //
  334. // Need to return STATUS_SUCCESS so that
  335. // the driver load will not fail.
  336. //
  337. status = STATUS_SUCCESS;
  338. IF_CNDBG(CN_DEBUG_INIT) {
  339. CNPRINT(("[CX] IP media sense disabled.\n"));
  340. }
  341. CnTrace(
  342. CXPNP, CxWmiPnpIPMediaSenseDisabled,
  343. "[CXPNP] IP media sense disabled.\n"
  344. );
  345. }
  346. } else {
  347. IF_CNDBG(CN_DEBUG_INIT) {
  348. CNPRINT(("[CX] Failed to allocate IP media sense "
  349. "disable IRP.\n"));
  350. }
  351. CnFreePool(CnpIpDisableMediaSenseEvent);
  352. CnpIpDisableMediaSenseEvent = NULL;
  353. status = STATUS_INSUFFICIENT_RESOURCES;
  354. }
  355. } else {
  356. IF_CNDBG(CN_DEBUG_INIT) {
  357. CNPRINT(("[CX] Failed to reference IP device "
  358. "file handle, status %lx\n", status));
  359. }
  360. CnFreePool(CnpIpDisableMediaSenseEvent);
  361. CnpIpDisableMediaSenseEvent = NULL;
  362. }
  363. } else {
  364. IF_CNDBG(CN_DEBUG_INIT) {
  365. CNPRINT(("[CX] Failed to allocate IP media sense "
  366. "disable event.\n"));
  367. }
  368. status = STATUS_INSUFFICIENT_RESOURCES;
  369. }
  370. return(status);
  371. } // CxWmiPnpLoad
  372. VOID
  373. CxWmiPnpUnload(
  374. VOID
  375. )
  376. /*++
  377. Notes:
  378. Called when clusnet driver is unloaded.
  379. --*/
  380. {
  381. CnAssert(CnpWmiNdisMediaStatusConnectObject == NULL);
  382. CnAssert(CnpWmiNdisMediaStatusDisconnectObject == NULL);
  383. //
  384. // Re-enable IP media sense. This works by cancelling our
  385. // IOCTL_IP_DISABLE_MEDIA_SENSE_REQUEST IRP, which should
  386. // still be pending.
  387. //
  388. if (CnpIpDisableMediaSenseIrp != NULL) {
  389. if (!IoCancelIrp(CnpIpDisableMediaSenseIrp)) {
  390. //
  391. // Our disable media sense IRP could not be cancelled. This
  392. // probably means that it was completed because somebody
  393. // else submitted a media sense enable request.
  394. //
  395. CnTrace(
  396. CXPNP, CnpWmiPnpDisableMediaSenseCompletionUnexpected,
  397. "[CXPNP] IP media sense re-enabled unexpectedly.\n"
  398. );
  399. } else {
  400. //
  401. // Irp was cancelled, and media sense is disabled as
  402. // expected.
  403. //
  404. CnTrace(
  405. CXPNP, CnpWmiPnpDisableMediaSenseCompletion,
  406. "[CXPNP] IP media sense re-enabled.\n"
  407. );
  408. }
  409. //
  410. // Regardless of who re-enabled media sense, we need to free
  411. // the media sense IRP and event. First we wait on the event,
  412. // which is signalled in our completion routine.
  413. //
  414. KeWaitForSingleObject(
  415. CnpIpDisableMediaSenseEvent,
  416. Executive,
  417. KernelMode,
  418. FALSE,
  419. NULL
  420. );
  421. CnFreePool(CnpIpDisableMediaSenseEvent);
  422. CnpIpDisableMediaSenseEvent = NULL;
  423. IoFreeIrp(CnpIpDisableMediaSenseIrp);
  424. CnpIpDisableMediaSenseIrp = NULL;
  425. }
  426. CnAssert(CnpIpDisableMediaSenseIrp == NULL);
  427. if (CnpIpMediaSenseFileHandle != NULL) {
  428. ZwClose(CnpIpMediaSenseFileHandle);
  429. CnpIpMediaSenseFileHandle = NULL;
  430. }
  431. if (CnpWmiNdisMediaStatusResource != NULL) {
  432. ExDeleteResourceLite(CnpWmiNdisMediaStatusResource);
  433. CnFreePool(CnpWmiNdisMediaStatusResource);
  434. CnpWmiNdisMediaStatusResource = NULL;
  435. }
  436. } // CxWmiPnpUnload
  437. NTSTATUS
  438. CxWmiPnpInitialize(
  439. VOID
  440. )
  441. /*++
  442. Notes:
  443. Called in response to initialize ioctl.
  444. --*/
  445. {
  446. NTSTATUS status = STATUS_SUCCESS;
  447. BOOLEAN acquired = FALSE;
  448. GUID wmiGuid;
  449. PAGED_CODE();
  450. acquired = CnAcquireResourceExclusive(
  451. CnpWmiNdisMediaStatusResource,
  452. TRUE
  453. );
  454. CnAssert(acquired == TRUE);
  455. //
  456. // Register WMI callbacks for NDIS media status events
  457. //
  458. if (CnpWmiNdisMediaStatusConnectObject == NULL) {
  459. wmiGuid = GUID_NDIS_STATUS_MEDIA_CONNECT;
  460. status = IoWMIOpenBlock(
  461. &wmiGuid,
  462. WMIGUID_NOTIFICATION,
  463. &CnpWmiNdisMediaStatusConnectObject
  464. );
  465. if (!NT_SUCCESS(status)) {
  466. CNPRINT((
  467. "[CX] Unable to open WMI NDIS status media connect "
  468. "datablock, status %lx\n",
  469. status
  470. ));
  471. CnpWmiNdisMediaStatusConnectObject = NULL;
  472. goto error_exit;
  473. }
  474. status = IoWMISetNotificationCallback(
  475. CnpWmiNdisMediaStatusConnectObject,
  476. CnpWmiNdisMediaStatusConnectCallback,
  477. NULL
  478. );
  479. if (!NT_SUCCESS(status)) {
  480. CNPRINT((
  481. "[CX] Unable to register WMI NDIS status media connect "
  482. "callback, status %lx\n",
  483. status
  484. ));
  485. goto error_exit;
  486. }
  487. }
  488. if (CnpWmiNdisMediaStatusDisconnectObject == NULL) {
  489. wmiGuid = GUID_NDIS_STATUS_MEDIA_DISCONNECT;
  490. status = IoWMIOpenBlock(
  491. &wmiGuid,
  492. WMIGUID_NOTIFICATION,
  493. &CnpWmiNdisMediaStatusDisconnectObject
  494. );
  495. if (!NT_SUCCESS(status)) {
  496. CNPRINT((
  497. "[CX] Unable to open WMI NDIS status media disconnect "
  498. "datablock, status %lx\n",
  499. status
  500. ));
  501. CnpWmiNdisMediaStatusDisconnectObject = NULL;
  502. goto error_exit;
  503. }
  504. status = IoWMISetNotificationCallback(
  505. CnpWmiNdisMediaStatusDisconnectObject,
  506. CnpWmiNdisMediaStatusDisconnectCallback,
  507. NULL
  508. );
  509. if (!NT_SUCCESS(status)) {
  510. CNPRINT((
  511. "[CX] Unable to register WMI NDIS status media disconnect "
  512. "callback, status %lx\n",
  513. status
  514. ));
  515. goto error_exit;
  516. }
  517. }
  518. goto release_exit;
  519. error_exit:
  520. if (CnpWmiNdisMediaStatusConnectObject != NULL) {
  521. ObDereferenceObject(CnpWmiNdisMediaStatusConnectObject);
  522. CnpWmiNdisMediaStatusConnectObject = NULL;
  523. }
  524. if (CnpWmiNdisMediaStatusDisconnectObject != NULL) {
  525. ObDereferenceObject(CnpWmiNdisMediaStatusDisconnectObject);
  526. CnpWmiNdisMediaStatusDisconnectObject = NULL;
  527. }
  528. release_exit:
  529. //
  530. // Release resource
  531. //
  532. if (acquired) {
  533. CnReleaseResourceForThread(
  534. CnpWmiNdisMediaStatusResource,
  535. (ERESOURCE_THREAD) PsGetCurrentThread()
  536. );
  537. }
  538. return(status);
  539. } // CxWmiPnpInitialize
  540. VOID
  541. CxWmiPnpShutdown(
  542. VOID
  543. )
  544. /*++
  545. Notes:
  546. Called in response to clusnet shutdown.
  547. --*/
  548. {
  549. BOOLEAN acquired = FALSE;
  550. PAGED_CODE();
  551. acquired = CnAcquireResourceExclusive(
  552. CnpWmiNdisMediaStatusResource,
  553. TRUE
  554. );
  555. CnAssert(acquired == TRUE);
  556. if (CnpWmiNdisMediaStatusConnectObject != NULL) {
  557. ObDereferenceObject(CnpWmiNdisMediaStatusConnectObject);
  558. CnpWmiNdisMediaStatusConnectObject = NULL;
  559. }
  560. if (CnpWmiNdisMediaStatusDisconnectObject != NULL) {
  561. ObDereferenceObject(CnpWmiNdisMediaStatusDisconnectObject);
  562. CnpWmiNdisMediaStatusDisconnectObject = NULL;
  563. }
  564. //
  565. // Release resource
  566. //
  567. if (acquired) {
  568. CnReleaseResourceForThread(
  569. CnpWmiNdisMediaStatusResource,
  570. (ERESOURCE_THREAD) PsGetCurrentThread()
  571. );
  572. }
  573. return;
  574. } // CxWmiPnpShutdown
  575. VOID
  576. CxReconnectLocalInterface(
  577. IN CL_NETWORK_ID NetworkId
  578. )
  579. /**
  580. Routine Description:
  581. Queues a worker thread to set the local interface
  582. associated with NetworkId to connected. Called when
  583. a heartbeat is received over a network that is marked
  584. locally disconnected.
  585. Arguments:
  586. NetworkId - network ID of network to be reconnected
  587. Return value:
  588. None
  589. Notes:
  590. Can fail without reporting an error if either
  591. allocation fails.
  592. --*/
  593. {
  594. PCNP_WMI_RECONNECT_WORKER_CONTEXT context;
  595. context = CnAllocatePool(sizeof(CNP_WMI_RECONNECT_WORKER_CONTEXT));
  596. if (context != NULL) {
  597. context->WorkItem = IoAllocateWorkItem(CnDeviceObject);
  598. if (context->WorkItem != NULL) {
  599. context->NetworkId = NetworkId;
  600. CnTrace(
  601. CXPNP, CxReconnectLocalInterface,
  602. "[CXPNP] Queueing worker thread to reconnect local "
  603. "interface for network ID %u.\n",
  604. NetworkId // LOGULONG
  605. );
  606. IoQueueWorkItem(
  607. context->WorkItem,
  608. CnpReconnectLocalInterfaceWrapper,
  609. DelayedWorkQueue,
  610. context
  611. );
  612. } else {
  613. CnFreePool(context);
  614. }
  615. }
  616. return;
  617. }
  618. VOID
  619. CxQueryMediaStatus(
  620. IN HANDLE AdapterDeviceHandle,
  621. IN CL_NETWORK_ID NetworkId,
  622. OUT PULONG MediaStatus
  623. )
  624. /**
  625. Routine Description:
  626. Queries the status of the adapter device. Used to
  627. determine whether a local interface is initially
  628. connected or disconnected.
  629. Arguments:
  630. AdapterHandle - adapter device object handle
  631. NetworkId - network ID of adapter to be queried
  632. Return value:
  633. None
  634. Notes:
  635. NDIS query formation modeled after ndis\lib\ndisapi.c
  636. --*/
  637. {
  638. BOOLEAN acquired = FALSE;
  639. NTSTATUS status;
  640. CnVerifyCpuLockMask(
  641. 0, // Required
  642. 0xFFFFFFFF, // Forbidden
  643. 0 // Maximum
  644. );
  645. //
  646. // Set default
  647. //
  648. *MediaStatus = NdisMediaStateDisconnected;
  649. //
  650. // Acquire resource
  651. //
  652. acquired = CnAcquireResourceExclusive(
  653. CnpWmiNdisMediaStatusResource,
  654. TRUE
  655. );
  656. CnAssert(acquired == TRUE);
  657. if (AdapterDeviceHandle != NULL) {
  658. //
  659. // Construct NDIS statistics query
  660. //
  661. NDIS_OID statsOidList[] =
  662. {
  663. OID_GEN_MEDIA_CONNECT_STATUS // | NDIS_OID_PRIVATE
  664. };
  665. UCHAR statsBuf[
  666. FIELD_OFFSET(NDIS_STATISTICS_VALUE, Data)
  667. + sizeof(LARGE_INTEGER)
  668. ];
  669. PNDIS_STATISTICS_VALUE pStatsBuf;
  670. LARGE_INTEGER value;
  671. IF_CNDBG( CN_DEBUG_CONFIG ) {
  672. CNPRINT((
  673. "[CXPNP] Querying NDIS for local adapter "
  674. "on network %u (handle %p).\n",
  675. NetworkId,
  676. AdapterDeviceHandle
  677. ));
  678. }
  679. pStatsBuf = (PNDIS_STATISTICS_VALUE) &statsBuf[0];
  680. status = CnpZwDeviceControl(
  681. AdapterDeviceHandle,
  682. IOCTL_NDIS_QUERY_SELECTED_STATS,
  683. statsOidList,
  684. sizeof(statsOidList),
  685. pStatsBuf,
  686. sizeof(statsBuf)
  687. );
  688. IF_CNDBG( CN_DEBUG_CONFIG ) {
  689. CNPRINT((
  690. "[CXPNP] NDIS query for local adapter "
  691. "on network %u returned status %lx.\n",
  692. NetworkId,
  693. status
  694. ));
  695. }
  696. if (pStatsBuf->DataLength == sizeof(LARGE_INTEGER)) {
  697. value.QuadPart = *(PULONGLONG)(&pStatsBuf->Data[0]);
  698. } else {
  699. value.LowPart = *(PULONG)(&pStatsBuf->Data[0]);
  700. }
  701. *MediaStatus = value.LowPart; // NdisMediaState{Disc|C}onnected
  702. IF_CNDBG( CN_DEBUG_CONFIG ) {
  703. CNPRINT((
  704. "[CXPNP] NDIS query for local adapter "
  705. "on network %u returned media status %lx.\n",
  706. NetworkId,
  707. *MediaStatus
  708. ));
  709. }
  710. CnTrace(
  711. CXPNP, CxQueryMediaStatus,
  712. "[CXPNP] Found media status %u for local network ID %u.\n",
  713. *MediaStatus, // LOGULONG
  714. NetworkId // LOGULONG
  715. );
  716. }
  717. //
  718. // If the media status is disconnected, we must disconnect the
  719. // local interface and network.
  720. //
  721. if (*MediaStatus == NdisMediaStateDisconnected) {
  722. PCNP_NETWORK network = NULL;
  723. PCNP_INTERFACE interface = NULL;
  724. CN_IRQL nodeTableIrql;
  725. PLIST_ENTRY entry;
  726. CnAcquireLock(&CnpNodeTableLock, &nodeTableIrql);
  727. if (CnpLocalNode != NULL) {
  728. CnAcquireLockAtDpc(&(CnpLocalNode->Lock));
  729. CnReleaseLockFromDpc(&CnpNodeTableLock);
  730. CnpLocalNode->Irql = nodeTableIrql;
  731. network = CnpFindNetwork(NetworkId);
  732. if (network != NULL) {
  733. //
  734. // Only go through the disconnect if the network
  735. // is currently marked as locally connected.
  736. // It is possible that we have already received
  737. // and processed a WMI disconnect event.
  738. //
  739. if (!CnpIsNetworkLocalDisconn(network)) {
  740. for (entry = CnpLocalNode->InterfaceList.Flink;
  741. entry != &(CnpLocalNode->InterfaceList);
  742. entry = entry->Flink
  743. )
  744. {
  745. interface = CONTAINING_RECORD(
  746. entry,
  747. CNP_INTERFACE,
  748. NodeLinkage
  749. );
  750. if (interface->Network == network) {
  751. CnpDisconnectLocalInterface(
  752. interface,
  753. network
  754. );
  755. //
  756. // Both node and network locks
  757. // were released.
  758. //
  759. break;
  760. } else {
  761. interface = NULL;
  762. }
  763. }
  764. } else {
  765. CnTrace(
  766. CXPNP, CxQueryMediaStatusDisconnectRedundant,
  767. "[CXPNP] Network ID %u is already disconnected; "
  768. "aborting disconnect.\n",
  769. network->Id // LOGULONG
  770. );
  771. }
  772. if (interface == NULL) {
  773. CnReleaseLock(&(network->Lock), network->Irql);
  774. }
  775. }
  776. if (interface == NULL) {
  777. CnReleaseLock(&(CnpLocalNode->Lock), CnpLocalNode->Irql);
  778. }
  779. } else {
  780. CnReleaseLock(&CnpNodeTableLock, nodeTableIrql);
  781. }
  782. }
  783. //
  784. // Release resource
  785. //
  786. if (acquired) {
  787. CnReleaseResourceForThread(
  788. CnpWmiNdisMediaStatusResource,
  789. (ERESOURCE_THREAD) PsGetCurrentThread()
  790. );
  791. }
  792. CnVerifyCpuLockMask(
  793. 0, // Required
  794. 0xFFFFFFFF, // Forbidden
  795. 0 // Maximum
  796. );
  797. return;
  798. } // CxQueryMediaStatus
  799. //
  800. // Local Routines
  801. //
  802. NTSTATUS
  803. CnpWmiPnpDisableMediaSenseCompletion(
  804. IN PDEVICE_OBJECT DeviceObject,
  805. IN PIRP Irp,
  806. IN PVOID Context
  807. )
  808. {
  809. //
  810. // Irp is always freed by our disable routine to prevent a race
  811. // condition where we don't know if we have called IoCancelIrp
  812. // yet or not.
  813. //
  814. KeSetEvent(CnpIpDisableMediaSenseEvent, IO_NO_INCREMENT, FALSE);
  815. return(STATUS_MORE_PROCESSING_REQUIRED);
  816. } // CnpWmiPnpDisableMediaSenseCompletion
  817. VOID
  818. CnpWmiPnpUpdateCurrentInterface(
  819. IN PCNP_INTERFACE UpdateInterface
  820. )
  821. /*++
  822. Routine Description:
  823. Updates the CurrentInterface for interfaces after the local
  824. interface is connected or disconnected. Called in response
  825. to WMI NDIS media status events.
  826. Arguments:
  827. Interface - A pointer to the interface on which to operate.
  828. Return Value:
  829. None.
  830. Notes:
  831. Called with associated node and network locks held.
  832. Returns with network lock released.
  833. Conforms to calling convention for PCNP_INTERFACE_UPDATE_ROUTINE.
  834. --*/
  835. {
  836. PCNP_NODE node = UpdateInterface->Node;
  837. CnVerifyCpuLockMask(
  838. (CNP_NODE_OBJECT_LOCK | CNP_NETWORK_OBJECT_LOCK), // Required
  839. 0, // Forbidden
  840. CNP_NETWORK_OBJECT_LOCK_MAX // Maximum
  841. );
  842. //
  843. // We don't really need the network lock. It's just part of
  844. // the calling convention.
  845. //
  846. CnReleaseLockFromDpc(&(UpdateInterface->Network->Lock));
  847. CnpUpdateNodeCurrentInterface(node);
  848. if ( (node->CurrentInterface == NULL)
  849. ||
  850. ( node->CurrentInterface->State <
  851. ClusnetInterfaceStateOnlinePending
  852. )
  853. )
  854. {
  855. //
  856. // This node is now unreachable.
  857. //
  858. CnTrace(
  859. CXPNP, CxWmiPnpNodeUnreach,
  860. "[CXPNP] Declaring node %u unreachable after "
  861. "handling media sense event.\n",
  862. node->Id
  863. );
  864. CnpDeclareNodeUnreachable(node);
  865. } else {
  866. //
  867. // This node may now be reachable.
  868. //
  869. if (CnpIsNodeUnreachable(node)) {
  870. CnTrace(
  871. CXPNP,
  872. CxWmiPnpNodeReach,
  873. "[CNP] Declaring node %u reachable after "
  874. "handling media sense event.\n",
  875. node->Id
  876. );
  877. CnpDeclareNodeReachable(node);
  878. }
  879. }
  880. CnVerifyCpuLockMask(
  881. (CNP_NODE_OBJECT_LOCK), // Required
  882. 0, // Forbidden
  883. CNP_NODE_OBJECT_LOCK_MAX // Maximum
  884. );
  885. return;
  886. } // CnpWmiPnpUpdateCurrentInterface
  887. VOID
  888. CnpReconnectLocalInterface(
  889. PCNP_INTERFACE Interface,
  890. PCNP_NETWORK Network
  891. )
  892. /*++
  893. Routine Description:
  894. Changes a local interface from being disconnected to
  895. connected. Called in response to a WMI NDIS media status
  896. connect event or a heartbeat received on a disconnected
  897. interface.
  898. Arguments:
  899. Interface - local interface that is reconnected
  900. Network - network associated with Interface
  901. Return value:
  902. None
  903. Notes:
  904. Called with CnpWmiNdisMediaStatusResource, local node lock,
  905. and Network lock held.
  906. Returns with CnpWmiNdisMediaStatusResource held but neither
  907. lock held.
  908. --*/
  909. {
  910. CnVerifyCpuLockMask(
  911. (CNP_NODE_OBJECT_LOCK | CNP_NETWORK_OBJECT_LOCK), // Required
  912. 0, // Forbidden
  913. CNP_NETWORK_OBJECT_LOCK_MAX // Maximum
  914. );
  915. CnTrace(
  916. CXPNP, CnpReconnectLocalInterface,
  917. "[CXPNP] Reconnecting local interface for "
  918. "network ID %u.\n",
  919. Network->Id // LOGULONG
  920. );
  921. //
  922. // Clear the local disconnect flag in the network
  923. // object
  924. //
  925. Network->Flags &= ~CNP_NET_FLAG_LOCALDISCONN;
  926. //
  927. // Reference the network so it can't go away while we
  928. // reprioritize the associated interfaces.
  929. //
  930. CnpReferenceNetwork(Network);
  931. //
  932. // Bring the interface online. This call releases the
  933. // network lock.
  934. //
  935. CnpOnlineInterface(Interface);
  936. //
  937. // Release the node lock before walking the interfaces
  938. // on the network.
  939. //
  940. CnReleaseLock(&(CnpLocalNode->Lock), CnpLocalNode->Irql);
  941. //
  942. // Update the CurrentInterface for the other
  943. // nodes in the cluster to reflect the connected
  944. // status of the local interface.
  945. //
  946. CnpWalkInterfacesOnNetwork(
  947. Network,
  948. CnpWmiPnpUpdateCurrentInterface
  949. );
  950. //
  951. // Issue InterfaceUp event to the cluster
  952. // service.
  953. //
  954. CnTrace(
  955. CXPNP, CxWmiNdisReconnectIssueEvent,
  956. "[CXPNP] Issuing InterfaceUp event "
  957. "for node %u on net %u, previous I/F state = %!ifstate!.",
  958. Interface->Node->Id, // LOGULONG
  959. Interface->Network->Id, // LOGULONG
  960. Interface->State // LOGIfState
  961. );
  962. CnIssueEvent(
  963. ClusnetEventNetInterfaceUp,
  964. Interface->Node->Id,
  965. Interface->Network->Id
  966. );
  967. //
  968. // Release the reference on the network object.
  969. //
  970. CnAcquireLock(&(Network->Lock), &(Network->Irql));
  971. CnpDereferenceNetwork(Network);
  972. return;
  973. } // CnpReconnectLocalInterface
  974. VOID
  975. CnpReconnectLocalInterfaceWrapper(
  976. IN PDEVICE_OBJECT DeviceObject,
  977. IN PVOID Context
  978. )
  979. {
  980. PCNP_WMI_RECONNECT_WORKER_CONTEXT context = Context;
  981. PCNP_NETWORK network = NULL;
  982. PCNP_INTERFACE interface = NULL;
  983. CN_IRQL nodeTableIrql;
  984. BOOLEAN acquired = FALSE;
  985. PLIST_ENTRY entry;
  986. CnVerifyCpuLockMask(
  987. 0, // Required
  988. 0xFFFFFFFF, // Forbidden
  989. 0 // Maximum
  990. );
  991. acquired = CnAcquireResourceExclusive(
  992. CnpWmiNdisMediaStatusResource,
  993. TRUE
  994. );
  995. CnAssert(acquired == TRUE);
  996. CnAcquireLock(&CnpNodeTableLock, &nodeTableIrql);
  997. if (CnpLocalNode != NULL) {
  998. CnAcquireLockAtDpc(&(CnpLocalNode->Lock));
  999. CnReleaseLockFromDpc(&CnpNodeTableLock);
  1000. CnpLocalNode->Irql = nodeTableIrql;
  1001. network = CnpFindNetwork(context->NetworkId);
  1002. if (network != NULL) {
  1003. //
  1004. // Only go through the reconnect if the network
  1005. // is currently marked as locally disconnected.
  1006. // It is possible that we have already received
  1007. // and processed a WMI connect event.
  1008. //
  1009. if (CnpIsNetworkLocalDisconn(network)) {
  1010. for (entry = CnpLocalNode->InterfaceList.Flink;
  1011. entry != &(CnpLocalNode->InterfaceList);
  1012. entry = entry->Flink
  1013. )
  1014. {
  1015. interface = CONTAINING_RECORD(
  1016. entry,
  1017. CNP_INTERFACE,
  1018. NodeLinkage
  1019. );
  1020. if (interface->Network == network) {
  1021. CnpReconnectLocalInterface(
  1022. interface,
  1023. network
  1024. );
  1025. //
  1026. // Both node and network locks
  1027. // were released.
  1028. //
  1029. break;
  1030. } else {
  1031. interface = NULL;
  1032. }
  1033. }
  1034. } else {
  1035. CnTrace(
  1036. CXPNP, CnpReconnectLocalInterfaceWrapperRedundant,
  1037. "[CXPNP] Network ID %u is already connected; "
  1038. "aborting reconnect in wrapper.\n",
  1039. network->Id // LOGULONG
  1040. );
  1041. }
  1042. if (interface == NULL) {
  1043. CnReleaseLock(&(network->Lock), network->Irql);
  1044. }
  1045. }
  1046. if (interface == NULL) {
  1047. CnReleaseLock(&(CnpLocalNode->Lock), CnpLocalNode->Irql);
  1048. }
  1049. } else {
  1050. CnReleaseLock(&CnpNodeTableLock, nodeTableIrql);
  1051. }
  1052. //
  1053. // Release resource
  1054. //
  1055. if (acquired) {
  1056. CnReleaseResourceForThread(
  1057. CnpWmiNdisMediaStatusResource,
  1058. (ERESOURCE_THREAD) PsGetCurrentThread()
  1059. );
  1060. }
  1061. //
  1062. // Free the workitem and context
  1063. //
  1064. IoFreeWorkItem(context->WorkItem);
  1065. CnFreePool(context);
  1066. CnVerifyCpuLockMask(
  1067. 0, // Required
  1068. 0xFFFFFFFF, // Forbidden
  1069. 0 // Maximum
  1070. );
  1071. return;
  1072. } // CnpReconnectLocalInterfaceWrapper
  1073. VOID
  1074. CnpDisconnectLocalInterface(
  1075. PCNP_INTERFACE Interface,
  1076. PCNP_NETWORK Network
  1077. )
  1078. /*++
  1079. Routine Description:
  1080. Changes a local interface from being connected to
  1081. disconnected. Called in response to a WMI NDIS media status
  1082. disconnect event or an NDIS query that returns media
  1083. disconnected.
  1084. Arguments:
  1085. Interface - local interface that is reconnected
  1086. Network - network associated with Interface
  1087. Return value:
  1088. None
  1089. Notes:
  1090. Called with CnpWmiNdisMediaStatusResource, local node lock,
  1091. and Network lock held.
  1092. Returns with CnpWmiNdisMediaStatusResource held but neither
  1093. lock held.
  1094. --*/
  1095. {
  1096. CnVerifyCpuLockMask(
  1097. (CNP_NODE_OBJECT_LOCK | CNP_NETWORK_OBJECT_LOCK), // Required
  1098. 0, // Forbidden
  1099. CNP_NETWORK_OBJECT_LOCK_MAX // Maximum
  1100. );
  1101. CnTrace(
  1102. CXPNP, CnpDisconnectLocalInterface,
  1103. "[CXPNP] Interface for network ID %u "
  1104. "disconnected.\n",
  1105. Network->Id // LOGULONG
  1106. );
  1107. //
  1108. // Set the local disconnect flag in the network
  1109. // object
  1110. //
  1111. Network->Flags |= CNP_NET_FLAG_LOCALDISCONN;
  1112. //
  1113. // Reference the network so it can't go away while we
  1114. // reprioritize the associated interfaces.
  1115. //
  1116. CnpReferenceNetwork(Network);
  1117. //
  1118. // Fail the interface. This call releases the
  1119. // network lock.
  1120. //
  1121. CnpFailInterface(Interface);
  1122. //
  1123. // Release the node lock before walking the interfaces
  1124. // on the network.
  1125. //
  1126. CnReleaseLock(&(CnpLocalNode->Lock), CnpLocalNode->Irql);
  1127. //
  1128. // Update the CurrentInterface for the other
  1129. // nodes in the cluster to reflect the disconnected
  1130. // status of the local interface.
  1131. //
  1132. CnpWalkInterfacesOnNetwork(
  1133. Network,
  1134. CnpWmiPnpUpdateCurrentInterface
  1135. );
  1136. //
  1137. // Issue InterfaceFailed event to the cluster
  1138. // service.
  1139. //
  1140. CnTrace(
  1141. CXPNP, CnpLocalDisconnectIssueEvent,
  1142. "[CXPNP] Issuing InterfaceFailed event "
  1143. "for node %u on net %u, previous I/F state = %!ifstate!.",
  1144. Interface->Node->Id, // LOGULONG
  1145. Interface->Network->Id, // LOGULONG
  1146. Interface->State // LOGIfState
  1147. );
  1148. CnIssueEvent(
  1149. ClusnetEventNetInterfaceFailed,
  1150. Interface->Node->Id,
  1151. Interface->Network->Id
  1152. );
  1153. //
  1154. // Release the reference on the network object.
  1155. //
  1156. CnAcquireLock(&(Network->Lock), &(Network->Irql));
  1157. CnpDereferenceNetwork(Network);
  1158. return;
  1159. } // CnpDisconnectLocalInterface
  1160. VOID
  1161. CnpWmiNdisMediaStatusConnectCallback(
  1162. IN PVOID Wnode,
  1163. IN PVOID Context
  1164. )
  1165. {
  1166. PWNODE_SINGLE_INSTANCE wnode = (PWNODE_SINGLE_INSTANCE) Wnode;
  1167. PCNP_INTERFACE interface;
  1168. PCNP_NETWORK network;
  1169. PLIST_ENTRY entry;
  1170. CN_IRQL nodeTableIrql;
  1171. BOOLEAN acquired = FALSE;
  1172. CnVerifyCpuLockMask(
  1173. 0, // Required
  1174. 0xFFFFFFFF, // Forbidden
  1175. 0 // Maximum
  1176. );
  1177. IF_CNDBG(CN_DEBUG_CONFIG) {
  1178. CNPRINT((
  1179. "[CX] Received WMI NDIS status media connect event.\n"
  1180. ));
  1181. }
  1182. //
  1183. // Serialize events as much as possible, since clusnet spinlocks
  1184. // may be acquired and released repeatedly.
  1185. //
  1186. // Note that there may not be any guarantees that WMI event
  1187. // ordering is guaranteed. The fallback mechanism for clusnet
  1188. // is heartbeats -- if a heartbeat is received on an interface,
  1189. // we know the interface is connected.
  1190. //
  1191. acquired = CnAcquireResourceExclusive(
  1192. CnpWmiNdisMediaStatusResource,
  1193. TRUE
  1194. );
  1195. CnAssert(acquired == TRUE);
  1196. //
  1197. // Figure out if this callback is for one of this node's
  1198. // registered interfaces by comparing the WMI provider ID
  1199. // in the WNODE header to the WMI provider IDs of this
  1200. // node's adapters.
  1201. //
  1202. CnAcquireLock(&CnpNodeTableLock, &nodeTableIrql);
  1203. if (CnpLocalNode != NULL) {
  1204. CnAcquireLockAtDpc(&(CnpLocalNode->Lock));
  1205. CnReleaseLockFromDpc(&CnpNodeTableLock);
  1206. CnpLocalNode->Irql = nodeTableIrql;
  1207. network = NULL;
  1208. for (entry = CnpLocalNode->InterfaceList.Flink;
  1209. entry != &(CnpLocalNode->InterfaceList);
  1210. entry = entry->Flink
  1211. )
  1212. {
  1213. interface = CONTAINING_RECORD(
  1214. entry,
  1215. CNP_INTERFACE,
  1216. NodeLinkage
  1217. );
  1218. if (wnode->WnodeHeader.ProviderId
  1219. == interface->AdapterWMIProviderId) {
  1220. //
  1221. // Found the local interface corresponding to this
  1222. // address.
  1223. //
  1224. network = interface->Network;
  1225. //
  1226. // Start by checking if we believe the network is
  1227. // currently disconnected.
  1228. //
  1229. CnAcquireLockAtDpc(&(network->Lock));
  1230. network->Irql = DISPATCH_LEVEL;
  1231. if (CnpIsNetworkLocalDisconn(network)) {
  1232. CnTrace(
  1233. CXPNP, CxWmiNdisConnectNet,
  1234. "[CXPNP] Interface for network ID %u "
  1235. "connected.\n",
  1236. network->Id // LOGULONG
  1237. );
  1238. CnpReconnectLocalInterface(interface, network);
  1239. //
  1240. // Node and network locks were released
  1241. //
  1242. } else {
  1243. CnTrace(
  1244. CXPNP, CxWmiNdisConnectNetRedundant,
  1245. "[CXPNP] Ignoring redundant WMI NDIS connect "
  1246. "event for interface for network ID %u.\n",
  1247. network->Id // LOGULONG
  1248. );
  1249. CnReleaseLockFromDpc(&(network->Lock));
  1250. CnReleaseLock(&(CnpLocalNode->Lock), CnpLocalNode->Irql);
  1251. }
  1252. break;
  1253. }
  1254. }
  1255. if (network == NULL) {
  1256. CnReleaseLock(&(CnpLocalNode->Lock), CnpLocalNode->Irql);
  1257. }
  1258. }
  1259. else {
  1260. CnReleaseLock(&CnpNodeTableLock, nodeTableIrql);
  1261. }
  1262. IF_CNDBG(CN_DEBUG_CONFIG) {
  1263. if (network != NULL) {
  1264. CNPRINT((
  1265. "[CX] Interface for network ID %u connected.\n",
  1266. network->Id
  1267. ));
  1268. } else {
  1269. CNPRINT((
  1270. "[CX] Unknown interface connected, provider id %lx\n",
  1271. wnode->WnodeHeader.ProviderId
  1272. ));
  1273. }
  1274. }
  1275. //
  1276. // Release resource
  1277. //
  1278. if (acquired) {
  1279. CnReleaseResourceForThread(
  1280. CnpWmiNdisMediaStatusResource,
  1281. (ERESOURCE_THREAD) PsGetCurrentThread()
  1282. );
  1283. }
  1284. CnVerifyCpuLockMask(
  1285. 0, // Required
  1286. 0xFFFFFFFF, // Forbidden
  1287. 0 // Maximum
  1288. );
  1289. return;
  1290. } // CnpWmiNdisMediaStatusConnectCallback
  1291. VOID
  1292. CnpWmiNdisMediaStatusDisconnectCallback(
  1293. IN PVOID Wnode,
  1294. IN PVOID Context
  1295. )
  1296. {
  1297. PWNODE_SINGLE_INSTANCE wnode = (PWNODE_SINGLE_INSTANCE) Wnode;
  1298. PCNP_INTERFACE interface;
  1299. PCNP_NETWORK network;
  1300. PLIST_ENTRY entry;
  1301. CN_IRQL nodeTableIrql;
  1302. BOOLEAN acquired = FALSE;
  1303. CnVerifyCpuLockMask(
  1304. 0, // Required
  1305. 0xFFFFFFFF, // Forbidden
  1306. 0 // Maximum
  1307. );
  1308. IF_CNDBG(CN_DEBUG_CONFIG) {
  1309. CNPRINT((
  1310. "[CX] Received WMI NDIS status media disconnect event.\n"
  1311. ));
  1312. }
  1313. CnTrace(CXPNP, CxWmiNdisDisconnect,
  1314. "[CXPNP] Received WMI NDIS status media disconnect event.\n"
  1315. );
  1316. //
  1317. // Serialize events as much as possible, since clusnet spinlocks
  1318. // may be acquired and released repeatedly.
  1319. //
  1320. // Note that there may not be any guarantees that WMI event
  1321. // ordering is guaranteed. The fallback mechanism for clusnet
  1322. // is heartbeats -- if a heartbeat is received on an interface,
  1323. // we know the interface is connected.
  1324. //
  1325. acquired = CnAcquireResourceExclusive(
  1326. CnpWmiNdisMediaStatusResource,
  1327. TRUE
  1328. );
  1329. CnAssert(acquired == TRUE);
  1330. //
  1331. // Figure out if this callback is for one of this node's
  1332. // registered interfaces by comparing the WMI provider ID
  1333. // in the WNODE header to the WMI provider IDs of this
  1334. // node's adapters.
  1335. //
  1336. CnAcquireLock(&CnpNodeTableLock, &nodeTableIrql);
  1337. if (CnpLocalNode != NULL) {
  1338. CnAcquireLockAtDpc(&(CnpLocalNode->Lock));
  1339. CnReleaseLockFromDpc(&CnpNodeTableLock);
  1340. CnpLocalNode->Irql = nodeTableIrql;
  1341. network = NULL;
  1342. for (entry = CnpLocalNode->InterfaceList.Flink;
  1343. entry != &(CnpLocalNode->InterfaceList);
  1344. entry = entry->Flink
  1345. )
  1346. {
  1347. interface = CONTAINING_RECORD(
  1348. entry,
  1349. CNP_INTERFACE,
  1350. NodeLinkage
  1351. );
  1352. if (wnode->WnodeHeader.ProviderId
  1353. == interface->AdapterWMIProviderId) {
  1354. //
  1355. // Found the local interface object corresponding
  1356. // to this adapter.
  1357. //
  1358. network = interface->Network;
  1359. CnAcquireLockAtDpc(&(network->Lock));
  1360. network->Irql = DISPATCH_LEVEL;
  1361. CnpDisconnectLocalInterface(interface, network);
  1362. break;
  1363. }
  1364. }
  1365. if (network == NULL) {
  1366. CnReleaseLock(&(CnpLocalNode->Lock), CnpLocalNode->Irql);
  1367. }
  1368. }
  1369. else {
  1370. CnReleaseLock(&CnpNodeTableLock, nodeTableIrql);
  1371. }
  1372. IF_CNDBG(CN_DEBUG_CONFIG) {
  1373. if (network != NULL) {
  1374. CNPRINT((
  1375. "[CX] Interface for network ID %u disconnected.\n",
  1376. network->Id
  1377. ));
  1378. } else {
  1379. CNPRINT((
  1380. "[CX] Unknown interface disconnected, provider id %lx\n",
  1381. wnode->WnodeHeader.ProviderId
  1382. ));
  1383. }
  1384. }
  1385. //
  1386. // Release resource
  1387. //
  1388. if (acquired) {
  1389. CnReleaseResourceForThread(
  1390. CnpWmiNdisMediaStatusResource,
  1391. (ERESOURCE_THREAD) PsGetCurrentThread()
  1392. );
  1393. }
  1394. CnVerifyCpuLockMask(
  1395. 0, // Required
  1396. 0xFFFFFFFF, // Forbidden
  1397. 0 // Maximum
  1398. );
  1399. return;
  1400. } // CnpWmiNdisMediaStatusDisconnectCallback