Source code of Windows XP (NT5)
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.

2520 lines
72 KiB

  1. /*++
  2. Copyright(c) 1999-2000 Microsoft Corporation
  3. Module Name:
  4. brdgmini.c
  5. Abstract:
  6. Ethernet MAC level bridge.
  7. Miniport section
  8. Author:
  9. Mark Aiken
  10. (original bridge by Jameel Hyder)
  11. Environment:
  12. Kernel mode driver
  13. Revision History:
  14. Sept 1999 - Original version
  15. Feb 2000 - Overhaul
  16. --*/
  17. #define NDIS_MINIPORT_DRIVER
  18. #define NDIS50_MINIPORT 1
  19. #define NDIS_WDM 1
  20. #pragma warning( push, 3 )
  21. #include <ndis.h>
  22. #pragma warning( pop )
  23. #include <netevent.h>
  24. #include "bridge.h"
  25. #include "brdgmini.h"
  26. #include "brdgfwd.h"
  27. #include "brdgprot.h"
  28. #include "brdgbuf.h"
  29. #include "brdgsta.h"
  30. #include "brdgcomp.h"
  31. // ===========================================================================
  32. //
  33. // GLOBALS
  34. //
  35. // ===========================================================================
  36. // NDIS Wrapper handle
  37. NDIS_HANDLE gNDISWrapperHandle = NULL;
  38. // Handle to our miniport driver
  39. NDIS_HANDLE gMiniPortDriverHandle = NULL;
  40. // ----------------------------------------------
  41. // The handle of the miniport (NULL if not initialized)
  42. NDIS_HANDLE gMiniPortAdapterHandle = NULL;
  43. // Refcount to allow waiting for other code to finish using the miniport
  44. WAIT_REFCOUNT gMiniPortAdapterRefcount;
  45. // Refcount indicating whether the bridge miniport is media-connected
  46. WAIT_REFCOUNT gMiniPortConnectedRefcount;
  47. // Refcount indicating whether the bridge miniport is in the middle of a media
  48. // state toggle.
  49. WAIT_REFCOUNT gMiniPortToggleRefcount;
  50. // ----------------------------------------------
  51. //
  52. // Refcount for use in passing through requests to underlying NICs
  53. // This works because NDIS doesn't make requests re-entrantly. That
  54. // is, only one SetInfo operation can be pending at any given time.
  55. //
  56. LONG gRequestRefCount;
  57. // ----------------------------------------------
  58. // Virtual characteristics of the bridge adapter
  59. ULONG gBridgeLinkSpeed = 10000L, // Start at 1Mbps, since reporting
  60. // zero makes some components unhappy.
  61. // Measured in 100's of bps.
  62. gBridgeMediaState = NdisMediaStateDisconnected;
  63. // MAC Address of the bridge. This does not change once
  64. // it has been set.
  65. UCHAR gBridgeAddress[ETH_LENGTH_OF_ADDRESS];
  66. // Whether we have chosen an address yet
  67. BOOLEAN gHaveAddress;
  68. // Current bridge packet filter
  69. ULONG gPacketFilter = 0L;
  70. // Current multicast list
  71. PUCHAR gMulticastList = NULL;
  72. ULONG gMulticastListLength = 0L;
  73. // Device name of the bridge miniport (from the registry)
  74. PWCHAR gBridgeDeviceName = NULL;
  75. ULONG gBridgeDeviceNameSize = 0L;
  76. // RW lock to protect all above bridge variables
  77. NDIS_RW_LOCK gBridgeStateLock;
  78. //-----------------------------------------------
  79. // Name of the registry entry for the device name
  80. const PWCHAR gDeviceNameEntry = L"Device";
  81. // Description of our miniport
  82. const PCHAR gDriverDescription = "Microsoft MAC Bridge Virtual NIC";
  83. // ----------------------------------------------
  84. // Device object so user-mode code can talk to us
  85. PDEVICE_OBJECT gDeviceObject = NULL;
  86. // NDIS handle to track the device object
  87. NDIS_HANDLE gDeviceHandle = NULL;
  88. // ----------------------------------------------
  89. // List of supported OIDs
  90. NDIS_OID gSupportedOIDs[] =
  91. {
  92. // General characteristics
  93. OID_GEN_SUPPORTED_LIST,
  94. OID_GEN_HARDWARE_STATUS,
  95. OID_GEN_MEDIA_SUPPORTED,
  96. OID_GEN_MEDIA_IN_USE,
  97. OID_GEN_MAXIMUM_LOOKAHEAD,
  98. OID_GEN_MAXIMUM_FRAME_SIZE,
  99. OID_GEN_LINK_SPEED,
  100. OID_GEN_TRANSMIT_BUFFER_SPACE,
  101. OID_GEN_RECEIVE_BUFFER_SPACE,
  102. OID_GEN_TRANSMIT_BLOCK_SIZE,
  103. OID_GEN_RECEIVE_BLOCK_SIZE,
  104. OID_GEN_VENDOR_ID,
  105. OID_GEN_VENDOR_DESCRIPTION,
  106. OID_GEN_CURRENT_PACKET_FILTER,
  107. OID_GEN_CURRENT_LOOKAHEAD,
  108. OID_GEN_DRIVER_VERSION,
  109. OID_GEN_MAXIMUM_TOTAL_SIZE,
  110. OID_GEN_PROTOCOL_OPTIONS,
  111. OID_GEN_MAC_OPTIONS,
  112. OID_GEN_MEDIA_CONNECT_STATUS,
  113. OID_GEN_MAXIMUM_SEND_PACKETS,
  114. OID_GEN_VENDOR_DRIVER_VERSION,
  115. // Set only characteristics (relayed)
  116. OID_GEN_NETWORK_LAYER_ADDRESSES,
  117. OID_GEN_TRANSPORT_HEADER_OFFSET,
  118. // General statistics
  119. OID_GEN_XMIT_OK,
  120. OID_GEN_RCV_OK,
  121. OID_GEN_XMIT_ERROR,
  122. OID_GEN_RCV_NO_BUFFER,
  123. OID_GEN_RCV_NO_BUFFER,
  124. OID_GEN_DIRECTED_BYTES_XMIT,
  125. OID_GEN_DIRECTED_FRAMES_XMIT,
  126. OID_GEN_MULTICAST_BYTES_XMIT,
  127. OID_GEN_MULTICAST_FRAMES_XMIT,
  128. OID_GEN_BROADCAST_BYTES_XMIT,
  129. OID_GEN_BROADCAST_FRAMES_XMIT,
  130. OID_GEN_DIRECTED_BYTES_RCV,
  131. OID_GEN_DIRECTED_FRAMES_RCV,
  132. OID_GEN_MULTICAST_BYTES_RCV,
  133. OID_GEN_MULTICAST_FRAMES_RCV,
  134. OID_GEN_BROADCAST_BYTES_RCV,
  135. OID_GEN_BROADCAST_FRAMES_RCV,
  136. // Ethernet characteristics
  137. OID_802_3_PERMANENT_ADDRESS,
  138. OID_802_3_CURRENT_ADDRESS,
  139. OID_802_3_MULTICAST_LIST,
  140. OID_802_3_MAXIMUM_LIST_SIZE,
  141. // Ethernet statistics
  142. OID_802_3_RCV_ERROR_ALIGNMENT,
  143. OID_802_3_XMIT_ONE_COLLISION,
  144. OID_802_3_XMIT_MORE_COLLISIONS,
  145. // PnP OIDs
  146. OID_PNP_QUERY_POWER,
  147. OID_PNP_SET_POWER,
  148. // tcp oids
  149. OID_TCP_TASK_OFFLOAD
  150. };
  151. // 1394 specific related global variables
  152. #define OID_1394_ENTER_BRIDGE_MODE 0xFF00C914
  153. #define OID_1394_EXIT_BRIDGE_MODE 0xFF00C915
  154. // Set when the bridge knows that tcpip has been loaded
  155. // set on receiving the OID_TCP_TASK_OFFLOAD Oid
  156. BOOLEAN g_fIsTcpIpLoaded = FALSE;
  157. // ===========================================================================
  158. //
  159. // PRIVATE PROTOTYPES
  160. //
  161. // ===========================================================================
  162. VOID
  163. BrdgMiniHalt(
  164. IN NDIS_HANDLE MiniportAdapterContext
  165. );
  166. NDIS_STATUS
  167. BrdgMiniInitialize(
  168. OUT PNDIS_STATUS OpenErrorStatus,
  169. OUT PUINT SelectedMediumIndex,
  170. IN PNDIS_MEDIUM MediumArray,
  171. IN UINT MediumArraySize,
  172. IN NDIS_HANDLE MiniportAdapterHandle,
  173. IN NDIS_HANDLE WrapperConfigurationContext
  174. );
  175. NDIS_STATUS
  176. BrdgMiniQueryInfo(
  177. IN NDIS_HANDLE MiniportAdapterContext,
  178. IN NDIS_OID Oid,
  179. IN PVOID InformationBuffer,
  180. IN ULONG InformationBufferLength,
  181. OUT PULONG BytesWritten,
  182. OUT PULONG BytesNeeded
  183. );
  184. NDIS_STATUS
  185. BrdgMiniReset(
  186. OUT PBOOLEAN AddressingReset,
  187. IN NDIS_HANDLE MiniportAdapterContext
  188. );
  189. VOID
  190. BrdgMiniSendPackets(
  191. IN NDIS_HANDLE MiniportAdapterContext,
  192. IN PPNDIS_PACKET PacketArray,
  193. IN UINT NumberOfPackets
  194. );
  195. NDIS_STATUS
  196. BrdgMiniSetInfo(
  197. IN NDIS_HANDLE MiniportAdapterContext,
  198. IN NDIS_OID Oid,
  199. IN PVOID InformationBuffer,
  200. IN ULONG InformationBufferLength,
  201. OUT PULONG BytesRead,
  202. OUT PULONG BytesNeeded
  203. );
  204. BOOLEAN
  205. BrdgMiniAddrIsInMultiList(
  206. IN PUCHAR pTargetAddr
  207. );
  208. VOID
  209. BrdgMiniRelayedRequestComplete(
  210. PNDIS_REQUEST_BETTER pRequest,
  211. PVOID unused
  212. );
  213. VOID
  214. BrdgMiniReAcquireMiniport();
  215. // ===========================================================================
  216. //
  217. // PUBLIC FUNCTIONS
  218. //
  219. // ===========================================================================
  220. NTSTATUS
  221. BrdgMiniDriverInit()
  222. /*++
  223. Routine Description:
  224. Load-time initialization function
  225. Must run at PASSIVE_LEVEL since we call NdisRegisterDevice().
  226. Arguments:
  227. None
  228. Return Value:
  229. Status of initialization. A return code != STATUS_SUCCESS causes driver load
  230. to fail. Any event causing a failure return code must be logged, as it
  231. prevents us from loading successfully.
  232. --*/
  233. {
  234. NDIS_MINIPORT_CHARACTERISTICS MiniPortChars;
  235. NDIS_STATUS NdisStatus;
  236. PDRIVER_DISPATCH DispatchTable[IRP_MJ_MAXIMUM_FUNCTION+1];
  237. NDIS_STRING DeviceName, LinkName;
  238. SAFEASSERT(CURRENT_IRQL == PASSIVE_LEVEL);
  239. NdisInitializeReadWriteLock( &gBridgeStateLock );
  240. BrdgInitializeWaitRef( &gMiniPortAdapterRefcount, FALSE );
  241. BrdgInitializeWaitRef( &gMiniPortConnectedRefcount, TRUE );
  242. BrdgInitializeWaitRef( &gMiniPortToggleRefcount, FALSE );
  243. // Put the miniport refcount into shutdown mode (so a refcount can't be acquired)
  244. // since we don't have a miniport yet
  245. BrdgShutdownWaitRefOnce( &gMiniPortAdapterRefcount );
  246. // We start out disconnected so shutdown the media-connect waitref too.
  247. BrdgShutdownWaitRefOnce( &gMiniPortConnectedRefcount );
  248. NdisInitUnicodeString( &DeviceName, DEVICE_NAME );
  249. NdisInitUnicodeString( &LinkName, SYMBOLIC_NAME );
  250. // Must first tell NDIS we're a miniport driver and initializing
  251. NdisMInitializeWrapper( &gNDISWrapperHandle, gDriverObject, &gRegistryPath, NULL );
  252. // Fill in the description of our miniport
  253. NdisZeroMemory(&MiniPortChars, sizeof(MiniPortChars));
  254. MiniPortChars.MajorNdisVersion = 5;
  255. MiniPortChars.MinorNdisVersion = 0;
  256. MiniPortChars.HaltHandler = BrdgMiniHalt;
  257. MiniPortChars.InitializeHandler = BrdgMiniInitialize;
  258. MiniPortChars.QueryInformationHandler = BrdgMiniQueryInfo;
  259. MiniPortChars.ResetHandler = BrdgMiniReset;
  260. MiniPortChars.SendPacketsHandler = BrdgMiniSendPackets;
  261. MiniPortChars.SetInformationHandler = BrdgMiniSetInfo;
  262. //
  263. // Wire the ReturnPacketHandler straight into the forwarding engine
  264. //
  265. MiniPortChars.ReturnPacketHandler = BrdgFwdReturnIndicatedPacket;
  266. // Create a virtual NIC
  267. NdisStatus = NdisIMRegisterLayeredMiniport( gNDISWrapperHandle, &MiniPortChars, sizeof(MiniPortChars),
  268. &gMiniPortDriverHandle );
  269. if (NdisStatus != NDIS_STATUS_SUCCESS)
  270. {
  271. NdisWriteEventLogEntry( gDriverObject, EVENT_BRIDGE_MINIPORT_REGISTER_FAILED, 0, 0, NULL,
  272. sizeof(NDIS_STATUS), &NdisStatus );
  273. DBGPRINT(MINI, ("Failed to create an NDIS virtual NIC: %08x\n", NdisStatus));
  274. NdisTerminateWrapper( gNDISWrapperHandle, NULL );
  275. return NdisStatus;
  276. }
  277. //
  278. // Initialize Dispatch Table array before setting selected members
  279. //
  280. NdisZeroMemory( DispatchTable, sizeof( DispatchTable ) );
  281. //
  282. // Register a device object and symbolic link so user-mode code can talk to us
  283. //
  284. DispatchTable[IRP_MJ_CREATE] = BrdgDispatchRequest;
  285. DispatchTable[IRP_MJ_CLEANUP] = BrdgDispatchRequest;
  286. DispatchTable[IRP_MJ_CLOSE] = BrdgDispatchRequest;
  287. DispatchTable[IRP_MJ_DEVICE_CONTROL] = BrdgDispatchRequest;
  288. NdisStatus = NdisMRegisterDevice( gNDISWrapperHandle, &DeviceName, &LinkName, DispatchTable,
  289. &gDeviceObject, &gDeviceHandle );
  290. if( NdisStatus != NDIS_STATUS_SUCCESS )
  291. {
  292. NdisWriteEventLogEntry( gDriverObject, EVENT_BRIDGE_DEVICE_CREATION_FAILED, 0, 0, NULL,
  293. sizeof(NDIS_STATUS), &NdisStatus );
  294. DBGPRINT(MINI, ("Failed to create a device object and sym link: %08x\n", NdisStatus));
  295. NdisIMDeregisterLayeredMiniport( gMiniPortDriverHandle );
  296. NdisTerminateWrapper( gNDISWrapperHandle, NULL );
  297. return NdisStatus;
  298. }
  299. // Register the unload function
  300. NdisMRegisterUnloadHandler(gNDISWrapperHandle, BrdgUnload);
  301. return STATUS_SUCCESS;
  302. }
  303. VOID
  304. BrdgMiniCleanup()
  305. /*++
  306. Routine Description:
  307. Unload-time orderly shutdown function
  308. This function is guaranteed to be called exactly once
  309. Must run at PASSIVE_LEVEL since we call NdisIMDeInitializeDeviceInstance
  310. Arguments:
  311. None
  312. Return Value:
  313. None
  314. --*/
  315. {
  316. NDIS_STATUS NdisStatus;
  317. SAFEASSERT(CURRENT_IRQL == PASSIVE_LEVEL);
  318. DBGPRINT(MINI, ("BrdgMiniCleanup\n"));
  319. if( gMiniPortAdapterHandle != NULL )
  320. {
  321. SAFEASSERT( gNDISWrapperHandle != NULL );
  322. // This should cause a call to BrdgMiniHalt where gMiniPortAdapterHandle
  323. // is NULLed out
  324. NdisStatus = NdisIMDeInitializeDeviceInstance( gMiniPortAdapterHandle );
  325. SAFEASSERT( NdisStatus == NDIS_STATUS_SUCCESS );
  326. }
  327. else
  328. {
  329. //
  330. // Tear down our device object. This is normally done when the miniport
  331. // shuts down, but in scenarios where the miniport was never created,
  332. // the device object still exists at this point.
  333. //
  334. NDIS_HANDLE Scratch = gDeviceHandle;
  335. if( Scratch != NULL )
  336. {
  337. // Tear down the device object
  338. gDeviceHandle = gDeviceObject = NULL;
  339. NdisMDeregisterDevice( Scratch );
  340. }
  341. }
  342. // Unregister ourselves as an intermediate driver
  343. NdisIMDeregisterLayeredMiniport( gMiniPortDriverHandle );
  344. }
  345. BOOLEAN
  346. BrdgMiniIsBridgeDeviceName(
  347. IN PNDIS_STRING pDeviceName
  348. )
  349. /*++
  350. Routine Description:
  351. Compares a device name to the current device name of the bridge miniport.
  352. This actually requires that we allocate memory, so it should be called sparingly.
  353. Arguments:
  354. pDeviceName The name of a device
  355. Return Value:
  356. TRUE if the names match (case is ignored), FALSE otherwise.
  357. --*/
  358. {
  359. LOCK_STATE LockState;
  360. BOOLEAN rc = FALSE;
  361. NDIS_STATUS Status;
  362. ULONG BridgeNameCopySize = 0L;
  363. PWCHAR pBridgeNameCopy = NULL;
  364. // The bridge device name must be read inside the gBridgeStateLock
  365. NdisAcquireReadWriteLock( &gBridgeStateLock, FALSE/*Read access*/, &LockState );
  366. if( gBridgeDeviceName != NULL )
  367. {
  368. if( gBridgeDeviceNameSize > 0 )
  369. {
  370. // Alloc memory for the copy of the name
  371. Status = NdisAllocateMemoryWithTag( &pBridgeNameCopy, gBridgeDeviceNameSize, 'gdrB' );
  372. if( Status == NDIS_STATUS_SUCCESS )
  373. {
  374. // Copy the name
  375. NdisMoveMemory( pBridgeNameCopy, gBridgeDeviceName, gBridgeDeviceNameSize );
  376. BridgeNameCopySize = gBridgeDeviceNameSize;
  377. }
  378. else
  379. {
  380. SAFEASSERT( pBridgeNameCopy == NULL );
  381. }
  382. }
  383. else
  384. {
  385. SAFEASSERT( FALSE );
  386. }
  387. }
  388. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  389. if( pBridgeNameCopy != NULL )
  390. {
  391. NDIS_STRING NdisStr;
  392. NdisInitUnicodeString( &NdisStr, pBridgeNameCopy );
  393. if( NdisEqualString( &NdisStr, pDeviceName, TRUE/*Ignore case*/ ) )
  394. {
  395. rc = TRUE;
  396. }
  397. NdisFreeMemory( pBridgeNameCopy, BridgeNameCopySize, 0 );
  398. }
  399. return rc;
  400. }
  401. VOID
  402. BrdgMiniInstantiateMiniport()
  403. /*++
  404. Routine Description:
  405. Instantiates the virtual NIC we expose to overlying protocols.
  406. At least one adapter must be in the global adapter list, since we build
  407. our MAC address with the MAC address of the first bound adapter.
  408. Must run at < DISPATCH_LEVEL since we call NdisIMInitializeDeviceInstanceEx
  409. Arguments:
  410. None
  411. Return Value:
  412. None
  413. --*/
  414. {
  415. NDIS_STATUS Status;
  416. NTSTATUS NtStatus;
  417. NDIS_STRING NdisString;
  418. LOCK_STATE LockState;
  419. PWCHAR pDeviceName;
  420. ULONG DeviceNameSize;
  421. SAFEASSERT(CURRENT_IRQL < DISPATCH_LEVEL);
  422. DBGPRINT(MINI, ("About to instantiate the miniport...\n"));
  423. //
  424. // Retrieve our device name from the registry
  425. // (it is written there by our notify object during install)
  426. //
  427. NtStatus = BrdgReadRegUnicode( &gRegistryPath, gDeviceNameEntry, &pDeviceName, &DeviceNameSize );
  428. if( NtStatus != STATUS_SUCCESS )
  429. {
  430. NdisWriteEventLogEntry( gDriverObject, EVENT_BRIDGE_MINIPROT_DEVNAME_MISSING, 0, 0, NULL,
  431. sizeof(NTSTATUS), &NtStatus );
  432. DBGPRINT(MINI, ("Failed to retrieve the miniport's device name: %08x\n", NtStatus));
  433. return;
  434. }
  435. SAFEASSERT( pDeviceName != NULL );
  436. DBGPRINT(MINI, ("Initializing miniport with device name %ws\n", pDeviceName));
  437. NdisAcquireReadWriteLock( &gBridgeStateLock, TRUE/*Write access*/, &LockState );
  438. if( ! gHaveAddress )
  439. {
  440. // We don't have a MAC address yet. This is fatal.
  441. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  442. NdisWriteEventLogEntry( gDriverObject, EVENT_BRIDGE_NO_BRIDGE_MAC_ADDR, 0L, 0L, NULL,
  443. sizeof(NDIS_STATUS), &Status );
  444. DBGPRINT(MINI, ("Failed to determine a MAC address: %08x\n", Status));
  445. NdisFreeMemory( pDeviceName, DeviceNameSize, 0 );
  446. return;
  447. }
  448. //
  449. // Save the device name in our global for use until we reinitialize.
  450. // Must do this before calling NdisIMInitializeDeviceInstanceEx, since NDIS calls
  451. // BrdgProtBindAdapter in the context of our call to NdisIMInitializeDeviceInstanceEx
  452. // and we want to consult the bridge's device name when binding
  453. //
  454. if( gBridgeDeviceName != NULL )
  455. {
  456. // Free the old name
  457. NdisFreeMemory( gBridgeDeviceName, gBridgeDeviceNameSize, 0 );
  458. }
  459. gBridgeDeviceName = pDeviceName;
  460. gBridgeDeviceNameSize = DeviceNameSize;
  461. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  462. // Go ahead and intantiate the miniport.
  463. NdisInitUnicodeString( &NdisString, pDeviceName );
  464. Status = NdisIMInitializeDeviceInstanceEx(gMiniPortDriverHandle, &NdisString, NULL);
  465. if( Status != NDIS_STATUS_SUCCESS )
  466. {
  467. // Log this error since it means we can't create the miniport
  468. NdisWriteEventLogEntry( gDriverObject, EVENT_BRIDGE_MINIPORT_INIT_FAILED, 0L, 0L, NULL,
  469. sizeof(NDIS_STATUS), &Status );
  470. DBGPRINT(MINI, ("NdisIMInitializeDeviceInstanceEx failed: %08x\n", Status));
  471. // Destroy the stored device name for the miniport
  472. NdisAcquireReadWriteLock( &gBridgeStateLock, TRUE/*Write access*/, &LockState );
  473. if( gBridgeDeviceName != NULL )
  474. {
  475. // Free the old name
  476. NdisFreeMemory( gBridgeDeviceName, gBridgeDeviceNameSize, 0 );
  477. }
  478. gBridgeDeviceName = NULL;
  479. gBridgeDeviceNameSize = 0L;
  480. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  481. }
  482. }
  483. BOOLEAN
  484. BrdgMiniShouldIndicatePacket(
  485. IN PUCHAR pTargetAddr
  486. )
  487. /*++
  488. Routine Description:
  489. Determines whether an inbound packet should be indicated to overlying protocols through
  490. our virtual NIC
  491. Arguments:
  492. pTargetAddr The target MAC address of a packet
  493. Return Value:
  494. TRUE : The packet should be indicated
  495. FALSE : The packet should not be indicated
  496. --*/
  497. {
  498. BOOLEAN bIsBroadcast, bIsMulticast, bIsLocalUnicast, rc = FALSE;
  499. LOCK_STATE LockState;
  500. if( gMiniPortAdapterHandle == NULL )
  501. {
  502. // Yikes! The miniport isn't set up yet. Definitely don't indicate!
  503. return FALSE;
  504. }
  505. bIsBroadcast = ETH_IS_BROADCAST(pTargetAddr);
  506. bIsMulticast = ETH_IS_MULTICAST(pTargetAddr);
  507. bIsLocalUnicast = BrdgMiniIsUnicastToBridge(pTargetAddr);
  508. // Get read access to the packet filter
  509. NdisAcquireReadWriteLock( &gBridgeStateLock, FALSE /*Read-only*/, &LockState );
  510. do
  511. {
  512. // Promiscuous / ALL_LOCAL means indicate everything
  513. if( (gPacketFilter & (NDIS_PACKET_TYPE_PROMISCUOUS | NDIS_PACKET_TYPE_ALL_LOCAL)) != 0 )
  514. {
  515. rc = TRUE;
  516. break;
  517. }
  518. if( ((gPacketFilter & NDIS_PACKET_TYPE_BROADCAST) != 0) && bIsBroadcast )
  519. {
  520. rc = TRUE;
  521. break;
  522. }
  523. if( ((gPacketFilter & NDIS_PACKET_TYPE_DIRECTED) != 0) && bIsLocalUnicast )
  524. {
  525. rc = TRUE;
  526. break;
  527. }
  528. if( bIsMulticast )
  529. {
  530. if( (gPacketFilter & NDIS_PACKET_TYPE_ALL_MULTICAST) != 0 )
  531. {
  532. rc = TRUE;
  533. break;
  534. }
  535. else if( (gPacketFilter & NDIS_PACKET_TYPE_MULTICAST) != 0 )
  536. {
  537. rc = BrdgMiniAddrIsInMultiList( pTargetAddr );
  538. }
  539. }
  540. }
  541. while (FALSE);
  542. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  543. return rc;
  544. }
  545. BOOLEAN
  546. BrdgMiniIsUnicastToBridge (
  547. IN PUCHAR Address
  548. )
  549. /*++
  550. Routine Description:
  551. Determines whether a given packet is a directed packet unicast straight to
  552. the bridge's host machine
  553. Arguments:
  554. Address The target MAC address of a packet
  555. Return Value:
  556. TRUE : This is a directed packet destined for the local machine
  557. FALSE : The above is not true
  558. --*/
  559. {
  560. UINT Result;
  561. if( gHaveAddress )
  562. {
  563. // Not necessary to acquire a lock to read gBridgeAddress since it cannot
  564. // change once it is set.
  565. ETH_COMPARE_NETWORK_ADDRESSES_EQ( Address, gBridgeAddress, &Result );
  566. }
  567. else
  568. {
  569. // We have no MAC address, so this can't be addressed to us!
  570. Result = 1; // Inequality
  571. }
  572. return (BOOLEAN)(Result == 0); // Zero is equality
  573. }
  574. VOID
  575. BrdgMiniAssociate()
  576. /*++
  577. Routine Description:
  578. Associates our miniport with our protocol
  579. Must run at PASSIVE_LEVEL
  580. Arguments:
  581. None
  582. Return Value:
  583. None
  584. --*/
  585. {
  586. SAFEASSERT(CURRENT_IRQL == PASSIVE_LEVEL);
  587. // Associate ourselves with the protocol section in NDIS's tortured mind
  588. NdisIMAssociateMiniport( gMiniPortDriverHandle, gProtHandle );
  589. }
  590. VOID
  591. BrdgMiniDeferredMediaDisconnect(
  592. IN PVOID arg
  593. )
  594. /*++
  595. Routine Description:
  596. Signals a media-disconnect to NDIS
  597. Must run at PASSIVE IRQL, since we have to wait for all packet indications
  598. to complete before indicating media-disconnect.
  599. Arguments:
  600. arg The bridge miniport handle (must be released)
  601. Return Value:
  602. None
  603. --*/
  604. {
  605. NDIS_HANDLE MiniportHandle = (NDIS_HANDLE)arg;
  606. if( BrdgShutdownBlockedWaitRef(&gMiniPortConnectedRefcount) )
  607. {
  608. // Nobody can indicate packets anymore.
  609. LOCK_STATE LockState;
  610. //
  611. // Close a timing window: we may have just gone media-connect but our high-IRQL
  612. // processing may not yet have reset the wait-refcount. Serialize here so it's
  613. // impossible for us to signal a disconnect to NDIS after we have actually
  614. // gone media-connect.
  615. //
  616. // This RELIES on high-IRQL processing acquiring gBridgeStateLock to set
  617. // gBridgeMediaState to NdisMediaStateConnected BEFORE signalling the
  618. // media-connected state to NDIS.
  619. //
  620. NdisAcquireReadWriteLock( &gBridgeStateLock, FALSE /*Read access*/, &LockState );
  621. if( gBridgeMediaState == NdisMediaStateDisconnected )
  622. {
  623. DBGPRINT(MINI, ("Signalled media-disconnect from deferred function\n"));
  624. NdisMIndicateStatus( MiniportHandle, NDIS_STATUS_MEDIA_DISCONNECT, NULL, 0L );
  625. }
  626. else
  627. {
  628. DBGPRINT(MINI, ("Aborted deferred media-disconnect: media state inconsistent\n"));
  629. }
  630. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  631. }
  632. else
  633. {
  634. // Someone set us back to the connected state before we got executed
  635. DBGPRINT(MINI, ("Aborted deferred media-disconnect: wait-ref reset\n"));
  636. }
  637. BrdgMiniReleaseMiniport();
  638. }
  639. VOID
  640. BrdgMiniDeferredMediaToggle(
  641. IN PVOID arg
  642. )
  643. /*++
  644. Routine Description:
  645. Signals a media-disconnect to NDIS followed quickly by a media-connect. Used
  646. to indicate to upper-layer protocols like TCPIP that the bridge may have
  647. disconnected from a network segment it could previously reach, or that we may
  648. now be able to reach a network segment that we couldn't before.
  649. Must run at PASSIVE IRQL, since we have to wait for all packet indications
  650. to complete before indicating media-disconnect.
  651. Arguments:
  652. arg The bridge miniport handle (must be released)
  653. Return Value:
  654. None
  655. --*/
  656. {
  657. NDIS_HANDLE MiniportHandle = (NDIS_HANDLE)arg;
  658. // We need a guarantee that the miniport is media-connect to be able to do
  659. // the toggle properly.
  660. if( BrdgIncrementWaitRef(&gMiniPortConnectedRefcount) )
  661. {
  662. // Stop people from indicating packets
  663. if( BrdgShutdownWaitRef(&gMiniPortToggleRefcount) )
  664. {
  665. DBGPRINT(MINI, ("Doing deferred media toggle\n"));
  666. // Toggle our media state with NDIS
  667. NdisMIndicateStatus( MiniportHandle, NDIS_STATUS_MEDIA_DISCONNECT, NULL, 0L );
  668. NdisMIndicateStatus( MiniportHandle, NDIS_STATUS_MEDIA_CONNECT, NULL, 0L );
  669. // Allow people to indicate packets again
  670. BrdgResetWaitRef( &gMiniPortToggleRefcount );
  671. }
  672. else
  673. {
  674. DBGPRINT(MINI, ("Multiple toggles in progress simultaneously\n"));
  675. }
  676. BrdgDecrementWaitRef( &gMiniPortConnectedRefcount );
  677. }
  678. // else the miniport isn't media-connect, so the toggle makes no sense.
  679. BrdgMiniReleaseMiniport();
  680. }
  681. VOID
  682. BrdgMiniUpdateCharacteristics(
  683. IN BOOLEAN bConnectivityChange
  684. )
  685. /*++
  686. Routine Description:
  687. Recalculates the link speed and media status (connected / disconnected) that
  688. our virtual NIC exposes to overlying protocols
  689. Arguments:
  690. bConnectivityChange Whether the change that prompted this call is a change
  691. in connectivity (i.e., we acquired or lost an adapter).
  692. Return Value:
  693. None
  694. --*/
  695. {
  696. LOCK_STATE LockState, ListLockState, AdaptersLockState;
  697. PADAPT pAdapt;
  698. ULONG MediaState = NdisMediaStateDisconnected;
  699. ULONG FastestSpeed = 0L;
  700. BOOLEAN UpdateSpeed = FALSE, UpdateMediaState = FALSE;
  701. NDIS_HANDLE MiniportHandle;
  702. // Need to read the adapter list and also have the adapters' characteristics not change
  703. NdisAcquireReadWriteLock( &gAdapterListLock, FALSE /*Read-only*/, &ListLockState );
  704. NdisAcquireReadWriteLock( &gAdapterCharacteristicsLock, FALSE /*Read-only*/, &AdaptersLockState );
  705. pAdapt = gAdapterList;
  706. while( pAdapt != NULL )
  707. {
  708. // An adapter must be connected and actively handling packets to change our
  709. // virtual media state.
  710. if( (pAdapt->MediaState == NdisMediaStateConnected) && (pAdapt->State == Forwarding) )
  711. {
  712. // We're connected if at least one NIC is connected
  713. MediaState = NdisMediaStateConnected;
  714. // The NIC must be connected for us to consider its speed
  715. if( pAdapt->LinkSpeed > FastestSpeed )
  716. {
  717. FastestSpeed = pAdapt->LinkSpeed;
  718. }
  719. }
  720. pAdapt = pAdapt->Next;
  721. }
  722. NdisReleaseReadWriteLock( &gAdapterCharacteristicsLock, &AdaptersLockState );
  723. NdisReleaseReadWriteLock( &gAdapterListLock, &ListLockState );
  724. // Update the characteristics
  725. NdisAcquireReadWriteLock( &gBridgeStateLock, TRUE /*Write access*/, &LockState );
  726. //
  727. // Only update our virtual link speed if we actually got at least one real speed
  728. // from our NICs. If everything is disconnected, the resulting FastestSpeed is
  729. // zero. In this case, we don't want to actually report a zero speed up to
  730. // overlying protocols; we stick at the last known speed until someone reconnects.
  731. //
  732. if( (gBridgeLinkSpeed != FastestSpeed) && (FastestSpeed != 0L) )
  733. {
  734. UpdateSpeed = TRUE;
  735. gBridgeLinkSpeed = FastestSpeed;
  736. DBGPRINT(MINI, ("Updated bridge speed to %iMbps\n", FastestSpeed / 10000));
  737. }
  738. if( gBridgeMediaState != MediaState )
  739. {
  740. UpdateMediaState = TRUE;
  741. gBridgeMediaState = MediaState;
  742. if( MediaState == NdisMediaStateConnected )
  743. {
  744. DBGPRINT(MINI, ("CONNECT\n"));
  745. }
  746. else
  747. {
  748. DBGPRINT(MINI, ("DISCONNECT\n"));
  749. }
  750. }
  751. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  752. MiniportHandle = BrdgMiniAcquireMiniport();
  753. if( MiniportHandle != NULL )
  754. {
  755. if( UpdateMediaState )
  756. {
  757. // Our link state has changed.
  758. if( MediaState == NdisMediaStateConnected )
  759. {
  760. //
  761. // Tell NDIS we will be indicating packets again.
  762. //
  763. // NOTE: BrdgMiniDeferredMediaDisconnect RELIES on us doing this after
  764. // having updated gBridgeMediaState inside gBridgeStateLock so it can
  765. // close the timing window between this call and the BrdgResetWaitRef() call.
  766. //
  767. NdisMIndicateStatus( MiniportHandle, NDIS_STATUS_MEDIA_CONNECT, NULL, 0L );
  768. // Re-enable packet indication.
  769. BrdgResetWaitRef( &gMiniPortConnectedRefcount );
  770. }
  771. else
  772. {
  773. SAFEASSERT( MediaState == NdisMediaStateDisconnected );
  774. // Stop people from indicating packets
  775. BrdgBlockWaitRef( &gMiniPortConnectedRefcount );
  776. // We hand MiniportHandle to our deferred function
  777. BrdgMiniReAcquireMiniport();
  778. // Have to do the media-disconnect indication at PASSIVE level, since we must
  779. // first wait for everyone to finish indicating packets.
  780. if( BrdgDeferFunction( BrdgMiniDeferredMediaDisconnect, MiniportHandle ) != NDIS_STATUS_SUCCESS )
  781. {
  782. // Failed to defer the function. Avoid leaking a refcount
  783. BrdgMiniReleaseMiniport();
  784. }
  785. }
  786. }
  787. else if( bConnectivityChange )
  788. {
  789. //
  790. // There was no actual change to our media state. However, if the change that prompted this call
  791. // is a connectivity change and our media state is currently CONNECTED, we toggle it to
  792. // DISCONNECTED and back again to "hint" to upper-layer protocols like IP that the underlying
  793. // network may have changed. For example, in IP's case, a DHCP server may have become visible
  794. // (or a previously visible server may have disappeared) because of a connectivity change.
  795. // The hint causes IP to look for a DHCP server afresh.
  796. //
  797. if( MediaState == NdisMediaStateConnected )
  798. {
  799. // We hand MiniportHandle to our deferred function
  800. BrdgMiniReAcquireMiniport();
  801. // Toggle has to be done at PASSIVE level.
  802. if( BrdgDeferFunction( BrdgMiniDeferredMediaToggle, MiniportHandle ) != NDIS_STATUS_SUCCESS )
  803. {
  804. // Failed to defer the function. Avoid leaking a refcount
  805. BrdgMiniReleaseMiniport();
  806. }
  807. }
  808. }
  809. if( UpdateSpeed )
  810. {
  811. // Tell overlying protocols that our speed has changed
  812. NdisMIndicateStatus( MiniportHandle, NDIS_STATUS_LINK_SPEED_CHANGE, &FastestSpeed, sizeof(ULONG) );
  813. }
  814. BrdgMiniReleaseMiniport();
  815. }
  816. }
  817. NDIS_HANDLE
  818. BrdgMiniAcquireMiniportForIndicate()
  819. /*++
  820. Routine Description:
  821. Acquires the bridge miniport handle for the purpose of indicating packets.
  822. In addition to guaranteeing that the miniport will exist until the caller calls
  823. BrdgMiniReleaseMiniportForIndicate(), the caller is also allowed to indicate
  824. packets using the returned miniport handle until the miniport is released.
  825. Arguments:
  826. None
  827. Return Value:
  828. The NDIS handle for the virtual NIC. This can be used to indicate packets
  829. until a reciprocal call to BrdgMiniReleaseMiniportForIndicate().
  830. --*/
  831. {
  832. if( BrdgIncrementWaitRef(&gMiniPortAdapterRefcount) )
  833. {
  834. SAFEASSERT( gMiniPortAdapterHandle != NULL );
  835. // The miniport needs to be media-connect to indicate packets.
  836. if( BrdgIncrementWaitRef(&gMiniPortConnectedRefcount) )
  837. {
  838. // A media-state toggle had better not be in progress
  839. if( BrdgIncrementWaitRef(&gMiniPortToggleRefcount) )
  840. {
  841. // Caller can use the miniport
  842. return gMiniPortAdapterHandle;
  843. }
  844. // else miniport exists but is toggling its state
  845. BrdgDecrementWaitRef( &gMiniPortConnectedRefcount );
  846. }
  847. // else miniport exists but is media-disconnected
  848. BrdgDecrementWaitRef( &gMiniPortAdapterRefcount );
  849. }
  850. // else miniport does not exist.
  851. return NULL;
  852. }
  853. NDIS_HANDLE
  854. BrdgMiniAcquireMiniport()
  855. /*++
  856. Routine Description:
  857. Increments the miniport's refcount so it cannot be torn down until a corresponding
  858. BrdgMiniReleaseMiniport() call is made.
  859. The caller may NOT use the returned miniport handle to indicate packets, since the
  860. miniport is not guaranteed to be in an appropriate state.
  861. Arguments:
  862. None
  863. Return Value:
  864. The NDIS handle for the virtual NIC. This can be used safely until a reciprocal call
  865. to BrdgMiniReleaseMiniport().
  866. --*/
  867. {
  868. if( BrdgIncrementWaitRef(&gMiniPortAdapterRefcount) )
  869. {
  870. SAFEASSERT( gMiniPortAdapterHandle != NULL );
  871. return gMiniPortAdapterHandle;
  872. }
  873. // else miniport does not exist.
  874. return NULL;
  875. }
  876. VOID
  877. BrdgMiniReAcquireMiniport()
  878. /*++
  879. Routine Description:
  880. Reacquires the miniport (caller must have previously called BrdgMiniAcquireMiniport()
  881. and not yet called BrdgMiniReleaseMiniport().
  882. Arguments:
  883. None
  884. Return Value:
  885. None. The caller should already be holding a handle for the miniport.
  886. --*/
  887. {
  888. BrdgReincrementWaitRef(&gMiniPortAdapterRefcount);
  889. }
  890. VOID
  891. BrdgMiniReleaseMiniport()
  892. /*++
  893. Routine Description:
  894. Decrements the miniport's refcount. The caller should no longer use the handle
  895. previously returned by BrdgMiniAcquireMiniport().
  896. Arguments:
  897. None
  898. Return Value:
  899. None
  900. --*/
  901. {
  902. BrdgDecrementWaitRef( &gMiniPortAdapterRefcount );
  903. }
  904. VOID
  905. BrdgMiniReleaseMiniportForIndicate()
  906. /*++
  907. Routine Description:
  908. Decrements the miniport's refcount. The caller should no longer use the handle
  909. previously returned by BrdgMiniAcquireMiniportForIndicate().
  910. Arguments:
  911. None
  912. Return Value:
  913. None
  914. --*/
  915. {
  916. BrdgDecrementWaitRef( &gMiniPortToggleRefcount );
  917. BrdgDecrementWaitRef( &gMiniPortConnectedRefcount );
  918. BrdgDecrementWaitRef( &gMiniPortAdapterRefcount );
  919. }
  920. BOOLEAN
  921. BrdgMiniReadMACAddress(
  922. OUT PUCHAR pAddr
  923. )
  924. /*++
  925. Routine Description:
  926. Reads the bridge miniport's MAC address.
  927. Arguments:
  928. Address of a buffer to receive the address
  929. Return Value:
  930. TRUE if the value was copied out successfully
  931. FALSE if we don't yet have a MAC address (nothing was copied)
  932. --*/
  933. {
  934. BOOLEAN rc;
  935. if( gHaveAddress )
  936. {
  937. // Not necessary to acquire a lock to read the address since
  938. // it cannot change once it is set
  939. ETH_COPY_NETWORK_ADDRESS( pAddr, gBridgeAddress );
  940. rc = TRUE;
  941. }
  942. else
  943. {
  944. rc = FALSE;
  945. }
  946. return rc;
  947. }
  948. // ===========================================================================
  949. //
  950. // PRIVATE FUNCTIONS
  951. //
  952. // ===========================================================================
  953. VOID
  954. BrdgMiniInitFromAdapter(
  955. IN PADAPT pAdapt
  956. )
  957. /*++
  958. Routine Description:
  959. Called by the protocol section to give us a chance to establish the bridge's
  960. MAC address when a new adapter arrives. If we succeed in determining a MAC
  961. address from the given adapter, we in turn call the STA module to tell it
  962. our MAC address, which it needs as early as possible.
  963. The MAC address of the bridge is set as the MAC address of the given adapter
  964. with the "locally administered" bit set. This should (hopefully) make the
  965. address unique in the local network as well as unique within our machine.
  966. This function is called for every new adapter but we only need to initialize
  967. once.
  968. Arguments:
  969. pAdapt An adapter to use to initialize
  970. Return Value:
  971. Status of the operation
  972. --*/
  973. {
  974. if( ! gHaveAddress )
  975. {
  976. LOCK_STATE LockState;
  977. NdisAcquireReadWriteLock( &gBridgeStateLock, TRUE/*Write access*/, &LockState );
  978. // Possible for the gHaveAddress flag to have changed before acquiring the lock
  979. if( ! gHaveAddress )
  980. {
  981. // Copy out the NIC's MAC address
  982. ETH_COPY_NETWORK_ADDRESS( gBridgeAddress, pAdapt->MACAddr );
  983. //
  984. // Set the second-least significant bit of the NIC's MAC address. This moves the address
  985. // into the locally administered space.
  986. //
  987. gBridgeAddress[0] |= (UCHAR)0x02;
  988. DBGPRINT(MINI, ("Using MAC Address %02x-%02x-%02x-%02x-%02x-%02x\n",
  989. (UINT)gBridgeAddress[0], (UINT)gBridgeAddress[1], (UINT)gBridgeAddress[2],
  990. (UINT)gBridgeAddress[3], (UINT)gBridgeAddress[4], (UINT)gBridgeAddress[5]));
  991. gHaveAddress = TRUE;
  992. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  993. if( !gDisableSTA )
  994. {
  995. // We are responsible for calling the STA module to complete its initialization once
  996. // we know our MAC address.
  997. BrdgSTADeferredInit( gBridgeAddress );
  998. }
  999. // We are also responsible for letting the compatibility-mode code know about our
  1000. // MAC address once it is set.
  1001. BrdgCompNotifyMACAddress( gBridgeAddress );
  1002. }
  1003. else
  1004. {
  1005. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  1006. }
  1007. }
  1008. }
  1009. BOOLEAN
  1010. BrdgMiniAddrIsInMultiList(
  1011. IN PUCHAR pTargetAddr
  1012. )
  1013. /*++
  1014. Routine Description:
  1015. Determines whether a given multicast address is in the list of addresses that
  1016. we must indicate to overlying protocols
  1017. The caller is responsible for synchronization; it must have AT LEAST a read lock on
  1018. gBridgeStateLock.
  1019. Arguments:
  1020. pTargetAddr The address to analyze
  1021. Return Value:
  1022. TRUE : This address is a multicast address that we have been asked
  1023. to indicate
  1024. FALSE : The above is not true
  1025. --*/
  1026. {
  1027. PUCHAR pCurAddr = gMulticastList;
  1028. ULONG i;
  1029. BOOLEAN rc = FALSE;
  1030. // The list must have an integral number of addresses!
  1031. SAFEASSERT( (gMulticastListLength % ETH_LENGTH_OF_ADDRESS) == 0 );
  1032. for( i = 0;
  1033. i < (gMulticastListLength / ETH_LENGTH_OF_ADDRESS);
  1034. i++, pCurAddr += ETH_LENGTH_OF_ADDRESS
  1035. )
  1036. {
  1037. UINT Result;
  1038. ETH_COMPARE_NETWORK_ADDRESSES_EQ( pTargetAddr, pCurAddr, &Result );
  1039. if( Result == 0 )
  1040. {
  1041. rc = TRUE;
  1042. break;
  1043. }
  1044. }
  1045. return rc;
  1046. }
  1047. VOID
  1048. BrdgMiniHalt(
  1049. IN NDIS_HANDLE MiniportAdapterContext
  1050. )
  1051. /*++
  1052. Routine Description:
  1053. Called when the virtual NIC is de-instantiated. We NULL out the miniport handle and
  1054. stall the tear-down until everyone is done using the miniport.
  1055. Must be called at PASSIVE_LEVEL since we wait on an event
  1056. Arguments:
  1057. MiniportAdapterContext Ignored
  1058. Return Value:
  1059. None
  1060. --*/
  1061. {
  1062. NDIS_HANDLE Scratch = gDeviceHandle;
  1063. LOCK_STATE LockState;
  1064. SAFEASSERT(CURRENT_IRQL == PASSIVE_LEVEL);
  1065. DBGPRINT(MINI, ("BrdgMiniHalt\n"));
  1066. if( Scratch != NULL )
  1067. {
  1068. // Tear down the device object
  1069. gDeviceHandle = gDeviceObject = NULL;
  1070. NdisMDeregisterDevice( Scratch );
  1071. }
  1072. if( gMiniPortAdapterHandle != NULL )
  1073. {
  1074. // Stall before returning until everyone is done using the miniport handle.
  1075. // This also prevents people from acquiring the miniport handle.
  1076. BrdgShutdownWaitRefOnce( &gMiniPortAdapterRefcount );
  1077. gMiniPortAdapterHandle = NULL;
  1078. DBGPRINT(MINI, ("Done stall\n"));
  1079. }
  1080. NdisAcquireReadWriteLock( &gBridgeStateLock, TRUE/*Write access*/, &LockState );
  1081. if( gBridgeDeviceName != NULL )
  1082. {
  1083. NdisFreeMemory( gBridgeDeviceName, gBridgeDeviceNameSize, 0 );
  1084. gBridgeDeviceName = NULL;
  1085. gBridgeDeviceNameSize = 0L;
  1086. }
  1087. // Ditch our packet filter and multicast list
  1088. gPacketFilter = 0L;
  1089. if( gMulticastList != NULL )
  1090. {
  1091. SAFEASSERT( gMulticastListLength > 0L );
  1092. NdisFreeMemory( gMulticastList, gMulticastListLength, 0 );
  1093. gMulticastList = NULL;
  1094. gMulticastListLength = 0L;
  1095. }
  1096. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  1097. }
  1098. NDIS_STATUS
  1099. BrdgMiniInitialize(
  1100. OUT PNDIS_STATUS OpenErrorStatus,
  1101. OUT PUINT SelectedMediumIndex,
  1102. IN PNDIS_MEDIUM MediumArray,
  1103. IN UINT MediumArraySize,
  1104. IN NDIS_HANDLE MiniPortAdapterHandle,
  1105. IN NDIS_HANDLE WrapperConfigurationContext
  1106. )
  1107. /*++
  1108. Routine Description:
  1109. NDIS entry point called to initialize our virtual NIC following a call to
  1110. NdisIMInitializeDeviceInstance
  1111. Must run at PASSIVE_LEVEL since we call NdisMSetAttributesEx
  1112. Arguments:
  1113. OpenErrorStatus Where to return the specific error code if an open fails
  1114. SelectedMediumIndex Where to specify which media we selected from MediumArray
  1115. MediumArray A list of media types to choose from
  1116. MediumArraySize Number of entries in MediumArray
  1117. MiniPortAdapterHandle The handle for our virtual NIC (we save this)
  1118. WrapperConfigurationContext Not used
  1119. Return Value:
  1120. Status of the initialization. A result != NDIS_STATUS_SUCCESS fails the NIC initialization
  1121. and the miniport is not exposed to upper-layer protocols
  1122. --*/
  1123. {
  1124. UINT i;
  1125. SAFEASSERT(CURRENT_IRQL == PASSIVE_LEVEL);
  1126. DBGPRINT(MINI, ("BrdgMiniInitialize\n"));
  1127. for( i = 0; i < MediumArraySize; i++ )
  1128. {
  1129. if( MediumArray[i] == NdisMedium802_3 ) // Ethernet
  1130. {
  1131. *SelectedMediumIndex = NdisMedium802_3;
  1132. break;
  1133. }
  1134. }
  1135. if( i == MediumArraySize )
  1136. {
  1137. // Log this error since it's fatal
  1138. NdisWriteEventLogEntry( gDriverObject, EVENT_BRIDGE_ETHERNET_NOT_OFFERED, 0L, 0L, NULL, 0L, NULL );
  1139. DBGPRINT(MINI, ("Ethernet not offered; failing\n"));
  1140. return NDIS_STATUS_UNSUPPORTED_MEDIA;
  1141. }
  1142. NdisMSetAttributesEx( MiniPortAdapterHandle,
  1143. NULL,
  1144. 0, // CheckForHangTimeInSeconds
  1145. NDIS_ATTRIBUTE_IGNORE_PACKET_TIMEOUT |
  1146. NDIS_ATTRIBUTE_IGNORE_REQUEST_TIMEOUT|
  1147. NDIS_ATTRIBUTE_INTERMEDIATE_DRIVER |
  1148. NDIS_ATTRIBUTE_DESERIALIZE |
  1149. NDIS_ATTRIBUTE_NO_HALT_ON_SUSPEND,
  1150. 0);
  1151. // Save the adapter handle for future use
  1152. gMiniPortAdapterHandle = MiniPortAdapterHandle;
  1153. // Allow people to acquire the miniport
  1154. BrdgResetWaitRef( &gMiniPortAdapterRefcount );
  1155. return NDIS_STATUS_SUCCESS;
  1156. }
  1157. NDIS_STATUS
  1158. BrdgMiniReset(
  1159. OUT PBOOLEAN AddressingReset,
  1160. IN NDIS_HANDLE MiniportAdapterContext
  1161. )
  1162. /*++
  1163. Routine Description:
  1164. NDIS entry point called reset our miniport. We do nothing in response to this.
  1165. Arguments:
  1166. AddressingReset Whether NDIS needs to prod us some more by calling MiniportSetInformation
  1167. after we return to restore various pieces of state
  1168. MiniportAdapterContext Ignored
  1169. Return Value:
  1170. Status of the reset
  1171. --*/
  1172. {
  1173. DBGPRINT(MINI, ("BrdgMiniReset\n"));
  1174. return NDIS_STATUS_SUCCESS;
  1175. }
  1176. VOID
  1177. BrdgMiniSendPackets(
  1178. IN NDIS_HANDLE MiniportAdapterContext,
  1179. IN PPNDIS_PACKET PacketArray,
  1180. IN UINT NumberOfPackets
  1181. )
  1182. /*++
  1183. Routine Description:
  1184. NDIS entry point called to send packets through our virtual NIC.
  1185. We just call the forwarding logic code to handle each packet.
  1186. Arguments:
  1187. MiniportAdapterContext Ignored
  1188. PacketArray Array of packet pointers to send
  1189. NumberOfPacket Like it says
  1190. Return Value:
  1191. None
  1192. --*/
  1193. {
  1194. UINT i;
  1195. NDIS_STATUS Status;
  1196. for (i = 0; i < NumberOfPackets; i++)
  1197. {
  1198. PNDIS_PACKET pPacket = PacketArray[i];
  1199. // Hand this packet to the forwarding engine for processing
  1200. Status = BrdgFwdSendPacket( pPacket );
  1201. if( Status != NDIS_STATUS_PENDING )
  1202. {
  1203. // The forwarding engine completed immediately
  1204. // NDIS should prevent the miniport from being shut down
  1205. // until we return from this function
  1206. SAFEASSERT( gMiniPortAdapterHandle != NULL );
  1207. NdisMSendComplete(gMiniPortAdapterHandle, pPacket, Status);
  1208. }
  1209. // else the forwarding engine will call NdisMSendComplete()
  1210. }
  1211. }
  1212. NDIS_STATUS
  1213. BrdgMiniQueryInfo(
  1214. IN NDIS_HANDLE MiniportAdapterContext,
  1215. IN NDIS_OID Oid,
  1216. IN PVOID InformationBuffer,
  1217. IN ULONG InformationBufferLength,
  1218. OUT PULONG BytesWritten,
  1219. OUT PULONG BytesNeeded
  1220. )
  1221. /*++
  1222. Routine Description:
  1223. NDIS entry point called to retrieve various pieces of info from our miniport
  1224. Arguments:
  1225. MiniportAdapterContext Ignored
  1226. Oid The request code
  1227. InformationBuffer Place to return information
  1228. InformationBufferLength Size of InformationBuffer
  1229. BytesWritten Output of the number of written bytes
  1230. BytesNeeded If the provided buffer is too small, this is how many we need.
  1231. Return Value:
  1232. Status of the request
  1233. --*/
  1234. {
  1235. // Macros for use in this function alone
  1236. #define REQUIRE_AT_LEAST(n) \
  1237. { \
  1238. if(InformationBufferLength < (n)) \
  1239. { \
  1240. *BytesNeeded = (n); \
  1241. return NDIS_STATUS_INVALID_LENGTH; \
  1242. }\
  1243. }
  1244. #define RETURN_BYTES(p,n) \
  1245. { \
  1246. NdisMoveMemory( InformationBuffer, (p), (n) ); \
  1247. *BytesWritten = (n); \
  1248. return NDIS_STATUS_SUCCESS; \
  1249. }
  1250. switch( Oid )
  1251. {
  1252. // General characteristics
  1253. case OID_GEN_SUPPORTED_LIST:
  1254. {
  1255. REQUIRE_AT_LEAST( sizeof(gSupportedOIDs) );
  1256. RETURN_BYTES( gSupportedOIDs, sizeof(gSupportedOIDs));
  1257. }
  1258. break;
  1259. case OID_GEN_HARDWARE_STATUS:
  1260. {
  1261. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1262. *((ULONG*)InformationBuffer) = NdisHardwareStatusReady;
  1263. *BytesWritten = sizeof(ULONG);
  1264. return NDIS_STATUS_SUCCESS;
  1265. }
  1266. break;
  1267. case OID_GEN_MEDIA_SUPPORTED:
  1268. case OID_GEN_MEDIA_IN_USE:
  1269. {
  1270. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1271. // We support Ethernet only
  1272. *((ULONG*)InformationBuffer) = NdisMedium802_3;
  1273. *BytesWritten = sizeof(ULONG);
  1274. return NDIS_STATUS_SUCCESS;
  1275. }
  1276. break;
  1277. case OID_GEN_TRANSMIT_BUFFER_SPACE:
  1278. {
  1279. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1280. // Lie and claim to have 15K of send space, a common
  1281. // Ethernet card value.
  1282. // REVIEW: Is there a better value?
  1283. *((ULONG*)InformationBuffer) = 15 * 1024;
  1284. *BytesWritten = sizeof(ULONG);
  1285. return NDIS_STATUS_SUCCESS;
  1286. }
  1287. break;
  1288. case OID_GEN_RECEIVE_BUFFER_SPACE:
  1289. {
  1290. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1291. // Lie and claim to have 150K of receive space, a common
  1292. // Ethernet card value.
  1293. // REVIEW: Is there a better value?
  1294. *((ULONG*)InformationBuffer) = 150 * 1024;
  1295. *BytesWritten = sizeof(ULONG);
  1296. return NDIS_STATUS_SUCCESS;
  1297. }
  1298. break;
  1299. case OID_GEN_MAXIMUM_SEND_PACKETS:
  1300. case OID_802_3_MAXIMUM_LIST_SIZE:
  1301. {
  1302. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1303. // Return a generic large integer
  1304. // REVIEW: Is there a better value to hand out?
  1305. *((ULONG*)InformationBuffer) = 0x000000FF;
  1306. *BytesWritten = sizeof(ULONG);
  1307. return NDIS_STATUS_SUCCESS;
  1308. }
  1309. break;
  1310. case OID_GEN_MAXIMUM_FRAME_SIZE:
  1311. {
  1312. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1313. // Ethernet payloads can be up to 1500 bytes
  1314. *((ULONG*)InformationBuffer) = 1500L;
  1315. *BytesWritten = sizeof(ULONG);
  1316. return NDIS_STATUS_SUCCESS;
  1317. }
  1318. break;
  1319. // We indicate full packets up to NDIS, so these values are the same and
  1320. // equal to the maximum size of a packet
  1321. case OID_GEN_MAXIMUM_LOOKAHEAD:
  1322. case OID_GEN_CURRENT_LOOKAHEAD:
  1323. // These are also just the maximum total size of a frame
  1324. case OID_GEN_MAXIMUM_TOTAL_SIZE:
  1325. case OID_GEN_RECEIVE_BLOCK_SIZE:
  1326. case OID_GEN_TRANSMIT_BLOCK_SIZE:
  1327. {
  1328. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1329. // Ethernet frames with header can be up to 1514 bytes
  1330. *((ULONG*)InformationBuffer) = 1514L;
  1331. *BytesWritten = sizeof(ULONG);
  1332. return NDIS_STATUS_SUCCESS;
  1333. }
  1334. break;
  1335. case OID_GEN_MAC_OPTIONS:
  1336. {
  1337. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1338. // We have no internal loopback support
  1339. *((ULONG*)InformationBuffer) = NDIS_MAC_OPTION_NO_LOOPBACK;
  1340. *BytesWritten = sizeof(ULONG);
  1341. return NDIS_STATUS_SUCCESS;
  1342. }
  1343. break;
  1344. case OID_GEN_LINK_SPEED:
  1345. {
  1346. LOCK_STATE LockState;
  1347. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1348. NdisAcquireReadWriteLock( &gBridgeStateLock, FALSE /*Read only*/, &LockState );
  1349. *((ULONG*)InformationBuffer) = gBridgeLinkSpeed;
  1350. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  1351. *BytesWritten = sizeof(ULONG);
  1352. return NDIS_STATUS_SUCCESS;
  1353. }
  1354. break;
  1355. // Ethernet characteristics
  1356. case OID_802_3_PERMANENT_ADDRESS:
  1357. case OID_802_3_CURRENT_ADDRESS:
  1358. {
  1359. SAFEASSERT( gHaveAddress );
  1360. // Don't need a read lock because the address shouldn't change once set
  1361. REQUIRE_AT_LEAST( sizeof(gBridgeAddress) );
  1362. RETURN_BYTES( gBridgeAddress, sizeof(gBridgeAddress));
  1363. }
  1364. break;
  1365. case OID_GEN_MEDIA_CONNECT_STATUS:
  1366. {
  1367. LOCK_STATE LockState;
  1368. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1369. NdisAcquireReadWriteLock( &gBridgeStateLock, FALSE /*Read only*/, &LockState );
  1370. *((ULONG*)InformationBuffer) = gBridgeMediaState;
  1371. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  1372. *BytesWritten = sizeof(ULONG);
  1373. return NDIS_STATUS_SUCCESS;
  1374. }
  1375. break;
  1376. case OID_GEN_VENDOR_ID:
  1377. {
  1378. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1379. // We don't have an IEEE-assigned ID so use this constant
  1380. *((ULONG*)InformationBuffer) = 0xFFFFFF;
  1381. *BytesWritten = sizeof(ULONG);
  1382. return NDIS_STATUS_SUCCESS;
  1383. }
  1384. break;
  1385. case OID_GEN_VENDOR_DESCRIPTION:
  1386. {
  1387. UINT len = (UINT)strlen( gDriverDescription ) + 1;
  1388. REQUIRE_AT_LEAST( len );
  1389. RETURN_BYTES( gDriverDescription, len);
  1390. }
  1391. break;
  1392. case OID_GEN_VENDOR_DRIVER_VERSION:
  1393. {
  1394. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1395. // We are version 1.0
  1396. *((ULONG*)InformationBuffer) = 0x00010000;
  1397. *BytesWritten = sizeof(ULONG);
  1398. return NDIS_STATUS_SUCCESS;
  1399. }
  1400. break;
  1401. case OID_GEN_DRIVER_VERSION:
  1402. {
  1403. REQUIRE_AT_LEAST( sizeof(USHORT) );
  1404. // We are using version 5.0 of NDIS
  1405. *((USHORT*)InformationBuffer) = 0x0500;
  1406. *BytesWritten = sizeof(USHORT);
  1407. return NDIS_STATUS_SUCCESS;
  1408. }
  1409. break;
  1410. //
  1411. // General Statistics
  1412. //
  1413. case OID_GEN_XMIT_OK:
  1414. {
  1415. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1416. // Reply with the number of local-sourced sent frames
  1417. *((ULONG*)InformationBuffer) = gStatTransmittedFrames.LowPart;
  1418. *BytesWritten = sizeof(ULONG);
  1419. return NDIS_STATUS_SUCCESS;
  1420. }
  1421. break;
  1422. case OID_GEN_XMIT_ERROR:
  1423. {
  1424. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1425. // Reply with the number of local-sourced frames sent with errors
  1426. *((ULONG*)InformationBuffer) = gStatTransmittedErrorFrames.LowPart;
  1427. *BytesWritten = sizeof(ULONG);
  1428. return NDIS_STATUS_SUCCESS;
  1429. }
  1430. break;
  1431. case OID_GEN_RCV_OK:
  1432. {
  1433. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1434. // Reply with the number of indicated frames
  1435. *((ULONG*)InformationBuffer) = gStatIndicatedFrames.LowPart;
  1436. *BytesWritten = sizeof(ULONG);
  1437. return NDIS_STATUS_SUCCESS;
  1438. }
  1439. break;
  1440. // Answer the same for these two
  1441. case OID_GEN_RCV_NO_BUFFER:
  1442. case OID_GEN_RCV_ERROR:
  1443. {
  1444. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1445. // Reply with the number of packets we wanted to indicate but couldn't
  1446. *((ULONG*)InformationBuffer) = gStatIndicatedDroppedFrames.LowPart;
  1447. *BytesWritten = sizeof(ULONG);
  1448. return NDIS_STATUS_SUCCESS;
  1449. }
  1450. break;
  1451. case OID_GEN_DIRECTED_BYTES_XMIT:
  1452. {
  1453. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1454. *((ULONG*)InformationBuffer) = gStatDirectedTransmittedBytes.LowPart;
  1455. *BytesWritten = sizeof(ULONG);
  1456. return NDIS_STATUS_SUCCESS;
  1457. }
  1458. break;
  1459. case OID_GEN_DIRECTED_FRAMES_XMIT:
  1460. {
  1461. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1462. // Reply with the number of packets we wanted to indicate but couldn't
  1463. *((ULONG*)InformationBuffer) = gStatDirectedTransmittedFrames.LowPart;
  1464. *BytesWritten = sizeof(ULONG);
  1465. return NDIS_STATUS_SUCCESS;
  1466. }
  1467. break;
  1468. case OID_GEN_MULTICAST_BYTES_XMIT:
  1469. {
  1470. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1471. // Reply with the number of packets we wanted to indicate but couldn't
  1472. *((ULONG*)InformationBuffer) = gStatMulticastTransmittedBytes.LowPart;
  1473. *BytesWritten = sizeof(ULONG);
  1474. return NDIS_STATUS_SUCCESS;
  1475. }
  1476. break;
  1477. case OID_GEN_MULTICAST_FRAMES_XMIT:
  1478. {
  1479. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1480. // Reply with the number of packets we wanted to indicate but couldn't
  1481. *((ULONG*)InformationBuffer) = gStatMulticastTransmittedFrames.LowPart;
  1482. *BytesWritten = sizeof(ULONG);
  1483. return NDIS_STATUS_SUCCESS;
  1484. }
  1485. break;
  1486. case OID_GEN_BROADCAST_BYTES_XMIT:
  1487. {
  1488. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1489. // Reply with the number of packets we wanted to indicate but couldn't
  1490. *((ULONG*)InformationBuffer) = gStatBroadcastTransmittedBytes.LowPart;
  1491. *BytesWritten = sizeof(ULONG);
  1492. return NDIS_STATUS_SUCCESS;
  1493. }
  1494. break;
  1495. case OID_GEN_BROADCAST_FRAMES_XMIT:
  1496. {
  1497. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1498. // Reply with the number of packets we wanted to indicate but couldn't
  1499. *((ULONG*)InformationBuffer) = gStatBroadcastTransmittedFrames.LowPart;
  1500. *BytesWritten = sizeof(ULONG);
  1501. return NDIS_STATUS_SUCCESS;
  1502. }
  1503. break;
  1504. case OID_GEN_DIRECTED_BYTES_RCV:
  1505. {
  1506. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1507. // Reply with the number of packets we wanted to indicate but couldn't
  1508. *((ULONG*)InformationBuffer) = gStatDirectedIndicatedBytes.LowPart;
  1509. *BytesWritten = sizeof(ULONG);
  1510. return NDIS_STATUS_SUCCESS;
  1511. }
  1512. break;
  1513. case OID_GEN_DIRECTED_FRAMES_RCV:
  1514. {
  1515. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1516. // Reply with the number of packets we wanted to indicate but couldn't
  1517. *((ULONG*)InformationBuffer) = gStatDirectedIndicatedFrames.LowPart;
  1518. *BytesWritten = sizeof(ULONG);
  1519. return NDIS_STATUS_SUCCESS;
  1520. }
  1521. break;
  1522. case OID_GEN_MULTICAST_BYTES_RCV:
  1523. {
  1524. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1525. // Reply with the number of packets we wanted to indicate but couldn't
  1526. *((ULONG*)InformationBuffer) = gStatMulticastIndicatedBytes.LowPart;
  1527. *BytesWritten = sizeof(ULONG);
  1528. return NDIS_STATUS_SUCCESS;
  1529. }
  1530. break;
  1531. case OID_GEN_MULTICAST_FRAMES_RCV:
  1532. {
  1533. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1534. // Reply with the number of packets we wanted to indicate but couldn't
  1535. *((ULONG*)InformationBuffer) = gStatMulticastIndicatedFrames.LowPart;
  1536. *BytesWritten = sizeof(ULONG);
  1537. return NDIS_STATUS_SUCCESS;
  1538. }
  1539. break;
  1540. case OID_GEN_BROADCAST_BYTES_RCV:
  1541. {
  1542. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1543. // Reply with the number of packets we wanted to indicate but couldn't
  1544. *((ULONG*)InformationBuffer) = gStatBroadcastIndicatedBytes.LowPart;
  1545. *BytesWritten = sizeof(ULONG);
  1546. return NDIS_STATUS_SUCCESS;
  1547. }
  1548. break;
  1549. case OID_GEN_BROADCAST_FRAMES_RCV:
  1550. {
  1551. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1552. // Reply with the number of packets we wanted to indicate but couldn't
  1553. *((ULONG*)InformationBuffer) = gStatBroadcastIndicatedFrames.LowPart;
  1554. *BytesWritten = sizeof(ULONG);
  1555. return NDIS_STATUS_SUCCESS;
  1556. }
  1557. break;
  1558. // Ethernet statistics
  1559. case OID_802_3_RCV_ERROR_ALIGNMENT:
  1560. case OID_802_3_XMIT_ONE_COLLISION:
  1561. case OID_802_3_XMIT_MORE_COLLISIONS:
  1562. {
  1563. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1564. // We have no way of collecting this information sensibly from lower NICs, so
  1565. // pretend these types of events never happen.
  1566. *((ULONG*)InformationBuffer) = 0L;
  1567. *BytesWritten = sizeof(ULONG);
  1568. return NDIS_STATUS_SUCCESS;
  1569. }
  1570. break;
  1571. case OID_GEN_CURRENT_PACKET_FILTER:
  1572. {
  1573. LOCK_STATE LockState;
  1574. REQUIRE_AT_LEAST( sizeof(ULONG) );
  1575. NdisAcquireReadWriteLock( &gBridgeStateLock, FALSE /*Read only*/, &LockState );
  1576. *((ULONG*)InformationBuffer) = gPacketFilter;
  1577. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  1578. *BytesWritten = sizeof(ULONG);
  1579. return NDIS_STATUS_SUCCESS;
  1580. }
  1581. break;
  1582. case OID_802_3_MULTICAST_LIST:
  1583. {
  1584. LOCK_STATE LockState;
  1585. NdisAcquireReadWriteLock( &gBridgeStateLock, FALSE /*Read only*/, &LockState );
  1586. if(InformationBufferLength < gMulticastListLength)
  1587. {
  1588. *BytesNeeded = gMulticastListLength;
  1589. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  1590. return NDIS_STATUS_INVALID_LENGTH;
  1591. }
  1592. NdisMoveMemory( InformationBuffer, gMulticastList, gMulticastListLength );
  1593. *BytesWritten = gMulticastListLength;
  1594. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  1595. return NDIS_STATUS_SUCCESS;
  1596. }
  1597. break;
  1598. case OID_PNP_QUERY_POWER:
  1599. {
  1600. return NDIS_STATUS_SUCCESS;
  1601. }
  1602. break;
  1603. case OID_TCP_TASK_OFFLOAD:
  1604. {
  1605. // Mark that Tcp.ip has been loaded
  1606. g_fIsTcpIpLoaded = TRUE;
  1607. // Set the underlying 1394 miniport to ON
  1608. BrdgSetMiniportsToBridgeMode(NULL,TRUE);
  1609. return NDIS_STATUS_NOT_SUPPORTED;
  1610. }
  1611. break;
  1612. }
  1613. // We don't understand the OID
  1614. return NDIS_STATUS_NOT_SUPPORTED;
  1615. #undef REQUIRE_AT_LEAST
  1616. #undef RETURN_BYTES
  1617. }
  1618. NDIS_STATUS
  1619. BrdgMiniSetInfo(
  1620. IN NDIS_HANDLE MiniportAdapterContext,
  1621. IN NDIS_OID Oid,
  1622. IN PVOID InformationBuffer,
  1623. IN ULONG InformationBufferLength,
  1624. OUT PULONG BytesRead,
  1625. OUT PULONG BytesNeeded
  1626. )
  1627. /*++
  1628. Routine Description:
  1629. NDIS entry point called to set various pieces of info to our miniport
  1630. Arguments:
  1631. MiniportAdapterContext Ignored
  1632. Oid The request code
  1633. InformationBuffer Input information buffer
  1634. InformationBufferLength Size of InformationBuffer
  1635. BytesRead Number of bytes read from InformationBuffer
  1636. BytesNeeded If the provided buffer is too small, this is how many we need.
  1637. Return Value:
  1638. Status of the request
  1639. --*/
  1640. {
  1641. LOCK_STATE LockState;
  1642. NDIS_STATUS Status;
  1643. switch( Oid )
  1644. {
  1645. case OID_GEN_CURRENT_PACKET_FILTER:
  1646. {
  1647. SAFEASSERT( InformationBufferLength == sizeof(ULONG) );
  1648. // Get write access to the packet filter
  1649. NdisAcquireReadWriteLock( &gBridgeStateLock, TRUE /*Read-Write*/, &LockState );
  1650. gPacketFilter = *((ULONG*)InformationBuffer);
  1651. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  1652. DBGPRINT(MINI, ("Set the packet filter to %08x\n", gPacketFilter));
  1653. *BytesRead = sizeof(ULONG);
  1654. return NDIS_STATUS_SUCCESS;
  1655. }
  1656. break;
  1657. case OID_802_3_MULTICAST_LIST:
  1658. {
  1659. PUCHAR pOldList, pNewList;
  1660. ULONG oldListLength;
  1661. // The incoming buffer should contain an integral number of Ethernet MAC
  1662. // addresses
  1663. SAFEASSERT( (InformationBufferLength % ETH_LENGTH_OF_ADDRESS) == 0 );
  1664. DBGPRINT(MINI, ("Modifying the multicast list; now has %i entries\n",
  1665. InformationBufferLength / ETH_LENGTH_OF_ADDRESS));
  1666. // Alloc and copy to the new multicast list
  1667. if( InformationBufferLength > 0 )
  1668. {
  1669. Status = NdisAllocateMemoryWithTag( &pNewList, InformationBufferLength, 'gdrB' );
  1670. if( Status != NDIS_STATUS_SUCCESS )
  1671. {
  1672. DBGPRINT(MINI, ("NdisAllocateMemoryWithTag failed while recording multicast list\n"));
  1673. return NDIS_STATUS_NOT_ACCEPTED;
  1674. }
  1675. // Copy the list
  1676. NdisMoveMemory( pNewList, InformationBuffer, InformationBufferLength );
  1677. }
  1678. else
  1679. {
  1680. pNewList = NULL;
  1681. }
  1682. // Swap in the new list
  1683. NdisAcquireReadWriteLock( &gBridgeStateLock, TRUE /*Read-Write*/, &LockState );
  1684. pOldList = gMulticastList;
  1685. oldListLength = gMulticastListLength;
  1686. gMulticastList = pNewList;
  1687. gMulticastListLength = InformationBufferLength;
  1688. NdisReleaseReadWriteLock( &gBridgeStateLock, &LockState );
  1689. // Free the old multicast list if there was one
  1690. if( pOldList != NULL )
  1691. {
  1692. NdisFreeMemory( pOldList, oldListLength, 0 );
  1693. }
  1694. *BytesRead = InformationBufferLength;
  1695. return NDIS_STATUS_SUCCESS;
  1696. }
  1697. break;
  1698. case OID_GEN_CURRENT_LOOKAHEAD:
  1699. case OID_GEN_PROTOCOL_OPTIONS:
  1700. {
  1701. // We accept these but do nothing
  1702. return NDIS_STATUS_SUCCESS;
  1703. }
  1704. break;
  1705. // Overlying protocols telling us about their network addresses
  1706. case OID_GEN_NETWORK_LAYER_ADDRESSES:
  1707. {
  1708. // Let the compatibility-mode code note the addresses
  1709. BrdgCompNotifyNetworkAddresses( InformationBuffer, InformationBufferLength );
  1710. }
  1711. //
  1712. // DELIBERATELY FALL THROUGH
  1713. //
  1714. // All relayed OIDs go here
  1715. case OID_GEN_TRANSPORT_HEADER_OFFSET:
  1716. {
  1717. LOCK_STATE LockState;
  1718. PADAPT Adapters[MAX_ADAPTERS], pAdapt;
  1719. LONG NumAdapters = 0L, i;
  1720. PNDIS_REQUEST_BETTER pRequest;
  1721. NDIS_STATUS Status, rc;
  1722. // We read the entire request
  1723. *BytesRead = InformationBufferLength;
  1724. // Pass these straight through to underlying NICs
  1725. NdisAcquireReadWriteLock( &gAdapterListLock, FALSE /*Read only*/, &LockState );
  1726. // Make a list of the adapters to send the request to
  1727. pAdapt = gAdapterList;
  1728. while( pAdapt != NULL )
  1729. {
  1730. if( NumAdapters < MAX_ADAPTERS )
  1731. {
  1732. Adapters[NumAdapters] = pAdapt;
  1733. // We will be using this adapter outside the list lock
  1734. BrdgAcquireAdapterInLock( pAdapt );
  1735. NumAdapters++;
  1736. }
  1737. else
  1738. {
  1739. SAFEASSERT( FALSE );
  1740. DBGPRINT(MINI, ("Too many adapters to relay a SetInfo request to!\n"));
  1741. }
  1742. pAdapt = pAdapt->Next;
  1743. }
  1744. // The refcount is the number of requests we will make
  1745. gRequestRefCount = NumAdapters;
  1746. NdisReleaseReadWriteLock( &gAdapterListLock, &LockState );
  1747. if( NumAdapters == 0 )
  1748. {
  1749. // Nothing to do!
  1750. rc = NDIS_STATUS_SUCCESS;
  1751. }
  1752. else
  1753. {
  1754. // Request will pend unless all adapters return immediately
  1755. rc = NDIS_STATUS_PENDING;
  1756. for( i = 0L; i < NumAdapters; i++ )
  1757. {
  1758. // Allocate memory for the request
  1759. Status = NdisAllocateMemoryWithTag( &pRequest, sizeof(NDIS_REQUEST_BETTER), 'gdrB' );
  1760. if( Status != NDIS_STATUS_SUCCESS )
  1761. {
  1762. LONG NewCount = InterlockedDecrement( &gRequestRefCount );
  1763. DBGPRINT(MINI, ("NdisAllocateMemoryWithTag failed while relaying an OID\n"));
  1764. if( NewCount == 0 )
  1765. {
  1766. // This could only have happened with the last adapter
  1767. SAFEASSERT( i == NumAdapters - 1 );
  1768. // We're all done since everyone else has completed too
  1769. rc = NDIS_STATUS_SUCCESS;
  1770. }
  1771. // Let go of the adapter
  1772. BrdgReleaseAdapter( Adapters[i] );
  1773. continue;
  1774. }
  1775. // Set up the request as a mirror of ours
  1776. pRequest->Request.RequestType = NdisRequestSetInformation;
  1777. pRequest->Request.DATA.SET_INFORMATION.Oid = Oid ;
  1778. pRequest->Request.DATA.SET_INFORMATION.InformationBuffer = InformationBuffer;
  1779. pRequest->Request.DATA.SET_INFORMATION.InformationBufferLength = InformationBufferLength;
  1780. NdisInitializeEvent( &pRequest->Event );
  1781. NdisResetEvent( &pRequest->Event );
  1782. pRequest->pFunc = BrdgMiniRelayedRequestComplete;
  1783. pRequest->FuncArg = NULL;
  1784. // Fire it off
  1785. NdisRequest( &Status, Adapters[i]->BindingHandle, &pRequest->Request );
  1786. // Let go of the adapter; NDIS should not permit it to be unbound while
  1787. // a request is in progress
  1788. BrdgReleaseAdapter( Adapters[i] );
  1789. if( Status != NDIS_STATUS_PENDING )
  1790. {
  1791. // The cleanup function won't get called
  1792. BrdgMiniRelayedRequestComplete( pRequest, NULL );
  1793. }
  1794. }
  1795. }
  1796. //
  1797. // Paranoia for future maintainance: can't refer to pointer parameters
  1798. // at this point, as the relayed requests may have completed already, making
  1799. // them stale.
  1800. //
  1801. InformationBuffer = NULL;
  1802. BytesRead = NULL;
  1803. BytesNeeded = NULL;
  1804. return rc;
  1805. }
  1806. break;
  1807. case OID_PNP_SET_POWER:
  1808. {
  1809. return NDIS_STATUS_SUCCESS;
  1810. }
  1811. break;
  1812. }
  1813. return NDIS_STATUS_NOT_SUPPORTED;
  1814. }
  1815. VOID
  1816. BrdgMiniRelayedRequestComplete(
  1817. PNDIS_REQUEST_BETTER pRequest,
  1818. PVOID unused
  1819. )
  1820. /*++
  1821. Routine Description:
  1822. Called when a SetInformation request that we relayed completes.
  1823. Arguments:
  1824. pRequest The NDIS_REQUEST_BETTER structure we allocated
  1825. in BrdgMiniSetInformation().
  1826. unused Unused
  1827. Return Value:
  1828. Status of the request
  1829. --*/
  1830. {
  1831. LONG NewCount = InterlockedDecrement( &gRequestRefCount );
  1832. if( NewCount == 0 )
  1833. {
  1834. // NDIS Should not permit the miniport to shut down with a request
  1835. // in progress
  1836. SAFEASSERT( gMiniPortAdapterHandle != NULL );
  1837. // The operation always succeeds
  1838. NdisMSetInformationComplete( gMiniPortAdapterHandle, NDIS_STATUS_SUCCESS );
  1839. }
  1840. // Free the request structure since we allocated it ourselves
  1841. NdisFreeMemory( pRequest, sizeof(PNDIS_REQUEST_BETTER), 0 );
  1842. }
  1843. VOID
  1844. BrdgMiniLocalRequestComplete(
  1845. PNDIS_REQUEST_BETTER pRequest,
  1846. PVOID pContext
  1847. )
  1848. /*++
  1849. Routine Description:
  1850. Called when bridge allocated request completes.
  1851. Arguments:
  1852. pRequest The NDIS_REQUEST_BETTER structure we allocated
  1853. in BrdgSetMiniportsToBridgeMode.
  1854. Context pAdapt structure
  1855. Return Value:
  1856. Status of the request
  1857. --*/
  1858. {
  1859. PADAPT pAdapt = (PADAPT)pContext;
  1860. // Let go of the adapter;
  1861. BrdgReleaseAdapter( pAdapt);
  1862. // Free the request structure since we allocated it ourselves
  1863. NdisFreeMemory( pRequest, sizeof(PNDIS_REQUEST_BETTER), 0 );
  1864. }
  1865. VOID
  1866. BrdgSetMiniportsToBridgeMode(
  1867. PADAPT pAdapt,
  1868. BOOLEAN fSet
  1869. )
  1870. /*++
  1871. Routine Description:
  1872. Sends a 1394 specific OID to the miniport informing it that TCP/IP
  1873. has been activated
  1874. Arguments:
  1875. pAdapt - If adapt is not NULL, send Request to this adapt.
  1876. Otherwise send it to all of them.
  1877. fSet - if True, then set Bridge Mode ON, otherwise set it OFF
  1878. Return Value:
  1879. Status of the request
  1880. --*/
  1881. {
  1882. LOCK_STATE LockState;
  1883. PADAPT Adapters[MAX_ADAPTERS];
  1884. LONG NumAdapters = 0L, i;
  1885. NDIS_OID Oid;
  1886. if (pAdapt != NULL)
  1887. {
  1888. if (pAdapt->PhysicalMedium == NdisPhysicalMedium1394)
  1889. {
  1890. // We have a 1394 adapt, ref it and send the request to it
  1891. if (BrdgAcquireAdapter (pAdapt))
  1892. {
  1893. Adapters[0] = pAdapt;
  1894. NumAdapters = 1;
  1895. }
  1896. }
  1897. }
  1898. else
  1899. {
  1900. // walk through the list and Acquire all the 1394 adapts
  1901. NdisAcquireReadWriteLock( &gAdapterListLock, FALSE /*Read only*/, &LockState );
  1902. // Make a list of the adapters to send the request to
  1903. pAdapt = gAdapterList;
  1904. while( pAdapt != NULL )
  1905. {
  1906. if( NumAdapters < MAX_ADAPTERS && pAdapt->PhysicalMedium == NdisPhysicalMedium1394)
  1907. {
  1908. Adapters[NumAdapters] = pAdapt;
  1909. // We will be using this adapter outside the list lock
  1910. BrdgAcquireAdapterInLock( pAdapt ); // cannot fail
  1911. NumAdapters++;
  1912. }
  1913. pAdapt = pAdapt->Next;
  1914. }
  1915. NdisReleaseReadWriteLock( &gAdapterListLock, &LockState );
  1916. }
  1917. if (NumAdapters == 0)
  1918. {
  1919. return;
  1920. }
  1921. if (fSet == TRUE)
  1922. {
  1923. Oid = OID_1394_ENTER_BRIDGE_MODE ;
  1924. DBGPRINT(MINI, ("Setting 1394 miniport bridge mode - ON !\n"));
  1925. }
  1926. else
  1927. {
  1928. Oid = OID_1394_EXIT_BRIDGE_MODE ;
  1929. DBGPRINT(MINI, ("Setting 1394 miniport bridge mode - OFF !\n"));
  1930. }
  1931. for( i = 0L; i < NumAdapters; i++ )
  1932. {
  1933. NDIS_STATUS Status;
  1934. PNDIS_REQUEST_BETTER pRequest;
  1935. Status = NdisAllocateMemoryWithTag( &pRequest, sizeof(NDIS_REQUEST_BETTER), 'gdrB' );
  1936. if( Status != NDIS_STATUS_SUCCESS )
  1937. {
  1938. DBGPRINT(MINI, ("NdisAllocateMemoryWithTag failed while allocating a request structure \n"));
  1939. // Let go of the adapter
  1940. BrdgReleaseAdapter( Adapters[i] );
  1941. continue;
  1942. }
  1943. // Set up the request
  1944. pRequest->Request.RequestType = NdisRequestSetInformation;
  1945. pRequest->Request.DATA.SET_INFORMATION.Oid = Oid;
  1946. pRequest->Request.DATA.SET_INFORMATION.InformationBuffer = NULL;
  1947. pRequest->Request.DATA.SET_INFORMATION.InformationBufferLength = 0 ;
  1948. NdisInitializeEvent( &pRequest->Event );
  1949. NdisResetEvent( &pRequest->Event );
  1950. pRequest->pFunc = BrdgMiniLocalRequestComplete;
  1951. pRequest->FuncArg = Adapters[i];
  1952. // Fire it off
  1953. NdisRequest( &Status, Adapters[i]->BindingHandle, &pRequest->Request );
  1954. if( Status != NDIS_STATUS_PENDING )
  1955. {
  1956. // The cleanup function won't get called
  1957. BrdgMiniLocalRequestComplete( pRequest, Adapters[i] );
  1958. }
  1959. } // end of for loop
  1960. return;
  1961. }