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.

1994 lines
64 KiB

  1. /*
  2. (C) Copyright 1999
  3. All rights reserved.
  4. Portions of this software are:
  5. (C) Copyright 1995 TriplePoint, Inc. -- http://www.TriplePoint.com
  6. License to use this software is granted under the same terms
  7. outlined in the Microsoft Windows Device Driver Development Kit.
  8. (C) Copyright 1992 Microsoft Corp. -- http://www.Microsoft.com
  9. License to use this software is granted under the terms outlined in
  10. the Microsoft Windows Device Driver Development Kit.
  11. @doc INTERNAL Card Card_c
  12. @module Card.c |
  13. This module implements the interface to the <t CARD_OBJECT>.
  14. Supports the low-level hardware control functions used by the NDIS WAN
  15. Minport driver.
  16. @comm
  17. This module isolates most the vendor specific hardware access interfaces.
  18. It will require signficant changes to accomodate your hardware device.
  19. You should try to isolate your changes to the <t CARD_OBJECT> rather then
  20. the <t MINIPORT_ADAPTER_OBJECT>. This will make it eaiser to reuse the
  21. upper portions of the driver should your hardware change in the future.
  22. @head3 Contents |
  23. @index class,mfunc,func,msg,mdata,struct,enum | Card_c
  24. @end
  25. */
  26. #define __FILEID__ CARD_OBJECT_TYPE
  27. // Unique file ID for error logging
  28. #include "Miniport.h" // Defines all the miniport objects
  29. #if defined(NDIS_LCODE)
  30. # pragma NDIS_LCODE // Windows 9x wants this code locked down!
  31. # pragma NDIS_LDATA
  32. #endif
  33. DBG_STATIC ULONG g_CardInstanceCounter // @globalv
  34. // Keeps track of how many <t CARD_OBJECT>s are created.
  35. = 0;
  36. /* @doc EXTERNAL INTERNAL Card Card_c g_CardParameters
  37. @topic 5.2 Card Parameters |
  38. This section describes the registry parameters read into the
  39. <t CARD_OBJECT>.
  40. @globalv PARAM_TABLE | g_CardParameters |
  41. This table defines the registry based parameters to be assigned to data
  42. members of the <t CARD_OBJECT>.
  43. <f Note>:
  44. If you add any registry based data members to <t CARD_OBJECT>
  45. you will need to modify <f CardReadParameters> and add the parameter
  46. definitions to the <f g_CardParameters> table.
  47. @flag <f BufferSize> (OPTIONAL) |
  48. This DWORD parameter allows you to control the maximum buffer size used
  49. to transmit and receive packets over the IDSN line. Typically, this is
  50. defined to be 1500 bytes for most Point to Point (PPP) connections.<nl>
  51. <tab><f Default Value:><tab><tab>1532<nl>
  52. <tab><f Valid Range N:><tab><tab>532 <lt>= N <lt>= 4032<nl>
  53. <f Note>: You must add 32 bytes to the maximum packet size you
  54. expect to send or receive. Therefore, if you have a maximum packet size
  55. of 1500 bytes, excluding media headers, you should set the <f BufferSize>
  56. value to 1532.<nl>
  57. @flag <f ReceiveBuffersPerLink> (OPTIONAL) |
  58. This DWORD parameter allows you to control the maximum number of incoming
  59. packets that can in progress at any one time. The Miniport will allocate
  60. this number of packets per BChannel and set them up for incoming packets.
  61. Typically, three or four should be sufficient to handle a few short bursts
  62. that may occur with small packets. If the Miniport is not able to service
  63. the incoming packets fast enough, new packets will be dropped and it is up
  64. to the NDIS WAN Wrapper to resynchronize with the remote station.<nl>
  65. <tab><f Default Value:><tab><tab>3<nl>
  66. <tab><f Valid Range N:><tab><tab>2 <lt>= N <lt>= 16<nl>
  67. @flag <f TransmitBuffersPerLink> (OPTIONAL) |
  68. This DWORD parameter allows you to control the maximum number of outgoing
  69. packets that can in progress at any one time. The Miniport will allow
  70. this number of packets per BChannel to be outstanding (i.e. in progress).
  71. Typically, two or three should be sufficient to keep the channel busy for
  72. normal sized packets. If there are alot of small packets being sent, the
  73. BChannel may become idle for brief periods while new packets are being
  74. queued. Windows does not normally work this way if it has large amounts
  75. of data to transfer, so the default value should be sufficient. <nl>
  76. <tab><f Default Value:><tab><tab>2<nl>
  77. <tab><f Valid Range N:><tab><tab>1 <lt>= N <lt>= 16<nl>
  78. @flag <f IsdnNumDChannels> (OPTIONAL) |
  79. This DWORD parameter allows you to control the number of ISDN D Channels
  80. allocated for the adapter. The driver assumes only one logical
  81. <t DCHANNEL_OBJECT>, but the card can have multiple physical D channels
  82. managed by the <t PORT_OBJECT>.
  83. <tab><f Default Value:><tab><tab>1<nl>
  84. <tab><f Valid Range N:><tab><tab>1 <lt>= N <lt>= 16<nl>
  85. */
  86. DBG_STATIC PARAM_TABLE g_CardParameters[] =
  87. {
  88. #if defined(PCI_BUS)
  89. PARAM_ENTRY(CARD_OBJECT,
  90. PciSlotNumber, PARAM_PciSlotNumber,
  91. TRUE, NdisParameterInteger, 0,
  92. 0, 0, 31),
  93. #endif // PCI_BUS
  94. PARAM_ENTRY(CARD_OBJECT,
  95. BufferSize, PARAM_BufferSize,
  96. FALSE, NdisParameterInteger, 0,
  97. CARD_DEFAULT_PACKET_SIZE, CARD_MIN_PACKET_SIZE, CARD_MAX_PACKET_SIZE),
  98. PARAM_ENTRY(CARD_OBJECT,
  99. ReceiveBuffersPerLink, PARAM_ReceiveBuffersPerLink,
  100. FALSE, NdisParameterInteger, 0,
  101. 2, 2, 16),
  102. PARAM_ENTRY(CARD_OBJECT,
  103. TransmitBuffersPerLink, PARAM_TransmitBuffersPerLink,
  104. FALSE, NdisParameterInteger, 0,
  105. 2, 1, 16),
  106. PARAM_ENTRY(CARD_OBJECT,
  107. NumDChannels, PARAM_NumDChannels,
  108. FALSE, NdisParameterInteger, 0,
  109. 1, 1, 4),
  110. /* The last entry must be an empty string! */
  111. { { 0 } }
  112. };
  113. /* @doc INTERNAL Card Card_c CardReadParameters
  114. @func
  115. <f CardReadParameters> reads the Card parameters from the registry
  116. and initializes the associated data members. This should only be called
  117. by <f CardCreate>.
  118. <f Note>:
  119. If you add any registry based data members to <t CARD_OBJECT>
  120. you will need to modify <f CardReadParameters> and add the parameter
  121. definitions to the <f g_CardParameters> table.
  122. @rdesc
  123. <f CardReadParameters> returns zero if it is successful.<nl>
  124. Otherwise, a non-zero return value indicates an error condition.
  125. */
  126. DBG_STATIC NDIS_STATUS CardReadParameters(
  127. IN PCARD_OBJECT pCard // @parm
  128. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  129. )
  130. {
  131. DBG_FUNC("CardReadParameters")
  132. NDIS_STATUS Status;
  133. // Status result returned from an NDIS function call.
  134. PMINIPORT_ADAPTER_OBJECT pAdapter;
  135. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  136. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  137. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  138. DBG_ENTER(pAdapter);
  139. /*
  140. // Parse the registry parameters.
  141. */
  142. Status = ParamParseRegistry(
  143. pAdapter->MiniportAdapterHandle,
  144. pAdapter->WrapperConfigurationContext,
  145. (PUCHAR)pCard,
  146. g_CardParameters
  147. );
  148. if (Status == NDIS_STATUS_SUCCESS)
  149. {
  150. /*
  151. // Make sure the parameters are valid.
  152. */
  153. if (pCard->BufferSize & 0x1F)
  154. {
  155. DBG_ERROR(pAdapter,("Invalid value 'BufferSize'=0x0x%X must be multiple of 32\n",
  156. pCard->BufferSize));
  157. NdisWriteErrorLogEntry(
  158. pAdapter->MiniportAdapterHandle,
  159. NDIS_ERROR_CODE_UNSUPPORTED_CONFIGURATION,
  160. 3,
  161. pCard->BufferSize,
  162. __FILEID__,
  163. __LINE__
  164. );
  165. Status = NDIS_STATUS_FAILURE;
  166. }
  167. else
  168. {
  169. /*
  170. // Finish setting up data members based on registry settings.
  171. */
  172. }
  173. }
  174. DBG_RETURN(pAdapter, Status);
  175. return (Status);
  176. }
  177. /* @doc INTERNAL Card Card_c CardFindNIC
  178. @func
  179. <f CardFindNIC> locates the NIC associated with this NDIS device.
  180. @rdesc
  181. <f CardFindNIC> returns zero if it is successful.<nl>
  182. Otherwise, a non-zero return value indicates an error condition.
  183. */
  184. DBG_STATIC NDIS_STATUS CardFindNIC(
  185. IN PCARD_OBJECT pCard // @parm
  186. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  187. )
  188. {
  189. DBG_FUNC("CardFindNIC")
  190. NDIS_STATUS Result = NDIS_STATUS_SUCCESS;
  191. // Holds the result code returned by this function.
  192. #if defined(PCI_BUS)
  193. ULONG Index;
  194. // Loop counter.
  195. PNDIS_RESOURCE_LIST pPciResourceList;
  196. PCM_PARTIAL_RESOURCE_DESCRIPTOR pPciResource;
  197. #endif // PCI_BUS
  198. PMINIPORT_ADAPTER_OBJECT pAdapter;
  199. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  200. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  201. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  202. DBG_ENTER(pAdapter);
  203. #if defined(PCI_BUS)
  204. /*
  205. // Read the PCI data and initialize the driver data structure
  206. // with the data returned.
  207. */
  208. pPciResourceList = NULL;
  209. Result = NdisMPciAssignResources(pAdapter->MiniportAdapterHandle,
  210. pCard->PciSlotNumber,
  211. &pPciResourceList);
  212. if (Result != NDIS_STATUS_SUCCESS)
  213. {
  214. DBG_ERROR(pAdapter,("NdisMPciAssignResources Result=0x%X\n",
  215. Result));
  216. NdisWriteErrorLogEntry(
  217. pAdapter->MiniportAdapterHandle,
  218. NDIS_ERROR_CODE_INVALID_VALUE_FROM_ADAPTER,
  219. 4,
  220. pCard->PciSlotNumber,
  221. Result,
  222. __FILEID__,
  223. __LINE__
  224. );
  225. }
  226. for (Index = 0; Result == NDIS_STATUS_SUCCESS &&
  227. Index < pPciResourceList->Count; ++Index)
  228. {
  229. ASSERT(pPciResourceList);
  230. pPciResource = &pPciResourceList->PartialDescriptors[Index];
  231. ASSERT(pPciResource);
  232. switch (pPciResource->Type)
  233. {
  234. case CmResourceTypePort:
  235. #if defined(CARD_MIN_IOPORT_SIZE)
  236. if (pPciResource->u.Port.Start.LowPart &&
  237. pPciResource->u.Port.Length >= CARD_MIN_IOPORT_SIZE)
  238. {
  239. DBG_NOTICE(pAdapter,("Port: Ptr=0x%X Len=%d<%d\n",
  240. pPciResource->u.Port.Start.LowPart,
  241. pPciResource->u.Port.Length,
  242. CARD_MIN_IOPORT_SIZE));
  243. pCard->ResourceInformation.IoPortPhysicalAddress =
  244. pPciResource->u.Port.Start;
  245. pCard->ResourceInformation.IoPortLength =
  246. pPciResource->u.Port.Length;
  247. }
  248. else
  249. {
  250. DBG_ERROR(pAdapter,("Invalid Port: Ptr=0x%X Len=%d<%d\n",
  251. pPciResource->u.Port.Start,
  252. pPciResource->u.Port.Length,
  253. CARD_MIN_IOPORT_SIZE));
  254. NdisWriteErrorLogEntry(
  255. pAdapter->MiniportAdapterHandle,
  256. NDIS_ERROR_CODE_INVALID_VALUE_FROM_ADAPTER,
  257. 4,
  258. pPciResource->u.Port.Length,
  259. CARD_MIN_IOPORT_SIZE,
  260. __FILEID__,
  261. __LINE__
  262. );
  263. Result = NDIS_STATUS_RESOURCE_CONFLICT;
  264. }
  265. #endif // CARD_MIN_IOPORT_SIZE
  266. break;
  267. case CmResourceTypeInterrupt:
  268. #if defined(CARD_REQUEST_ISR)
  269. if (pPciResource->u.Interrupt.Level)
  270. {
  271. DBG_NOTICE(pAdapter,("Interrupt: Lev=%d,Vec=%d\n",
  272. pPciResource->u.Interrupt.Level,
  273. pPciResource->u.Interrupt.Vector));
  274. pCard->ResourceInformation.InterruptLevel =
  275. pPciResource->u.Interrupt.Level;
  276. pCard->ResourceInformation.InterruptVector =
  277. pPciResource->u.Interrupt.Vector;
  278. pCard->ResourceInformation.InterruptShared = CARD_INTERRUPT_SHARED;
  279. pCard->ResourceInformation.InterruptMode = CARD_INTERRUPT_MODE;
  280. }
  281. else
  282. {
  283. DBG_ERROR(pAdapter,("Invalid Interrupt: Lev=%d,Vec=%d\n",
  284. pPciResource->u.Interrupt.Level,
  285. pPciResource->u.Interrupt.Vector));
  286. NdisWriteErrorLogEntry(
  287. pAdapter->MiniportAdapterHandle,
  288. NDIS_ERROR_CODE_INVALID_VALUE_FROM_ADAPTER,
  289. 4,
  290. pPciResource->u.Interrupt.Level,
  291. pPciResource->u.Interrupt.Vector,
  292. __FILEID__,
  293. __LINE__
  294. );
  295. Result = NDIS_STATUS_RESOURCE_CONFLICT;
  296. }
  297. #endif // defined(CARD_REQUEST_ISR)
  298. break;
  299. case CmResourceTypeMemory:
  300. #if defined(CARD_MIN_MEMORY_SIZE)
  301. if (pPciResource->u.Memory.Start.LowPart &&
  302. pPciResource->u.Memory.Length >= CARD_MIN_MEMORY_SIZE)
  303. {
  304. DBG_NOTICE(pAdapter,("Memory: Ptr=0x%X Len=%d<%d\n",
  305. pPciResource->u.Memory.Start.LowPart,
  306. pPciResource->u.Memory.Length,
  307. CARD_MIN_MEMORY_SIZE));
  308. pCard->ResourceInformation.MemoryPhysicalAddress =
  309. pPciResource->u.Memory.Start;
  310. pCard->ResourceInformation.MemoryLength =
  311. pPciResource->u.Memory.Length;
  312. }
  313. else
  314. {
  315. DBG_ERROR(pAdapter,("Invalid Memory: Ptr=0x%X Len=%d<%d\n",
  316. pPciResource->u.Memory.Start.LowPart,
  317. pPciResource->u.Memory.Length,
  318. CARD_MIN_MEMORY_SIZE));
  319. NdisWriteErrorLogEntry(
  320. pAdapter->MiniportAdapterHandle,
  321. NDIS_ERROR_CODE_INVALID_VALUE_FROM_ADAPTER,
  322. 4,
  323. pPciResource->u.Memory.Length,
  324. CARD_MIN_MEMORY_SIZE,
  325. __FILEID__,
  326. __LINE__
  327. );
  328. Result = NDIS_STATUS_RESOURCE_CONFLICT;
  329. }
  330. break;
  331. #endif // CARD_MIN_MEMORY_SIZE
  332. default:
  333. DBG_ERROR(pAdapter,("Unknown resource type=%d\n",
  334. pPciResource->Type));
  335. break;
  336. }
  337. }
  338. pCard->ResourceInformation.BusInterfaceType = NdisInterfacePci;
  339. #endif // PCI_BUS
  340. pCard->ResourceInformation.Master = CARD_IS_BUS_MASTER;
  341. #if (CARD_IS_BUS_MASTER)
  342. pCard->ResourceInformation.DmaChannel = 0;
  343. pCard->ResourceInformation.Dma32BitAddresses = TRUE,
  344. pCard->ResourceInformation.MaximumPhysicalMapping = pCard->BufferSize;
  345. pCard->ResourceInformation.PhysicalMapRegistersNeeded = CARD_MAP_REGISTERS_NEEDED;
  346. #endif // (CARD_IS_BUS_MASTER)
  347. DBG_RETURN(pAdapter, Result);
  348. return (Result);
  349. }
  350. /* @doc INTERNAL Card Card_c CardCreateInterface
  351. @func
  352. <f CardCreateInterface> allocates a shared memory pool and uses it to
  353. establish the message interface between the Miniport and the NIC.
  354. @rdesc
  355. <f CardCreateInterface> returns zero if it is successful.<nl>
  356. Otherwise, a non-zero return value indicates an error condition.
  357. */
  358. DBG_STATIC NDIS_STATUS CardCreateInterface(
  359. IN PCARD_OBJECT pCard // @parm
  360. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  361. )
  362. {
  363. DBG_FUNC("CardCreateObjects")
  364. NDIS_STATUS Result = NDIS_STATUS_SUCCESS;
  365. // Holds the result code returned by this function.
  366. PMINIPORT_ADAPTER_OBJECT pAdapter;
  367. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  368. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  369. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  370. DBG_ENTER(pAdapter);
  371. DBG_RETURN(pAdapter, Result);
  372. return (Result);
  373. }
  374. /* @doc INTERNAL Card Card_c CardCreateObjects
  375. @func
  376. <f CardCreateObjects> calls the create routines for all the objects
  377. contained in <t CARD_OBJECT>. This should only be called
  378. by <f CardCreate>.
  379. <f Note>:
  380. If you add any new objects to <t CARD_OBJECT> you will need
  381. to modify <f CardCreateObjects> and <f CardDestroyObjects> so they
  382. will get created and destroyed properly.
  383. @rdesc
  384. <f CardCreateObjects> returns zero if it is successful.<nl>
  385. Otherwise, a non-zero return value indicates an error condition.
  386. */
  387. DBG_STATIC NDIS_STATUS CardCreateObjects(
  388. IN PCARD_OBJECT pCard // @parm
  389. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  390. )
  391. {
  392. DBG_FUNC("CardCreateObjects")
  393. NDIS_STATUS Result = NDIS_STATUS_SUCCESS;
  394. // Holds the result code returned by this function.
  395. ULONG Index;
  396. // Loop counter.
  397. ULONG NumPorts;
  398. // The number of Ports supported by the NIC.
  399. PMINIPORT_ADAPTER_OBJECT pAdapter;
  400. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  401. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  402. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  403. DBG_ENTER(pAdapter);
  404. /*
  405. // Try to locate the NIC on the PCI bus.
  406. */
  407. Result = CardFindNIC(pCard);
  408. if (Result != NDIS_STATUS_SUCCESS)
  409. {
  410. goto ExceptionExit;
  411. }
  412. /*
  413. // Create the message interface to the NIC.
  414. */
  415. Result = CardCreateInterface(pCard);
  416. if (Result != NDIS_STATUS_SUCCESS)
  417. {
  418. goto ExceptionExit;
  419. }
  420. /*
  421. // Create the Port objects.
  422. */
  423. NumPorts = CardNumPorts(pCard);
  424. Result = ALLOCATE_MEMORY(pCard->pPortArray,
  425. sizeof(PVOID) * NumPorts,
  426. pAdapter->MiniportAdapterHandle);
  427. for (Index = 0; Result == NDIS_STATUS_SUCCESS &&
  428. Index < NumPorts; Index++)
  429. {
  430. Result = PortCreate(&pCard->pPortArray[Index], pCard);
  431. /*
  432. // Keep track of how many are created.
  433. */
  434. if (Result == NDIS_STATUS_SUCCESS)
  435. {
  436. pCard->NumPorts++;
  437. }
  438. }
  439. /*
  440. // We allocate (ReceiveBuffersPerLink * NumBChannels) buffers
  441. // to be used to receive incoming messages from the card.
  442. */
  443. pCard->NumMessageBuffers = (CardNumChannels(pCard) *
  444. pCard->ReceiveBuffersPerLink);
  445. Result = ALLOCATE_MEMORY(pCard->MessagesVirtualAddress,
  446. pCard->NumMessageBuffers * pCard->BufferSize,
  447. pAdapter->MiniportAdapterHandle);
  448. if (Result == NDIS_STATUS_SUCCESS)
  449. {
  450. PUCHAR MessageBuffer = pCard->MessagesVirtualAddress;
  451. ASSERT(pCard->MessagesVirtualAddress);
  452. /*
  453. // Allocate the buffer list spin lock to use as a MUTEX.
  454. */
  455. NdisAllocateSpinLock(&pCard->MessageBufferLock);
  456. InitializeListHead(&pCard->MessageBufferList);
  457. for (Index = 0; Index < pCard->NumMessageBuffers; Index++)
  458. {
  459. InsertTailList(&pCard->MessageBufferList,
  460. (PLIST_ENTRY)MessageBuffer);
  461. MessageBuffer += pCard->BufferSize;
  462. }
  463. }
  464. /*
  465. // Allocate the message buffer pool.
  466. */
  467. if (Result == NDIS_STATUS_SUCCESS)
  468. {
  469. NdisAllocateBufferPool(&Result,
  470. &pCard->BufferPoolHandle,
  471. pCard->NumMessageBuffers
  472. );
  473. if (Result != NDIS_STATUS_SUCCESS)
  474. {
  475. pCard->BufferPoolHandle = NULL_BUFFER_POOL;
  476. DBG_ERROR(pAdapter,("NdisAllocateBufferPool: Result=0x%X\n",
  477. Result));
  478. NdisWriteErrorLogEntry(
  479. pCard->pAdapter->MiniportAdapterHandle,
  480. NDIS_ERROR_CODE_OUT_OF_RESOURCES,
  481. 3,
  482. Result,
  483. __FILEID__,
  484. __LINE__
  485. );
  486. }
  487. else
  488. {
  489. ASSERT(pCard->BufferPoolHandle != NULL_BUFFER_POOL);
  490. DBG_FILTER(pAdapter, DBG_BUFFER_ON,
  491. ("BufferPoolSize=%d\n",
  492. pCard->NumMessageBuffers
  493. ));
  494. }
  495. }
  496. /*
  497. // Allocate the message packet pool.
  498. */
  499. if (Result == NDIS_STATUS_SUCCESS)
  500. {
  501. NdisAllocatePacketPool(&Result,
  502. &pCard->PacketPoolHandle,
  503. pCard->NumMessageBuffers,
  504. 0
  505. );
  506. if (Result != NDIS_STATUS_SUCCESS)
  507. {
  508. DBG_ERROR(pAdapter,("NdisAllocatePacketPool: Result=0x%X\n",
  509. Result));
  510. NdisWriteErrorLogEntry(
  511. pCard->pAdapter->MiniportAdapterHandle,
  512. NDIS_ERROR_CODE_OUT_OF_RESOURCES,
  513. 3,
  514. Result,
  515. __FILEID__,
  516. __LINE__
  517. );
  518. }
  519. else
  520. {
  521. ASSERT(pCard->PacketPoolHandle);
  522. DBG_FILTER(pAdapter, DBG_PACKET_ON,
  523. ("PacketPoolSize=%d\n",
  524. pCard->NumMessageBuffers
  525. ));
  526. }
  527. }
  528. ExceptionExit:
  529. DBG_RETURN(pAdapter, Result);
  530. return (Result);
  531. }
  532. /* @doc INTERNAL Card Card_c CardCreate
  533. @func
  534. <f CardCreate> allocates memory for a <t CARD_OBJECT> and then
  535. initializes the data members to their starting state.
  536. If successful, <p ppCard> will be set to point to the newly created
  537. <t CARD_OBJECT>. Otherwise, <p ppCard> will be set to NULL.
  538. @comm
  539. This function should be called only once when the Miniport is loaded.
  540. Before the Miniport is unloaded, <f CardDestroy> must be called to
  541. release the <t CARD_OBJECT> created by this function.
  542. @rdesc
  543. <f CardCreate> returns zero if it is successful.<nl>
  544. Otherwise, a non-zero return value indicates an error condition.
  545. */
  546. NDIS_STATUS CardCreate(
  547. OUT PCARD_OBJECT * ppCard, // @parm
  548. // Points to a caller-defined memory location to which this function
  549. // writes the virtual address of the allocated <t CARD_OBJECT>.
  550. IN PMINIPORT_ADAPTER_OBJECT pAdapter // @parm
  551. // A pointer to the <t MINIPORT_ADAPTER_OBJECT> instance.
  552. )
  553. {
  554. DBG_FUNC("CardCreate")
  555. PCARD_OBJECT pCard;
  556. // Pointer to our newly allocated object.
  557. NDIS_STATUS Result = NDIS_STATUS_SUCCESS;
  558. // Holds the result code returned by this function.
  559. ASSERT(pAdapter && pAdapter->ObjectType == MINIPORT_ADAPTER_OBJECT_TYPE);
  560. DBG_ENTER(pAdapter);
  561. /*
  562. // Make sure the caller's object pointer is NULL to begin with.
  563. // It will be set later only if everything is successful.
  564. */
  565. *ppCard = NULL;
  566. /*
  567. // Allocate memory for the object.
  568. */
  569. Result = ALLOCATE_OBJECT(pCard, pAdapter->MiniportAdapterHandle);
  570. if (Result == NDIS_STATUS_SUCCESS)
  571. {
  572. /*
  573. // Zero everything to begin with.
  574. // Then set the object type and assign a unique ID .
  575. */
  576. pCard->ObjectType = CARD_OBJECT_TYPE;
  577. pCard->ObjectID = ++g_CardInstanceCounter;
  578. /*
  579. // Initialize the member variables to their default settings.
  580. */
  581. pCard->pAdapter = pAdapter;
  582. // TODO - Add code here
  583. /*
  584. // Parse the registry parameters.
  585. */
  586. Result = CardReadParameters(pCard);
  587. /*
  588. // If all goes well, we are ready to create the sub-components.
  589. */
  590. if (Result == NDIS_STATUS_SUCCESS)
  591. {
  592. Result = CardCreateObjects(pCard);
  593. }
  594. if (Result == NDIS_STATUS_SUCCESS)
  595. {
  596. /*
  597. // All is well, so return the object pointer to the caller.
  598. */
  599. *ppCard = pCard;
  600. }
  601. else
  602. {
  603. /*
  604. // Something went wrong, so let's make sure everything is
  605. // cleaned up.
  606. */
  607. CardDestroy(pCard);
  608. }
  609. }
  610. DBG_RETURN(pAdapter, Result);
  611. return (Result);
  612. }
  613. /* @doc INTERNAL Card Card_c CardDestroyObjects
  614. @func
  615. <f CardDestroyObjects> calls the destroy routines for all the objects
  616. contained in <t CARD_OBJECT>. This should only be called by
  617. <f CardDestroy>.
  618. <f Note>:
  619. If you add any new objects to <t PCARD_OBJECT> you will need to
  620. modify <f CardCreateObjects> and <f CardDestroyObjects> so they
  621. will get created and destroyed properly.
  622. */
  623. DBG_STATIC void CardDestroyObjects(
  624. IN PCARD_OBJECT pCard // @parm
  625. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  626. )
  627. {
  628. DBG_FUNC("CardDestroyObjects")
  629. ULONG NumPorts;
  630. // The number of Ports supported by the NIC.
  631. PMINIPORT_ADAPTER_OBJECT pAdapter;
  632. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  633. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  634. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  635. DBG_ENTER(pAdapter);
  636. // TODO - Add code here
  637. /*
  638. // Release the packet, buffer, and message memory back to NDIS.
  639. */
  640. if (pCard->PacketPoolHandle)
  641. {
  642. NdisFreePacketPool(pCard->PacketPoolHandle);
  643. }
  644. if (pCard->BufferPoolHandle != NULL_BUFFER_POOL)
  645. {
  646. NdisFreeBufferPool(pCard->BufferPoolHandle);
  647. }
  648. if (pCard->MessagesVirtualAddress)
  649. {
  650. FREE_MEMORY(pCard->MessagesVirtualAddress,
  651. pCard->NumMessageBuffers * pCard->BufferSize);
  652. }
  653. if (pCard->MessageBufferLock.SpinLock)
  654. {
  655. NdisFreeSpinLock(&pCard->MessageBufferLock);
  656. }
  657. /*
  658. // Destory the Port objects.
  659. */
  660. NumPorts = pCard->NumPorts;
  661. while (NumPorts--)
  662. {
  663. PortDestroy(pCard->pPortArray[NumPorts]);
  664. }
  665. pCard->NumPorts = 0;
  666. /*
  667. // Free space for the Ports.
  668. */
  669. if (pCard->pPortArray)
  670. {
  671. NumPorts = CardNumPorts(pCard);
  672. FREE_MEMORY(pCard->pPortArray, sizeof(PVOID) * NumPorts);
  673. }
  674. /*
  675. // Release the system resources back to NDIS.
  676. */
  677. #if defined(CARD_REQUEST_ISR)
  678. if (pCard->Interrupt.InterruptObject)
  679. {
  680. NdisMDeregisterInterrupt(&pCard->Interrupt);
  681. pCard->Interrupt.InterruptObject = NULL;
  682. }
  683. #endif // defined(CARD_REQUEST_ISR)
  684. #if defined(CARD_MIN_IOPORT_SIZE)
  685. if (pCard->pIoPortVirtualAddress)
  686. {
  687. NdisMDeregisterIoPortRange(
  688. pAdapter->MiniportAdapterHandle,
  689. pCard->ResourceInformation.IoPortPhysicalAddress.LowPart,
  690. pCard->ResourceInformation.IoPortLength,
  691. pCard->pIoPortVirtualAddress);
  692. pCard->pIoPortVirtualAddress = NULL;
  693. }
  694. #endif // CARD_MIN_IOPORT_SIZE
  695. #if defined(CARD_MIN_MEMORY_SIZE)
  696. if (pCard->pMemoryVirtualAddress)
  697. {
  698. NdisMUnmapIoSpace(
  699. pAdapter->MiniportAdapterHandle,
  700. pCard->pMemoryVirtualAddress,
  701. pCard->ResourceInformation.MemoryLength
  702. );
  703. pCard->pMemoryVirtualAddress = NULL;
  704. }
  705. #endif // CARD_MIN_MEMORY_SIZE
  706. DBG_LEAVE(pAdapter);
  707. }
  708. /* @doc INTERNAL Card Card_c CardDestroy
  709. @func
  710. <f CardDestroy> frees the memory for this <t CARD_OBJECT>. All memory
  711. allocated by <f CardCreate> will be released back to the OS.
  712. */
  713. void CardDestroy(
  714. IN PCARD_OBJECT pCard // @parm
  715. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  716. )
  717. {
  718. DBG_FUNC("CardDestroy")
  719. PMINIPORT_ADAPTER_OBJECT pAdapter;
  720. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  721. if (pCard)
  722. {
  723. ASSERT(pCard->ObjectType == CARD_OBJECT_TYPE);
  724. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  725. DBG_ENTER(pAdapter);
  726. // TODO - Add code here
  727. /*
  728. // Release all objects allocated within this object.
  729. */
  730. CardDestroyObjects(pCard);
  731. /*
  732. // Make sure we fail the ASSERT if we see this object again.
  733. */
  734. pCard->ObjectType = 0;
  735. FREE_OBJECT(pCard);
  736. DBG_LEAVE(pAdapter);
  737. }
  738. }
  739. /* @doc INTERNAL Card Card_c CardNumPorts
  740. @func
  741. <f CardNumPorts> will return the total number of ports available on the
  742. NIC.
  743. @rdesc
  744. <f CardNumPorts> returns the total number of ports available.
  745. */
  746. ULONG CardNumPorts(
  747. IN PCARD_OBJECT pCard // @parm
  748. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  749. )
  750. {
  751. DBG_FUNC("CardNumPorts")
  752. // TODO - Get the actual number of ports from the card.
  753. return (pCard->NumDChannels);
  754. }
  755. /* @doc INTERNAL Card Card_c CardNumChannels
  756. @func
  757. <f CardNumChannels> will return the total number of channels capable
  758. of supporting data connections to a remote end-point.
  759. @rdesc
  760. <f CardNumChannels> returns the total number of data channels supported
  761. on all the NIC ports.
  762. */
  763. ULONG CardNumChannels(
  764. IN PCARD_OBJECT pCard // @parm
  765. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  766. )
  767. {
  768. DBG_FUNC("CardNumChannels")
  769. UINT PortIndex;
  770. // Loop index.
  771. if (pCard->NumChannels == 0)
  772. {
  773. // NumPorts should already be known.
  774. ASSERT(pCard->NumPorts);
  775. // Get the actual number of channels configured on all ports.
  776. for (PortIndex = 0; PortIndex < pCard->NumPorts; PortIndex++)
  777. {
  778. pCard->NumChannels += pCard->pPortArray[PortIndex]->NumChannels;
  779. }
  780. ASSERT(pCard->NumChannels);
  781. }
  782. return (pCard->NumChannels);
  783. }
  784. /* @doc INTERNAL Card Card_c CardInitialize
  785. @func
  786. <f CardInitialize> will attempt to initialize the NIC, but will not
  787. enable transmits or receives.
  788. @rdesc
  789. <f CardInitialize> returns zero if it is successful.<nl>
  790. Otherwise, a non-zero return value indicates an error condition.
  791. */
  792. NDIS_STATUS CardInitialize(
  793. IN PCARD_OBJECT pCard // @parm
  794. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  795. )
  796. {
  797. DBG_FUNC("CardInitialize")
  798. int num_dial_chan = 0;
  799. int num_sync_chan = 0;
  800. // The number of channels supported by card is based on InterfaceType.
  801. NDIS_STATUS Result = NDIS_STATUS_SUCCESS;
  802. // Holds the result code returned by this function.
  803. PMINIPORT_ADAPTER_OBJECT pAdapter;
  804. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  805. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  806. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  807. DBG_ENTER(pAdapter);
  808. /*
  809. // Inform the wrapper of the physical attributes of this adapter.
  810. // This must be called before any NdisMRegister functions!
  811. // This call also associates the MiniportAdapterHandle with this pAdapter.
  812. */
  813. NdisMSetAttributes(pAdapter->MiniportAdapterHandle,
  814. (NDIS_HANDLE) pAdapter,
  815. pCard->ResourceInformation.Master,
  816. pCard->ResourceInformation.BusInterfaceType
  817. );
  818. #if (CARD_IS_BUS_MASTER)
  819. if (pCard->ResourceInformation.Master)
  820. {
  821. ASSERT(pCard->ResourceInformation.DmaChannel == 0 ||
  822. pCard->ResourceInformation.BusInterfaceType == NdisInterfaceIsa);
  823. Result = NdisMAllocateMapRegisters(
  824. pAdapter->MiniportAdapterHandle,
  825. pCard->ResourceInformation.DmaChannel,
  826. pCard->ResourceInformation.Dma32BitAddresses,
  827. pCard->ResourceInformation.PhysicalMapRegistersNeeded + 1,
  828. pCard->ResourceInformation.MaximumPhysicalMapping
  829. );
  830. if (Result != NDIS_STATUS_SUCCESS)
  831. {
  832. DBG_ERROR(pAdapter,("NdisMAllocateMapRegisters(%d,%d) Result=0x%X\n",
  833. pCard->ResourceInformation.PhysicalMapRegistersNeeded,
  834. pCard->ResourceInformation.MaximumPhysicalMapping,
  835. Result));
  836. NdisWriteErrorLogEntry(
  837. pAdapter->MiniportAdapterHandle,
  838. NDIS_ERROR_CODE_RESOURCE_CONFLICT,
  839. 5,
  840. pCard->ResourceInformation.PhysicalMapRegistersNeeded,
  841. pCard->ResourceInformation.MaximumPhysicalMapping,
  842. Result,
  843. __FILEID__,
  844. __LINE__
  845. );
  846. }
  847. }
  848. #endif // (CARD_IS_BUS_MASTER)
  849. #if defined(CARD_MIN_MEMORY_SIZE)
  850. if (Result == NDIS_STATUS_SUCCESS &&
  851. pCard->ResourceInformation.MemoryLength)
  852. {
  853. Result = NdisMMapIoSpace(
  854. &pCard->pMemoryVirtualAddress,
  855. pAdapter->MiniportAdapterHandle,
  856. pCard->ResourceInformation.MemoryPhysicalAddress,
  857. pCard->ResourceInformation.MemoryLength);
  858. if (Result != NDIS_STATUS_SUCCESS)
  859. {
  860. DBG_ERROR(pAdapter,("NdisMMapIoSpace(0x%X,0x%X) Result=0x%X\n",
  861. pCard->ResourceInformation.MemoryPhysicalAddress.LowPart,
  862. pCard->ResourceInformation.MemoryLength,
  863. Result));
  864. NdisWriteErrorLogEntry(
  865. pAdapter->MiniportAdapterHandle,
  866. NDIS_ERROR_CODE_RESOURCE_CONFLICT,
  867. 5,
  868. pCard->ResourceInformation.MemoryPhysicalAddress.LowPart,
  869. pCard->ResourceInformation.MemoryLength,
  870. Result,
  871. __FILEID__,
  872. __LINE__
  873. );
  874. }
  875. else
  876. {
  877. DBG_NOTICE(pAdapter,("NdisMMapIoSpace(0x%X,0x%X) VirtualAddress=0x%X\n",
  878. pCard->ResourceInformation.MemoryPhysicalAddress.LowPart,
  879. pCard->ResourceInformation.MemoryLength,
  880. pCard->pMemoryVirtualAddress));
  881. }
  882. }
  883. #endif // CARD_MIN_MEMORY_SIZE
  884. #if defined(CARD_MIN_IOPORT_SIZE)
  885. if (Result == NDIS_STATUS_SUCCESS &&
  886. pCard->ResourceInformation.IoPortLength)
  887. {
  888. Result = NdisMRegisterIoPortRange(
  889. &pCard->pIoPortVirtualAddress,
  890. pAdapter->MiniportAdapterHandle,
  891. pCard->ResourceInformation.IoPortPhysicalAddress.LowPart,
  892. pCard->ResourceInformation.IoPortLength);
  893. if (Result != NDIS_STATUS_SUCCESS)
  894. {
  895. DBG_ERROR(pAdapter,("NdisMRegisterIoPortRange(0x%X,0x%X) Result=0x%X\n",
  896. pCard->ResourceInformation.IoPortPhysicalAddress.LowPart,
  897. pCard->ResourceInformation.IoPortLength,
  898. Result));
  899. NdisWriteErrorLogEntry(
  900. pAdapter->MiniportAdapterHandle,
  901. NDIS_ERROR_CODE_RESOURCE_CONFLICT,
  902. 5,
  903. pCard->ResourceInformation.IoPortPhysicalAddress.LowPart,
  904. pCard->ResourceInformation.IoPortLength,
  905. Result,
  906. __FILEID__,
  907. __LINE__
  908. );
  909. }
  910. else
  911. {
  912. DBG_NOTICE(pAdapter,("NdisMRegisterIoPortRange(0x%X,0x%X) VirtualAddress=0x%X\n",
  913. pCard->ResourceInformation.IoPortPhysicalAddress.LowPart,
  914. pCard->ResourceInformation.IoPortLength,
  915. pCard->pIoPortVirtualAddress));
  916. }
  917. }
  918. #endif // CARD_MIN_IOPORT_SIZE
  919. #if defined(CARD_REQUEST_ISR)
  920. if (Result == NDIS_STATUS_SUCCESS &&
  921. pCard->ResourceInformation.InterruptVector)
  922. {
  923. ASSERT(pCard->ResourceInformation.InterruptShared == FALSE ||
  924. (pCard->ResourceInformation.InterruptMode == NdisInterruptLevelSensitive &&
  925. CARD_REQUEST_ISR == TRUE));
  926. Result = NdisMRegisterInterrupt(
  927. &pCard->Interrupt,
  928. pAdapter->MiniportAdapterHandle,
  929. pCard->ResourceInformation.InterruptVector,
  930. pCard->ResourceInformation.InterruptLevel,
  931. CARD_REQUEST_ISR,
  932. pCard->ResourceInformation.InterruptShared,
  933. pCard->ResourceInformation.InterruptMode
  934. );
  935. if (Result != NDIS_STATUS_SUCCESS)
  936. {
  937. DBG_ERROR(pAdapter,("NdisMRegisterInterrupt failed: Vec=%d, Lev=%d\n",
  938. (UINT)pCard->ResourceInformation.InterruptVector,
  939. (UINT)pCard->ResourceInformation.InterruptLevel));
  940. NdisWriteErrorLogEntry(
  941. pAdapter->MiniportAdapterHandle,
  942. NDIS_ERROR_CODE_RESOURCE_CONFLICT,
  943. 5,
  944. pCard->ResourceInformation.InterruptVector,
  945. pCard->ResourceInformation.InterruptLevel,
  946. Result,
  947. __FILEID__,
  948. __LINE__
  949. );
  950. }
  951. }
  952. #endif // defined(CARD_REQUEST_ISR)
  953. // TODO - Add your card initialization here.
  954. if (Result == NDIS_STATUS_SUCCESS)
  955. {
  956. }
  957. DBG_RETURN(pAdapter, Result);
  958. return (Result);
  959. }
  960. /* @doc INTERNAL Card Card_c CardTransmitPacket
  961. @func
  962. <f CardTransmitPacket> will start sending the current packet out.
  963. @rdesc
  964. <f CardTransmitPacket> returns TRUE if the packet is being transmitted,
  965. otherwise FALSE is returned.
  966. */
  967. BOOLEAN CardTransmitPacket(
  968. IN PCARD_OBJECT pCard, // @parm
  969. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  970. IN PBCHANNEL_OBJECT pBChannel, // @parm
  971. // A pointer to the <t BCHANNEL_OBJECT> returned by <f BChannelCreate>.
  972. IN PNDIS_PACKET pNdisPacket // @parm
  973. // A pointer to the associated NDIS packet structure <t NDIS_PACKET>.
  974. )
  975. {
  976. DBG_FUNC("CardTransmitPacket")
  977. BOOLEAN bResult = FALSE;
  978. // Holds the result code returned by this function.
  979. PMINIPORT_ADAPTER_OBJECT pAdapter;
  980. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  981. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  982. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  983. DBG_ENTER(pAdapter);
  984. #if defined(SAMPLE_DRIVER)
  985. {
  986. PBCHANNEL_OBJECT pPeerBChannel;
  987. // A pointer to the peer <t BCHANNEL_OBJECT>.
  988. PCARD_EVENT_OBJECT pEvent;
  989. // A pointer to the <t CARD_EVENT_OBJECT> associated with this event.
  990. // If you can transmit the packet on pBChannel, do it now.
  991. pPeerBChannel = pBChannel->pPeerBChannel;
  992. if (pPeerBChannel)
  993. {
  994. pEvent = CardEventAllocate(pPeerBChannel->pAdapter->pCard);
  995. if (pEvent)
  996. {
  997. /*
  998. // Append the packet onto TransmitBusyList while it is being sent.
  999. // Then move it to the TransmitCompleteList in CardInterruptHandler
  1000. // after the card is done with it.
  1001. */
  1002. NdisAcquireSpinLock(&pAdapter->TransmitLock);
  1003. InsertTailList(&pBChannel->TransmitBusyList,
  1004. GET_QUEUE_FROM_PACKET(pNdisPacket));
  1005. NdisReleaseSpinLock(&pAdapter->TransmitLock);
  1006. pEvent->ulEventCode = CARD_EVENT_RECEIVE;
  1007. pEvent->pSendingObject = pBChannel;
  1008. pEvent->pReceivingObject = pPeerBChannel;
  1009. pEvent->pNdisPacket = pNdisPacket;
  1010. CardNotifyEvent(pPeerBChannel->pAdapter->pCard, pEvent);
  1011. bResult = TRUE;
  1012. }
  1013. }
  1014. else
  1015. {
  1016. DBG_ERROR(pAdapter,("pPeerBChannel == NULL\n"));
  1017. }
  1018. }
  1019. #else // SAMPLE_DRIVER
  1020. // TODO - Add code here to transmit the packet.
  1021. DBG_TX(pAdapter, pBChannel->ObjectID,
  1022. BytesToSend, pNdisPacket->CurrentBuffer);
  1023. #endif // SAMPLE_DRIVER
  1024. DBG_RETURN(pAdapter, bResult);
  1025. return (bResult);
  1026. }
  1027. /* @doc EXTERNAL Card Card_c TpiCopyFromPacketToBuffer
  1028. @func
  1029. <f TpiCopyFromPacketToBuffer> copies from an NDIS packet into a memory
  1030. buffer.
  1031. */
  1032. DBG_STATIC VOID TpiCopyFromPacketToBuffer(
  1033. IN PNDIS_PACKET Packet, // @parm
  1034. // The packet to copy from.
  1035. IN UINT Offset, // @parm
  1036. // The offset from which to start the copy.
  1037. IN UINT BytesToCopy, // @parm
  1038. // The number of bytes to copy from the packet.
  1039. IN PUCHAR Buffer, // @parm
  1040. // The destination of the copy.
  1041. OUT PUINT BytesCopied // @parm
  1042. // The number of bytes actually copied. Can be less then
  1043. // BytesToCopy if the packet is shorter than BytesToCopy.
  1044. )
  1045. {
  1046. UINT NdisBufferCount;
  1047. PNDIS_BUFFER CurrentBuffer;
  1048. PVOID VirtualAddress;
  1049. UINT CurrentLength;
  1050. UINT LocalBytesCopied = 0;
  1051. UINT AmountToMove;
  1052. *BytesCopied = 0;
  1053. if (!BytesToCopy)
  1054. {
  1055. return;
  1056. }
  1057. //
  1058. // Get the first buffer.
  1059. //
  1060. NdisQueryPacket(
  1061. Packet,
  1062. NULL,
  1063. &NdisBufferCount,
  1064. &CurrentBuffer,
  1065. NULL
  1066. );
  1067. //
  1068. // Could have a null packet.
  1069. //
  1070. if (!NdisBufferCount)
  1071. {
  1072. return;
  1073. }
  1074. NdisQueryBufferSafe(
  1075. CurrentBuffer,
  1076. &VirtualAddress,
  1077. &CurrentLength,
  1078. NormalPagePriority
  1079. );
  1080. while (LocalBytesCopied < BytesToCopy)
  1081. {
  1082. if (!CurrentLength)
  1083. {
  1084. NdisGetNextBuffer(
  1085. CurrentBuffer,
  1086. &CurrentBuffer
  1087. );
  1088. //
  1089. // We've reached the end of the packet. We return
  1090. // with what we've done so far. (Which must be shorter
  1091. // than requested.
  1092. //
  1093. if (!CurrentBuffer)
  1094. {
  1095. break;
  1096. }
  1097. NdisQueryBufferSafe(
  1098. CurrentBuffer,
  1099. &VirtualAddress,
  1100. &CurrentLength,
  1101. NormalPagePriority
  1102. );
  1103. continue;
  1104. }
  1105. //
  1106. // Try to get us up to the point to start the copy.
  1107. //
  1108. if (Offset)
  1109. {
  1110. if (Offset > CurrentLength)
  1111. {
  1112. //
  1113. // What we want isn't in this buffer.
  1114. //
  1115. Offset -= CurrentLength;
  1116. CurrentLength = 0;
  1117. continue;
  1118. }
  1119. else
  1120. {
  1121. VirtualAddress = (PCHAR)VirtualAddress + Offset;
  1122. CurrentLength -= Offset;
  1123. Offset = 0;
  1124. }
  1125. }
  1126. //
  1127. // Copy the data.
  1128. //
  1129. AmountToMove =
  1130. ((CurrentLength <= (BytesToCopy - LocalBytesCopied)) ?
  1131. (CurrentLength):(BytesToCopy - LocalBytesCopied));
  1132. NdisMoveMemory(Buffer,VirtualAddress,AmountToMove);
  1133. Buffer = (PUCHAR)Buffer + AmountToMove;
  1134. VirtualAddress = (PCHAR)VirtualAddress + AmountToMove;
  1135. LocalBytesCopied += AmountToMove;
  1136. CurrentLength -= AmountToMove;
  1137. }
  1138. *BytesCopied = LocalBytesCopied;
  1139. }
  1140. /* @doc INTERNAL Card Card_c CardInterruptHandler
  1141. @func
  1142. <f CardInterruptHandler> dequeues an event from the asynchronous event
  1143. callback queue <t CARD_EVENT_OBJECT>, and processes it according to
  1144. whether it is a BChannel event, Card event, or B-Advise event.
  1145. The associated callback routines are responsible for processing the
  1146. event.
  1147. @comm
  1148. <f NdisAcquireSpinLock> and <f NdisReleaseSpinLock> are used to provide
  1149. protection around the dequeueing code and keep it from being re-entered
  1150. as a result of another asynchronous callback event.
  1151. */
  1152. VOID CardInterruptHandler(
  1153. IN PCARD_OBJECT pCard // @parm
  1154. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  1155. )
  1156. {
  1157. DBG_FUNC("CardInterruptHandler")
  1158. PMINIPORT_ADAPTER_OBJECT pAdapter;
  1159. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  1160. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  1161. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  1162. // DBG_ENTER(pAdapter);
  1163. #if defined(SAMPLE_DRIVER)
  1164. {
  1165. NDIS_STATUS Status = NDIS_STATUS_SUCCESS;
  1166. PNDIS_BUFFER pDstNdisBuffer;
  1167. // A pointer to the NDIS buffer we use to indicate the receive.
  1168. PUCHAR pMemory;
  1169. // A pointer to a memory area we use to create a copy of the incoming
  1170. // packet.
  1171. ULONG ByteCount = 0;
  1172. ULONG BytesCopied = 0;
  1173. PLIST_ENTRY pList;
  1174. PCARD_EVENT_OBJECT pEvent;
  1175. PCARD_EVENT_OBJECT pNewEvent;
  1176. // A pointer to the <t CARD_EVENT_OBJECT> associated with this event.
  1177. PBCHANNEL_OBJECT pBChannel;
  1178. PBCHANNEL_OBJECT pPeerBChannel;
  1179. // A pointer to the <t BCHANNEL_OBJECT> returned by <f BChannelCreate>.
  1180. /*
  1181. // Clear out all packets in the receive buffer.
  1182. */
  1183. NdisDprAcquireSpinLock(&pAdapter->EventLock);
  1184. while (!IsListEmpty(&pAdapter->EventList))
  1185. {
  1186. pEvent = (PCARD_EVENT_OBJECT)RemoveHeadList(&pAdapter->EventList);
  1187. NdisDprReleaseSpinLock(&pAdapter->EventLock);
  1188. ASSERT(pEvent->pReceivingObject);
  1189. switch (pEvent->ulEventCode)
  1190. {
  1191. case CARD_EVENT_RING:
  1192. // The caller has already removed the BChannel from the available
  1193. // list, so we just pass it up to SetupIncomingCall so it can
  1194. // get the same one from ProtocolCoCreateVc.
  1195. pBChannel = pEvent->pReceivingObject;
  1196. ASSERT(pBChannel && pBChannel->ObjectType == BCHANNEL_OBJECT_TYPE);
  1197. pPeerBChannel = pEvent->pSendingObject;
  1198. ASSERT(pPeerBChannel && pPeerBChannel->ObjectType == BCHANNEL_OBJECT_TYPE);
  1199. pBChannel->pPeerBChannel = pPeerBChannel;
  1200. DBG_FILTER(pAdapter,DBG_TAPICALL_ON,
  1201. ("#%d CallState=0x%X CARD_EVENT_RING from #%d\n",
  1202. pBChannel->ObjectID, pBChannel->CallState,
  1203. (pBChannel->pPeerBChannel == NULL) ? -1 :
  1204. pBChannel->pPeerBChannel->ObjectID));
  1205. Status = SetupIncomingCall(pAdapter, &pBChannel);
  1206. if (Status == NDIS_STATUS_SUCCESS)
  1207. {
  1208. ASSERT(pBChannel == pEvent->pReceivingObject);
  1209. }
  1210. else if (Status != NDIS_STATUS_PENDING)
  1211. {
  1212. DChannelRejectCall(pAdapter->pDChannel, pBChannel);
  1213. }
  1214. else
  1215. {
  1216. ASSERT(pBChannel == pEvent->pReceivingObject);
  1217. }
  1218. break;
  1219. case CARD_EVENT_CONNECT:
  1220. // The other side answered the call.
  1221. pBChannel = pEvent->pReceivingObject;
  1222. ASSERT(pBChannel && pBChannel->ObjectType == BCHANNEL_OBJECT_TYPE);
  1223. pBChannel->pPeerBChannel = pEvent->pSendingObject;
  1224. DBG_FILTER(pAdapter,DBG_TAPICALL_ON,
  1225. ("#%d CallState=0x%X CARD_EVENT_CONNECT from #%d\n",
  1226. pBChannel->ObjectID, pBChannel->CallState,
  1227. (pBChannel->pPeerBChannel == NULL) ? -1 :
  1228. pBChannel->pPeerBChannel->ObjectID));
  1229. if (pBChannel->Flags & VCF_OUTGOING_CALL)
  1230. {
  1231. // The other side answered the call.
  1232. CompleteCmMakeCall(pBChannel, NDIS_STATUS_SUCCESS);
  1233. }
  1234. break;
  1235. case CARD_EVENT_DISCONNECT:
  1236. // The other side has closed the call.
  1237. pBChannel = pEvent->pReceivingObject;
  1238. ASSERT(pBChannel && pBChannel->ObjectType == BCHANNEL_OBJECT_TYPE);
  1239. DBG_FILTER(pAdapter,DBG_TAPICALL_ON,
  1240. ("#%d CallState=0x%X CARD_EVENT_DISCONNECT from #%d\n",
  1241. pBChannel->ObjectID, pBChannel->CallState,
  1242. (pBChannel->pPeerBChannel == NULL) ? -1 :
  1243. pBChannel->pPeerBChannel->ObjectID));
  1244. pBChannel->pPeerBChannel = NULL;
  1245. if (pBChannel->Flags & VCF_OUTGOING_CALL)
  1246. {
  1247. if (pBChannel->CallState != LINECALLSTATE_CONNECTED)
  1248. {
  1249. // Call never made it to the connected state.
  1250. CompleteCmMakeCall(pBChannel, NDIS_STATUS_FAILURE);
  1251. }
  1252. else
  1253. {
  1254. // Call was disconnected by remote endpoint.
  1255. InitiateCallTeardown(pAdapter, pBChannel);
  1256. }
  1257. }
  1258. else if (pBChannel->Flags & VCF_INCOMING_CALL)
  1259. {
  1260. InitiateCallTeardown(pAdapter, pBChannel);
  1261. }
  1262. break;
  1263. case CARD_EVENT_RECEIVE:
  1264. pBChannel = pEvent->pReceivingObject;
  1265. ASSERT(pBChannel && pBChannel->ObjectType == BCHANNEL_OBJECT_TYPE);
  1266. DBG_FILTER(pAdapter,DBG_TXRX_VERBOSE_ON,
  1267. ("#%d CallState=0x%X CARD_EVENT_RECEIVE from #%d\n",
  1268. pBChannel->ObjectID, pBChannel->CallState,
  1269. (pBChannel->pPeerBChannel == NULL) ? -1 :
  1270. pBChannel->pPeerBChannel->ObjectID));
  1271. if (pBChannel->CallState == LINECALLSTATE_CONNECTED)
  1272. {
  1273. // Find out how big the packet is.
  1274. NdisQueryPacket(pEvent->pNdisPacket, NULL, NULL, NULL,
  1275. &ByteCount);
  1276. // Allocate memory for a copy of the data.
  1277. Status = ALLOCATE_MEMORY(pMemory, ByteCount,
  1278. pAdapter->MiniportAdapterHandle);
  1279. if (Status == NDIS_STATUS_SUCCESS)
  1280. {
  1281. NdisAllocateBuffer(&Status, &pDstNdisBuffer,
  1282. pAdapter->pCard->BufferPoolHandle,
  1283. pMemory, ByteCount);
  1284. if (Status == NDIS_STATUS_SUCCESS)
  1285. {
  1286. TpiCopyFromPacketToBuffer(pEvent->pNdisPacket, 0,
  1287. ByteCount, pMemory,
  1288. &BytesCopied);
  1289. ASSERT(BytesCopied == ByteCount);
  1290. ReceivePacketHandler(pBChannel, pDstNdisBuffer,
  1291. ByteCount);
  1292. }
  1293. else
  1294. {
  1295. FREE_MEMORY(pMemory, ByteCount);
  1296. DBG_ERROR(pAdapter,("NdisAllocateBuffer Error=0x%X\n",
  1297. Status));
  1298. }
  1299. }
  1300. }
  1301. pPeerBChannel = pBChannel->pPeerBChannel;
  1302. if (pPeerBChannel)
  1303. {
  1304. pNewEvent = CardEventAllocate(pPeerBChannel->pAdapter->pCard);
  1305. if (pNewEvent)
  1306. {
  1307. pNewEvent->ulEventCode = CARD_EVENT_TRANSMIT_COMPLETE;
  1308. pNewEvent->pSendingObject = pBChannel;
  1309. pNewEvent->pReceivingObject = pPeerBChannel;
  1310. CardNotifyEvent(pPeerBChannel->pAdapter->pCard, pNewEvent);
  1311. }
  1312. }
  1313. else
  1314. {
  1315. DBG_WARNING(pAdapter,("pPeerBChannel == NULL\n"));
  1316. }
  1317. break;
  1318. case CARD_EVENT_TRANSMIT_COMPLETE:
  1319. pBChannel = pEvent->pReceivingObject;
  1320. ASSERT(pBChannel && pBChannel->ObjectType == BCHANNEL_OBJECT_TYPE);
  1321. DBG_FILTER(pAdapter,DBG_TXRX_VERBOSE_ON,
  1322. ("#%d CallState=0x%X CARD_EVENT_TRANSMIT_COMPLETE from #%d\n",
  1323. pBChannel->ObjectID, pBChannel->CallState,
  1324. (pBChannel->pPeerBChannel == NULL) ? -1 :
  1325. pBChannel->pPeerBChannel->ObjectID));
  1326. /*
  1327. // Remove the packet from the BChannel's TransmitBusyList and
  1328. // place it on the adapter's TransmitCompleteList now that the
  1329. // card has completed the transmit.
  1330. */
  1331. NdisAcquireSpinLock(&pAdapter->TransmitLock);
  1332. if (!IsListEmpty(&pBChannel->TransmitBusyList))
  1333. {
  1334. pList = RemoveHeadList(&pBChannel->TransmitBusyList);
  1335. InsertTailList(&pBChannel->pAdapter->TransmitCompleteList, pList);
  1336. }
  1337. NdisReleaseSpinLock(&pAdapter->TransmitLock);
  1338. TransmitCompleteHandler(pAdapter);
  1339. break;
  1340. default:
  1341. DBG_ERROR(pAdapter,("Unknown event code=%d\n",
  1342. pEvent->ulEventCode));
  1343. break;
  1344. }
  1345. CardEventRelease(pCard, pEvent);
  1346. NdisDprAcquireSpinLock(&pAdapter->EventLock);
  1347. }
  1348. NdisDprReleaseSpinLock(&pAdapter->EventLock);
  1349. }
  1350. #else // SAMPLE_DRIVER
  1351. // TODO - Add interrupt handler code here
  1352. #endif // SAMPLE_DRIVER
  1353. // DBG_LEAVE(pAdapter);
  1354. }
  1355. /* @doc INTERNAL Card Card_c CardCleanPhoneNumber
  1356. @func
  1357. <f CardCleanPhoneNumber> copies the phone number from the input string
  1358. to the output string, deleting any non-phone number characters (i.e.
  1359. dashes, parens, modem keywords, etc.).
  1360. @rdesc
  1361. <f CardCleanPhoneNumber> returns the length of the output string in bytes.
  1362. */
  1363. USHORT CardCleanPhoneNumber(
  1364. OUT PUCHAR Dst, // @parm
  1365. // A pointer to the output string.
  1366. IN PUSHORT Src, // @parm
  1367. // A pointer to the input string.
  1368. IN USHORT Length // @parm
  1369. // The length of the input string in bytes.
  1370. )
  1371. {
  1372. DBG_FUNC("CardCleanPhoneNumber")
  1373. USHORT NumDigits;
  1374. /*
  1375. // Strip out any character which are not digits or # or *.
  1376. */
  1377. for (NumDigits = 0; Length > 0; --Length)
  1378. {
  1379. if ((*Src >= '0' && *Src <= '9') ||
  1380. (*Src == '#' || *Src == '*'))
  1381. {
  1382. /*
  1383. // Make sure dial string is within the limit of the adapter.
  1384. */
  1385. if (NumDigits < CARD_MAX_DIAL_DIGITS)
  1386. {
  1387. ++NumDigits;
  1388. *Dst++ = (UCHAR) *Src;
  1389. }
  1390. else
  1391. {
  1392. break;
  1393. }
  1394. }
  1395. Src++;
  1396. }
  1397. *Dst++ = 0;
  1398. return (NumDigits);
  1399. }
  1400. /* @doc INTERNAL Card Card_c CardReset
  1401. @func
  1402. <f CardReset> issues a hard reset to the NIC. Same as power up.
  1403. @rdesc
  1404. <f CardReset> returns zero if it is successful.<nl>
  1405. Otherwise, a non-zero return value indicates an error condition.
  1406. */
  1407. NDIS_STATUS CardReset(
  1408. IN PCARD_OBJECT pCard // @parm
  1409. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  1410. )
  1411. {
  1412. DBG_FUNC("CardReset")
  1413. NDIS_STATUS Result = NDIS_STATUS_SUCCESS;
  1414. // Holds the result code returned by this function.
  1415. PMINIPORT_ADAPTER_OBJECT pAdapter;
  1416. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  1417. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  1418. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  1419. DBG_ENTER(pAdapter);
  1420. DBG_BREAK(pAdapter);
  1421. // TODO - Add code here to reset your hardware to its initial state.
  1422. DBG_RETURN(pAdapter, Result);
  1423. return (Result);
  1424. }
  1425. #if defined(SAMPLE_DRIVER)
  1426. /* @doc INTERNAL Card Card_c GET_BCHANNEL_FROM_PHONE_NUMBER
  1427. @func
  1428. <f GET_BCHANNEL_FROM_PHONE_NUMBER> assumes the phone number is a BChannel
  1429. index, and uses it to lookup the associated BChannel on one of our
  1430. adapters. Zero means use the first availble BChannel on another adapter.
  1431. @rdesc
  1432. <f GET_BCHANNEL_FROM_PHONE_NUMBER> returns a pointer to the associated
  1433. <t BCHANNEL_OBJECT> if successful. Otherwise, NULL is returned.
  1434. */
  1435. PBCHANNEL_OBJECT GET_BCHANNEL_FROM_PHONE_NUMBER(
  1436. IN PUCHAR pDialString // @parm
  1437. // A pointer to the dial string.
  1438. )
  1439. {
  1440. DBG_FUNC("GET_BCHANNEL_FROM_PHONE_NUMBER")
  1441. ULONG ulCalledID = 0;
  1442. // Phone number converted to BChannel ObjectID (spans all adapters).
  1443. ULONG ulAdapterIndex;
  1444. // Loop index.
  1445. /*
  1446. // Strip out any character which are not digits or # or *.
  1447. */
  1448. while (*pDialString)
  1449. {
  1450. if (*pDialString >= '0' && *pDialString <= '9')
  1451. {
  1452. ulCalledID *= 10;
  1453. ulCalledID += *pDialString - '0';
  1454. }
  1455. else
  1456. {
  1457. break;
  1458. }
  1459. pDialString++;
  1460. }
  1461. if (*pDialString)
  1462. {
  1463. DBG_ERROR(DbgInfo,("Invalid dial string '%s'\n", pDialString));
  1464. }
  1465. else
  1466. {
  1467. PMINIPORT_ADAPTER_OBJECT pAdapter;
  1468. for (ulAdapterIndex = 0; ulAdapterIndex < MAX_ADAPTERS; ++ulAdapterIndex)
  1469. {
  1470. // Does call want to look on specific adapter, or any?
  1471. if (ulCalledID == 0 || ulCalledID == ulAdapterIndex+1)
  1472. {
  1473. pAdapter = g_Adapters[ulAdapterIndex];
  1474. if (pAdapter)
  1475. {
  1476. // Find first available channel.
  1477. NdisAcquireSpinLock(&pAdapter->EventLock);
  1478. if (!IsListEmpty(&pAdapter->BChannelAvailableList))
  1479. {
  1480. PBCHANNEL_OBJECT pBChannel;
  1481. pBChannel = (PBCHANNEL_OBJECT) pAdapter->BChannelAvailableList.Blink;
  1482. if (pBChannel->NdisSapHandle &&
  1483. pBChannel->NdisVcHandle == NULL)
  1484. {
  1485. // Find first available listening channel.
  1486. pBChannel = (PBCHANNEL_OBJECT) RemoveTailList(
  1487. &pAdapter->BChannelAvailableList);
  1488. // Reset the link info so we can tell that it's
  1489. // not on the list.
  1490. InitializeListHead(&pBChannel->LinkList);
  1491. NdisReleaseSpinLock(&pAdapter->EventLock);
  1492. return (pBChannel);
  1493. }
  1494. }
  1495. NdisReleaseSpinLock(&pAdapter->EventLock);
  1496. }
  1497. }
  1498. }
  1499. }
  1500. return (NULL);
  1501. }
  1502. /* @doc INTERNAL Card Card_c CardNotifyEvent
  1503. @func
  1504. <f CardNotifyEvent> queues an IMS event to be processed by the DPC
  1505. handler when things quiet down.
  1506. @comm
  1507. We have to queue the event to be processed in DPC context. We have
  1508. to make sure that the queue is protected by a mutual exclusion
  1509. primative which cannot be violated by the callback.
  1510. */
  1511. VOID CardNotifyEvent(
  1512. IN PCARD_OBJECT pCard, // @parm
  1513. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  1514. IN PCARD_EVENT_OBJECT pEvent // @parm
  1515. // A pointer to the <t CARD_EVENT_OBJECT> associated with this event.
  1516. )
  1517. {
  1518. DBG_FUNC("CardNotifyEvent")
  1519. PMINIPORT_ADAPTER_OBJECT pAdapter;
  1520. // A pointer to the <t MINIPORT_ADAPTER_OBJECT>.
  1521. ASSERT(pCard && pCard->ObjectType == CARD_OBJECT_TYPE);
  1522. pAdapter = GET_ADAPTER_FROM_CARD(pCard);
  1523. // DBG_ENTER(pAdapter);
  1524. /*
  1525. // Schedule the event handler to run as soon as possible.
  1526. // We must schedule the event to go through the NDIS wrapper
  1527. // so the proper spin locks will be held.
  1528. // Don't schedule another event if processing is already in progress.
  1529. */
  1530. NdisAcquireSpinLock(&pAdapter->EventLock);
  1531. InsertTailList(&pAdapter->EventList, &pEvent->Queue);
  1532. NdisReleaseSpinLock(&pAdapter->EventLock);
  1533. if (pEvent->ulEventCode == CARD_EVENT_RING ||
  1534. pEvent->ulEventCode == CARD_EVENT_CONNECT ||
  1535. pEvent->ulEventCode == CARD_EVENT_DISCONNECT)
  1536. {
  1537. NdisMSetTimer(&pAdapter->EventTimer, 100);
  1538. }
  1539. else
  1540. {
  1541. NdisMSetTimer(&pAdapter->EventTimer, 0);
  1542. }
  1543. // DBG_LEAVE(pAdapter);
  1544. }
  1545. /* @doc INTERNAL Card Card_c CardEventAllocate
  1546. @func
  1547. <f CardEventAllocate> allocates an <t CARD_EVENT_OBJECT> from the
  1548. <p pCard>'s EventList.
  1549. @rdesc
  1550. <f CardEventAllocate> returns a pointer to a <t CARD_EVENT_OBJECT>
  1551. if it is successful.<nl>
  1552. Otherwise, a NULL return value indicates an error condition.
  1553. */
  1554. PCARD_EVENT_OBJECT CardEventAllocate(
  1555. IN PCARD_OBJECT pCard // @parm
  1556. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  1557. )
  1558. {
  1559. PCARD_EVENT_OBJECT pEvent;
  1560. // A pointer to the <t CARD_EVENT_OBJECT> associated with this event.
  1561. pEvent = &pCard->EventArray[pCard->NextEvent++];
  1562. ASSERT(pEvent->pReceivingObject == NULL);
  1563. if (pCard->NextEvent >= MAX_EVENTS)
  1564. {
  1565. pCard->NextEvent = 0;
  1566. }
  1567. return (pEvent);
  1568. }
  1569. /* @doc INTERNAL Card Card_c CardEventRelease
  1570. @func
  1571. <f CardEventRelease> returns a previously allocate <t CARD_EVENT_OBJECT>
  1572. to the <p pCard>'s EventList.
  1573. */
  1574. VOID CardEventRelease(
  1575. IN PCARD_OBJECT pCard, // @parm
  1576. // A pointer to the <t CARD_OBJECT> returned by <f CardCreate>.
  1577. IN PCARD_EVENT_OBJECT pEvent // @parm
  1578. // A pointer to the <t CARD_EVENT_OBJECT> associated with this event.
  1579. )
  1580. {
  1581. pEvent->pReceivingObject = NULL;
  1582. }
  1583. #endif // SAMPLE_DRIVER