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.

1314 lines
34 KiB

  1. /*++
  2. Copyright (c) 1989-1993 Microsoft Corporation
  3. Module Name:
  4. receive.c
  5. Abstract:
  6. This module contains the code to handle receive indication
  7. and posted receives for the Netbios module of the ISN transport.
  8. Author:
  9. Adam Barr (adamba) 22-November-1993
  10. Environment:
  11. Kernel mode
  12. Revision History:
  13. --*/
  14. #include "precomp.h"
  15. #pragma hdrstop
  16. //
  17. // This routine is a no-op to put in the NbiCallbacks table so
  18. // we can avoid checking for runt session frames (this is because
  19. // of how the if is structure below).
  20. //
  21. VOID
  22. NbiProcessSessionRunt(
  23. IN PIPX_LOCAL_TARGET RemoteAddress,
  24. IN ULONG MacOptions,
  25. IN PUCHAR PacketBuffer,
  26. IN UINT PacketSize
  27. )
  28. {
  29. return;
  30. }
  31. NB_CALLBACK_NO_TRANSFER NbiCallbacksNoTransfer[] = {
  32. NbiProcessFindName,
  33. NbiProcessNameRecognized,
  34. NbiProcessAddName,
  35. NbiProcessAddName, // processes name in use frames also
  36. NbiProcessDeleteName,
  37. NbiProcessSessionRunt, // in case get a short session packet
  38. NbiProcessSessionEnd,
  39. NbiProcessSessionEndAck,
  40. NbiProcessStatusQuery
  41. };
  42. #ifdef RSRC_TIMEOUT_DBG
  43. VOID
  44. NbiProcessDeathPacket(
  45. IN NDIS_HANDLE MacBindingHandle,
  46. IN NDIS_HANDLE MacReceiveContext,
  47. IN PIPX_LOCAL_TARGET RemoteAddress,
  48. IN ULONG MacOptions,
  49. IN PUCHAR LookaheadBuffer,
  50. IN UINT LookaheadBufferSize,
  51. IN UINT LookaheadBufferOffset,
  52. IN UINT PacketSize
  53. )
  54. /*++
  55. Routine Description:
  56. This routine handles NB_CMD_SESSION_DATA frames.
  57. Arguments:
  58. MacBindingHandle - A handle to use when calling NdisTransferData.
  59. MacReceiveContext - A context to use when calling NdisTransferData.
  60. RemoteAddress - The local target this packet was received from.
  61. MacOptions - The MAC options for the underlying NDIS binding.
  62. LookaheadBuffer - The lookahead buffer, starting at the IPX
  63. header.
  64. LookaheadBufferSize - The length of the lookahead data.
  65. LookaheadBufferOffset - The offset to add when calling
  66. NdisTransferData.
  67. PacketSize - The total length of the packet, starting at the
  68. IPX header.
  69. Return Value:
  70. None.
  71. --*/
  72. {
  73. NB_CONNECTION UNALIGNED * Conn = (NB_CONNECTION UNALIGNED *)LookaheadBuffer;
  74. NB_SESSION UNALIGNED * Sess = (NB_SESSION UNALIGNED *)(&Conn->Session);
  75. PCONNECTION Connection;
  76. PDEVICE Device = NbiDevice;
  77. ULONG Hash;
  78. NB_DEFINE_LOCK_HANDLE (LockHandle)
  79. DbgPrint("******Received death packet - connid %x\n",Sess->DestConnectionId);
  80. if ( !NbiGlobalDebugResTimeout ) {
  81. return;
  82. }
  83. if (Sess->DestConnectionId != 0xffff) {
  84. //
  85. // This is an active connection, find it using
  86. // our session id.
  87. //
  88. Hash = (Sess->DestConnectionId & CONNECTION_HASH_MASK) >> CONNECTION_HASH_SHIFT;
  89. NB_SYNC_GET_LOCK (&Device->Lock, &LockHandle);
  90. Connection = Device->ConnectionHash[Hash].Connections;
  91. while (Connection != NULL) {
  92. if (Connection->LocalConnectionId == Sess->DestConnectionId) {
  93. break;
  94. }
  95. Connection = Connection->NextConnection;
  96. }
  97. if (Connection == NULL) {
  98. DbgPrint("********No Connection found with %x id\n",Sess->DestConnectionId);
  99. NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
  100. return;
  101. }
  102. DbgPrint("******Received death packet on conn %lx from <%.16s>\n",Connection,Connection->RemoteName);
  103. DbgBreakPoint();
  104. NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
  105. }
  106. }
  107. #endif //RSRC_TIMEOUT_DBG
  108. BOOLEAN
  109. NbiReceive(
  110. IN NDIS_HANDLE MacBindingHandle,
  111. IN NDIS_HANDLE MacReceiveContext,
  112. IN ULONG_PTR FwdAdapterCtx,
  113. IN PIPX_LOCAL_TARGET RemoteAddress,
  114. IN ULONG MacOptions,
  115. IN PUCHAR LookaheadBuffer,
  116. IN UINT LookaheadBufferSize,
  117. IN UINT LookaheadBufferOffset,
  118. IN UINT PacketSize,
  119. IN PMDL pMdl
  120. )
  121. /*++
  122. Routine Description:
  123. This routine handles receive indications from IPX.
  124. Arguments:
  125. MacBindingHandle - A handle to use when calling NdisTransferData.
  126. MacReceiveContext - A context to use when calling NdisTransferData.
  127. RemoteAddress - The local target this packet was received from.
  128. MacOptions - The MAC options for the underlying NDIS binding.
  129. LookaheadBuffer - The lookahead buffer, starting at the IPX
  130. header.
  131. LookaheadBufferSize - The length of the lookahead data.
  132. LookaheadBufferOffset - The offset to add when calling
  133. NdisTransferData.
  134. PacketSize - The total length of the packet, starting at the
  135. IPX header.
  136. Return Value:
  137. TRUE - receivepacket taken, will return later with NdisReturnPacket.
  138. Currently, we always return FALSE.
  139. --*/
  140. {
  141. PNB_FRAME NbFrame = (PNB_FRAME)LookaheadBuffer;
  142. UCHAR DataStreamType;
  143. //
  144. // We know that this is a frame with a valid IPX header
  145. // because IPX would not give it to use otherwise. However,
  146. // it does not check the source socket.
  147. //
  148. if (NbFrame->Connectionless.IpxHeader.SourceSocket != NB_SOCKET) {
  149. return FALSE;
  150. }
  151. ++NbiDevice->Statistics.PacketsReceived;
  152. // First assume that the DataStreamType is at the normal place i.e 2nd byte
  153. //
  154. // Now see if this is a name frame.
  155. //
  156. if ( PacketSize == sizeof(IPX_HEADER) + sizeof(NB_NAME_FRAME) ) {
  157. // In the internet mode, the DataStreamType2 becomes DataStreamType
  158. if (NbFrame->Connectionless.IpxHeader.PacketType == 0x14 ) {
  159. DataStreamType = NbFrame->Connectionless.NameFrame.DataStreamType2;
  160. } else {
  161. DataStreamType = NbFrame->Connectionless.NameFrame.DataStreamType;
  162. }
  163. // Is this a name frame?
  164. // NB_CMD_FIND_NAME = 1 .... NB_CMD_DELETE_NAME = 5
  165. //
  166. if ((DataStreamType >= NB_CMD_FIND_NAME) && (DataStreamType <= NB_CMD_DELETE_NAME)) {
  167. if (LookaheadBufferSize == PacketSize) {
  168. (*NbiCallbacksNoTransfer[DataStreamType-1])(
  169. RemoteAddress,
  170. MacOptions,
  171. LookaheadBuffer,
  172. LookaheadBufferSize);
  173. }
  174. return FALSE;
  175. }
  176. }
  177. #ifdef RSRC_TIMEOUT_DBG
  178. if ((PacketSize >= sizeof(NB_CONNECTION)) &&
  179. (NbFrame->Connection.Session.DataStreamType == NB_CMD_DEATH_PACKET)) {
  180. NbiProcessDeathPacket(
  181. MacBindingHandle,
  182. MacReceiveContext,
  183. RemoteAddress,
  184. MacOptions,
  185. LookaheadBuffer,
  186. LookaheadBufferSize,
  187. LookaheadBufferOffset,
  188. PacketSize);
  189. }
  190. #endif //RSRC_TIMEOUT_DBG
  191. if ((PacketSize >= sizeof(NB_CONNECTION)) &&
  192. (NbFrame->Connection.Session.DataStreamType == NB_CMD_SESSION_DATA)) {
  193. NbiProcessSessionData(
  194. MacBindingHandle,
  195. MacReceiveContext,
  196. RemoteAddress,
  197. MacOptions,
  198. LookaheadBuffer,
  199. LookaheadBufferSize,
  200. LookaheadBufferOffset,
  201. PacketSize);
  202. } else {
  203. DataStreamType = NbFrame->Connectionless.NameFrame.DataStreamType;
  204. // Handle NB_CMD_SESSION_END = 7 ... NB_CMD_STATUS_QUERY = 9
  205. //
  206. if ((DataStreamType >= NB_CMD_SESSION_END ) && (DataStreamType <= NB_CMD_STATUS_QUERY)) {
  207. if (LookaheadBufferSize == PacketSize) {
  208. (*NbiCallbacksNoTransfer[DataStreamType-1])(
  209. RemoteAddress,
  210. MacOptions,
  211. LookaheadBuffer,
  212. LookaheadBufferSize);
  213. }
  214. } else if (DataStreamType == NB_CMD_STATUS_RESPONSE) {
  215. NbiProcessStatusResponse(
  216. MacBindingHandle,
  217. MacReceiveContext,
  218. RemoteAddress,
  219. MacOptions,
  220. LookaheadBuffer,
  221. LookaheadBufferSize,
  222. LookaheadBufferOffset,
  223. PacketSize);
  224. } else if ((DataStreamType == NB_CMD_DATAGRAM) ||
  225. (DataStreamType == NB_CMD_BROADCAST_DATAGRAM)) {
  226. NbiProcessDatagram(
  227. MacBindingHandle,
  228. MacReceiveContext,
  229. RemoteAddress,
  230. MacOptions,
  231. LookaheadBuffer,
  232. LookaheadBufferSize,
  233. LookaheadBufferOffset,
  234. PacketSize,
  235. (BOOLEAN)(DataStreamType == NB_CMD_BROADCAST_DATAGRAM));
  236. }
  237. }
  238. return FALSE;
  239. } /* NbiReceive */
  240. VOID
  241. NbiReceiveComplete(
  242. IN USHORT NicId
  243. )
  244. /*++
  245. Routine Description:
  246. This routine handles receive complete indications from IPX.
  247. Arguments:
  248. NicId - The NIC ID on which a receive was previously indicated.
  249. Return Value:
  250. None.
  251. --*/
  252. {
  253. PLIST_ENTRY p;
  254. PADDRESS Address;
  255. PREQUEST Request;
  256. PNB_RECEIVE_BUFFER ReceiveBuffer;
  257. PDEVICE Device = NbiDevice;
  258. LIST_ENTRY LocalList;
  259. PCONNECTION Connection;
  260. NB_DEFINE_LOCK_HANDLE (LockHandle);
  261. //
  262. // Complete any pending receive requests.
  263. //
  264. if (!IsListEmpty (&Device->ReceiveCompletionQueue)) {
  265. p = NB_REMOVE_HEAD_LIST(
  266. &Device->ReceiveCompletionQueue,
  267. &Device->Lock);
  268. while (!NB_LIST_WAS_EMPTY(&Device->ReceiveCompletionQueue, p)) {
  269. Request = LIST_ENTRY_TO_REQUEST (p);
  270. Connection = (PCONNECTION)REQUEST_OPEN_CONTEXT(Request);
  271. NB_DEBUG2 (RECEIVE, ("Completing receive %lx (%d), status %lx\n",
  272. Request, REQUEST_INFORMATION(Request), REQUEST_STATUS(Request)));
  273. NbiCompleteRequest (Request);
  274. NbiFreeRequest (NbiDevice, Request);
  275. Connection->ReceiveState = CONNECTION_RECEIVE_IDLE;
  276. NbiDereferenceConnection (Connection, CREF_RECEIVE);
  277. p = NB_REMOVE_HEAD_LIST(
  278. &Device->ReceiveCompletionQueue,
  279. &Device->Lock);
  280. }
  281. }
  282. //
  283. // Indicate any datagrams to clients.
  284. //
  285. if (!IsListEmpty (&Device->ReceiveDatagrams)) {
  286. p = NB_REMOVE_HEAD_LIST(
  287. &Device->ReceiveDatagrams,
  288. &Device->Lock);
  289. while (!NB_LIST_WAS_EMPTY(&Device->ReceiveDatagrams, p)) {
  290. ReceiveBuffer = CONTAINING_RECORD (p, NB_RECEIVE_BUFFER, WaitLinkage);
  291. Address = ReceiveBuffer->Address;
  292. NbiIndicateDatagram(
  293. Address,
  294. ReceiveBuffer->RemoteName,
  295. ReceiveBuffer->Data,
  296. ReceiveBuffer->DataLength);
  297. #if defined(_PNP_POWER)
  298. NbiPushReceiveBuffer ( ReceiveBuffer );
  299. #else
  300. NB_PUSH_ENTRY_LIST(
  301. &Device->ReceiveBufferList,
  302. &ReceiveBuffer->PoolLinkage,
  303. &Device->Lock);
  304. #endif _PNP_POWER
  305. NbiDereferenceAddress (Address, AREF_FIND);
  306. p = NB_REMOVE_HEAD_LIST(
  307. &Device->ReceiveDatagrams,
  308. &Device->Lock);
  309. }
  310. }
  311. //
  312. // Start packetizing connections.
  313. //
  314. if (!IsListEmpty (&Device->PacketizeConnections)) {
  315. NB_SYNC_GET_LOCK (&Device->Lock, &LockHandle);
  316. //
  317. // Check again because it may just have become
  318. // empty, and the code below depends on it being
  319. // non-empty.
  320. //
  321. if (!IsListEmpty (&Device->PacketizeConnections)) {
  322. //
  323. // We copy the list locally, in case someone gets
  324. // put back on it. We have to hack the end so
  325. // it points to LocalList instead of PacketizeConnections.
  326. //
  327. LocalList = Device->PacketizeConnections;
  328. LocalList.Flink->Blink = &LocalList;
  329. LocalList.Blink->Flink = &LocalList;
  330. InitializeListHead (&Device->PacketizeConnections);
  331. //
  332. // Set all these connections to not be on the list, so
  333. // NbiStopConnection won't try to take them off.
  334. //
  335. for (p = LocalList.Flink; p != &LocalList; p = p->Flink) {
  336. Connection = CONTAINING_RECORD (p, CONNECTION, PacketizeLinkage);
  337. CTEAssert (Connection->OnPacketizeQueue);
  338. Connection->OnPacketizeQueue = FALSE;
  339. }
  340. NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
  341. while (TRUE) {
  342. p = RemoveHeadList (&LocalList);
  343. if (p == &LocalList) {
  344. break;
  345. }
  346. Connection = CONTAINING_RECORD (p, CONNECTION, PacketizeLinkage);
  347. NB_SYNC_GET_LOCK (&Connection->Lock, &LockHandle);
  348. if ((Connection->State == CONNECTION_STATE_ACTIVE) &&
  349. (Connection->SubState == CONNECTION_SUBSTATE_A_PACKETIZE)) {
  350. NbiPacketizeSend(
  351. Connection
  352. NB_LOCK_HANDLE_ARG (LockHandle)
  353. );
  354. } else {
  355. NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
  356. }
  357. NbiDereferenceConnection (Connection, CREF_PACKETIZE);
  358. }
  359. } else {
  360. NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
  361. }
  362. }
  363. } /* NbiReceiveComplete */
  364. VOID
  365. NbiTransferDataComplete(
  366. IN PNDIS_PACKET Packet,
  367. IN NDIS_STATUS Status,
  368. IN UINT BytesTransferred
  369. )
  370. /*++
  371. Routine Description:
  372. This routine handles a transfer data complete indication from
  373. IPX, indicating that a previously issued NdisTransferData
  374. call has completed.
  375. Arguments:
  376. Packet - The packet associated with the transfer.
  377. Status - The status of the transfer.
  378. BytesTransferred - The number of bytes transferred.
  379. Return Value:
  380. None.
  381. --*/
  382. {
  383. PNB_RECEIVE_RESERVED ReceiveReserved;
  384. PNB_RECEIVE_BUFFER ReceiveBuffer;
  385. PADDRESS Address;
  386. PCONNECTION Connection;
  387. PNDIS_BUFFER CurBuffer, TmpBuffer;
  388. PREQUEST AdapterStatusRequest;
  389. PDEVICE Device = NbiDevice;
  390. CTELockHandle CancelLH;
  391. NB_DEFINE_LOCK_HANDLE (LockHandle);
  392. ReceiveReserved = (PNB_RECEIVE_RESERVED)(Packet->ProtocolReserved);
  393. switch (ReceiveReserved->Type) {
  394. case RECEIVE_TYPE_DATA:
  395. CTEAssert (ReceiveReserved->TransferInProgress);
  396. ReceiveReserved->TransferInProgress = FALSE;
  397. Connection = ReceiveReserved->u.RR_CO.Connection;
  398. NB_GET_CANCEL_LOCK( &CancelLH );
  399. NB_SYNC_GET_LOCK (&Connection->Lock, &LockHandle);
  400. if (Status != NDIS_STATUS_SUCCESS) {
  401. if (Connection->State == CONNECTION_STATE_ACTIVE) {
  402. Connection->CurrentReceive = Connection->PreviousReceive;
  403. Connection->ReceiveState = CONNECTION_RECEIVE_ACTIVE;
  404. NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
  405. NB_FREE_CANCEL_LOCK( CancelLH );
  406. //
  407. // Send a resend ack?
  408. //
  409. } else {
  410. //
  411. // This aborts the current receive and
  412. // releases the connection lock.
  413. //
  414. NbiCompleteReceive(
  415. Connection,
  416. ReceiveReserved->u.RR_CO.EndOfMessage,
  417. CancelLH
  418. NB_LOCK_HANDLE_ARG(LockHandle));
  419. }
  420. } else {
  421. Connection->CurrentReceive.Offset += BytesTransferred;
  422. Connection->CurrentReceive.MessageOffset += BytesTransferred;
  423. if (ReceiveReserved->u.RR_CO.CompleteReceive ||
  424. (Connection->State != CONNECTION_STATE_ACTIVE)) {
  425. if (ReceiveReserved->u.RR_CO.EndOfMessage) {
  426. CTEAssert (!ReceiveReserved->u.RR_CO.PartialReceive);
  427. ++Connection->ReceiveSequence;
  428. ++Connection->LocalRcvSequenceMax; // harmless if NewNetbios is FALSE
  429. Connection->CurrentReceive.MessageOffset = 0;
  430. Connection->CurrentIndicateOffset = 0;
  431. } else if (Connection->NewNetbios) {
  432. if (ReceiveReserved->u.RR_CO.PartialReceive) {
  433. Connection->CurrentIndicateOffset += BytesTransferred;
  434. } else {
  435. ++Connection->ReceiveSequence;
  436. ++Connection->LocalRcvSequenceMax;
  437. Connection->CurrentIndicateOffset = 0;
  438. }
  439. }
  440. //
  441. // This sends an ack and releases the connection lock.
  442. //
  443. NbiCompleteReceive(
  444. Connection,
  445. ReceiveReserved->u.RR_CO.EndOfMessage,
  446. CancelLH
  447. NB_LOCK_HANDLE_ARG(LockHandle));
  448. } else {
  449. NB_SYNC_SWAP_IRQL( CancelLH, LockHandle );
  450. NB_FREE_CANCEL_LOCK( CancelLH );
  451. Connection->ReceiveState = CONNECTION_RECEIVE_ACTIVE;
  452. if (Connection->NewNetbios) {
  453. //
  454. // A partial receive should only happen if we are
  455. // completing the receive.
  456. //
  457. CTEAssert (!ReceiveReserved->u.RR_CO.PartialReceive);
  458. ++Connection->ReceiveSequence;
  459. ++Connection->LocalRcvSequenceMax;
  460. Connection->CurrentIndicateOffset = 0;
  461. if ((Connection->CurrentReceiveNoPiggyback) ||
  462. ((Device->AckWindow != 0) &&
  463. (++Connection->ReceivesWithoutAck >= Device->AckWindow))) {
  464. NbiSendDataAck(
  465. Connection,
  466. NbiAckResponse
  467. NB_LOCK_HANDLE_ARG(LockHandle));
  468. } else {
  469. NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
  470. }
  471. } else {
  472. NbiSendDataAck(
  473. Connection,
  474. NbiAckResponse
  475. NB_LOCK_HANDLE_ARG(LockHandle));
  476. }
  477. }
  478. }
  479. //
  480. // Free the NDIS buffer chain if we allocated one.
  481. //
  482. if (!ReceiveReserved->u.RR_CO.NoNdisBuffer) {
  483. NdisQueryPacket (Packet, NULL, NULL, &CurBuffer, NULL);
  484. while (CurBuffer) {
  485. TmpBuffer = NDIS_BUFFER_LINKAGE (CurBuffer);
  486. NdisFreeBuffer (CurBuffer);
  487. CurBuffer = TmpBuffer;
  488. }
  489. }
  490. NdisReinitializePacket (Packet);
  491. ExInterlockedPushEntrySList(
  492. &Device->ReceivePacketList,
  493. &ReceiveReserved->PoolLinkage,
  494. &NbiGlobalPoolInterlock);
  495. NbiDereferenceConnection (Connection, CREF_INDICATE);
  496. break;
  497. case RECEIVE_TYPE_DATAGRAM:
  498. CTEAssert (ReceiveReserved->TransferInProgress);
  499. ReceiveReserved->TransferInProgress = FALSE;
  500. ReceiveBuffer = ReceiveReserved->u.RR_DG.ReceiveBuffer;
  501. //
  502. // Free the packet used for the transfer.
  503. //
  504. ReceiveReserved->u.RR_DG.ReceiveBuffer = NULL;
  505. NdisReinitializePacket (Packet);
  506. ExInterlockedPushEntrySList(
  507. &Device->ReceivePacketList,
  508. &ReceiveReserved->PoolLinkage,
  509. &NbiGlobalPoolInterlock);
  510. //
  511. // If it succeeded then queue it for indication,
  512. // otherwise free the receive buffer also.
  513. //
  514. if (Status == STATUS_SUCCESS) {
  515. ReceiveBuffer->DataLength = BytesTransferred;
  516. NB_INSERT_HEAD_LIST(
  517. &Device->ReceiveDatagrams,
  518. &ReceiveBuffer->WaitLinkage,
  519. &Device->Lock);
  520. } else {
  521. Address = ReceiveBuffer->Address;
  522. #if defined(_PNP_POWER)
  523. NbiPushReceiveBuffer ( ReceiveBuffer );
  524. #else
  525. NB_PUSH_ENTRY_LIST(
  526. &Device->ReceiveBufferList,
  527. &ReceiveBuffer->PoolLinkage,
  528. &Device->Lock);
  529. #endif _PNP_POWER
  530. NbiDereferenceAddress (Address, AREF_FIND);
  531. }
  532. break;
  533. case RECEIVE_TYPE_ADAPTER_STATUS:
  534. CTEAssert (ReceiveReserved->TransferInProgress);
  535. ReceiveReserved->TransferInProgress = FALSE;
  536. AdapterStatusRequest = ReceiveReserved->u.RR_AS.Request;
  537. //
  538. // Free the packet used for the transfer.
  539. //
  540. NdisReinitializePacket (Packet);
  541. ExInterlockedPushEntrySList(
  542. &Device->ReceivePacketList,
  543. &ReceiveReserved->PoolLinkage,
  544. &NbiGlobalPoolInterlock);
  545. //
  546. // Complete the request.
  547. //
  548. if (Status == STATUS_SUCCESS) {
  549. //
  550. // REQUEST_STATUS() is already to set to SUCCESS or
  551. // BUFFER_OVERFLOW based on whether the buffer was
  552. // big enough.
  553. //
  554. REQUEST_INFORMATION(AdapterStatusRequest) = BytesTransferred;
  555. } else {
  556. REQUEST_INFORMATION(AdapterStatusRequest) = 0;
  557. REQUEST_STATUS(AdapterStatusRequest) = STATUS_UNEXPECTED_NETWORK_ERROR;
  558. }
  559. NbiCompleteRequest (AdapterStatusRequest);
  560. NbiFreeRequest (Device, AdapterStatusRequest);
  561. NbiDereferenceDevice (Device, DREF_STATUS_QUERY);
  562. break;
  563. }
  564. } /* NbiTransferDataComplete */
  565. VOID
  566. NbiAcknowledgeReceive(
  567. IN PCONNECTION Connection
  568. IN NB_LOCK_HANDLE_PARAM(LockHandle)
  569. )
  570. /*++
  571. Routine Description:
  572. This routine is called when a receive needs to be acked to
  573. the remote. It either sends a data ack or queues up a piggyback
  574. ack request.
  575. NOTE: THIS FUNCTION IS CALLED WITH THE CONNECTION LOCK HELD
  576. AND RETURNS WITH IT RELEASED.
  577. Arguments:
  578. Connection - Pointer to the connection.
  579. LockHandle - The handle with which Connection->Lock was acquired.
  580. Return Value:
  581. None.
  582. --*/
  583. {
  584. PDEVICE Device = NbiDevice;
  585. if (Connection->NewNetbios) {
  586. //
  587. // CurrentReceiveNoPiggyback is based on the bits he
  588. // set in his frame, NoPiggybackHeuristic is based on
  589. // guesses about the traffic pattern, it is set to
  590. // TRUE if we think we should not piggyback.
  591. //
  592. if ((!Device->EnablePiggyBackAck) ||
  593. (Connection->CurrentReceiveNoPiggyback) ||
  594. (Connection->PiggybackAckTimeout) ||
  595. (Connection->NoPiggybackHeuristic)) {
  596. //
  597. // This releases the lock.
  598. //
  599. NbiSendDataAck(
  600. Connection,
  601. NbiAckResponse
  602. NB_LOCK_HANDLE_ARG(LockHandle));
  603. } else {
  604. if (!Connection->DataAckPending) {
  605. NB_DEFINE_LOCK_HANDLE (LockHandle1)
  606. //
  607. // Some stacks can have multiple messages
  608. // outstanding, so we may already have an
  609. // ack queued.
  610. //
  611. Connection->DataAckTimeouts = 0;
  612. Connection->DataAckPending = TRUE;
  613. ++Device->Statistics.PiggybackAckQueued;
  614. if (!Connection->OnDataAckQueue) {
  615. NB_SYNC_GET_LOCK (&Device->TimerLock, &LockHandle1);
  616. if (!Connection->OnDataAckQueue) {
  617. Connection->OnDataAckQueue = TRUE;
  618. InsertTailList (&Device->DataAckConnections, &Connection->DataAckLinkage);
  619. }
  620. if (!Device->DataAckActive) {
  621. NbiStartShortTimer (Device);
  622. Device->DataAckActive = TRUE;
  623. }
  624. NB_SYNC_FREE_LOCK (&Device->TimerLock, LockHandle1);
  625. }
  626. //
  627. // Clear this, since a message ack resets the count.
  628. //
  629. Connection->ReceivesWithoutAck = 0;
  630. }
  631. NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
  632. }
  633. } else {
  634. //
  635. // This releases the lock.
  636. //
  637. NbiSendDataAck(
  638. Connection,
  639. NbiAckResponse
  640. NB_LOCK_HANDLE_ARG(LockHandle));
  641. }
  642. }
  643. VOID
  644. NbiCompleteReceive(
  645. IN PCONNECTION Connection,
  646. IN BOOLEAN EndOfMessage,
  647. IN CTELockHandle CancelLH
  648. IN NB_LOCK_HANDLE_PARAM(LockHandle)
  649. )
  650. /*++
  651. Routine Description:
  652. This routine is called when we have filled up a receive request
  653. and need to complete it.
  654. NOTE: THIS FUNCTION IS CALLED WITH THE CONNECTION LOCK HELD
  655. AND RETURNS WITH IT RELEASED.
  656. THIS ROUTINE ALSO HOLDS CANCEL SPIN LOCK WHEN IT IS CALLED
  657. AND RELEASES IT WHEN IT RETURNS.
  658. Arguments:
  659. Connection - Pointer to the connection.
  660. EndOfMessage - BOOLEAN set to true if the message end was received.
  661. LockHandle - The handle with which Connection->Lock was acquired.
  662. Return Value:
  663. None.
  664. --*/
  665. {
  666. PREQUEST Request;
  667. PDEVICE Device = NbiDevice;
  668. //
  669. // Complete the current receive request. If the connection
  670. // has shut down then we complete it right here, otherwise
  671. // we queue it for completion in the receive complete
  672. // handler.
  673. //
  674. Request = Connection->ReceiveRequest;
  675. IoSetCancelRoutine (Request, (PDRIVER_CANCEL)NULL);
  676. NB_SYNC_SWAP_IRQL( CancelLH, LockHandle );
  677. NB_FREE_CANCEL_LOCK( CancelLH );
  678. if (Connection->State != CONNECTION_STATE_ACTIVE) {
  679. Connection->ReceiveRequest = NULL; // StopConnection won't do this
  680. REQUEST_STATUS(Request) = Connection->Status;
  681. NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
  682. NB_DEBUG2 (RECEIVE, ("Completing receive %lx (%d), status %lx\n",
  683. Request, REQUEST_INFORMATION(Request), REQUEST_STATUS(Request)));
  684. NbiCompleteRequest (Request);
  685. NbiFreeRequest (NbiDevice, Request);
  686. ++Connection->ConnectionInfo.ReceiveErrors;
  687. NbiDereferenceConnection (Connection, CREF_RECEIVE);
  688. } else {
  689. REQUEST_INFORMATION (Request) = Connection->CurrentReceive.Offset;
  690. if (EndOfMessage) {
  691. REQUEST_STATUS(Request) = STATUS_SUCCESS;
  692. } else {
  693. REQUEST_STATUS(Request) = STATUS_BUFFER_OVERFLOW;
  694. }
  695. //
  696. // If we indicated to the client, adjust this down by the
  697. // amount of data taken, when it hits zero we can reindicate.
  698. //
  699. if (Connection->ReceiveUnaccepted) {
  700. NB_DEBUG2 (RECEIVE, ("Moving Unaccepted %d down by %d\n",
  701. Connection->ReceiveUnaccepted, Connection->CurrentReceive.Offset));
  702. if (Connection->CurrentReceive.Offset >= Connection->ReceiveUnaccepted) {
  703. Connection->ReceiveUnaccepted = 0;
  704. } else {
  705. Connection->ReceiveUnaccepted -= Connection->CurrentReceive.Offset;
  706. }
  707. }
  708. //
  709. // Check whether to activate another receive?
  710. //
  711. Connection->ReceiveState = CONNECTION_RECEIVE_PENDING;
  712. Connection->ReceiveRequest = NULL;
  713. //
  714. // This releases the lock.
  715. //
  716. if (Connection->NewNetbios) {
  717. if (EndOfMessage) {
  718. NbiAcknowledgeReceive(
  719. Connection
  720. NB_LOCK_HANDLE_ARG(LockHandle));
  721. } else {
  722. if (Connection->CurrentIndicateOffset != 0) {
  723. NbiSendDataAck(
  724. Connection,
  725. NbiAckResend
  726. NB_LOCK_HANDLE_ARG(LockHandle));
  727. } else if ((Connection->CurrentReceiveNoPiggyback) ||
  728. ((Device->AckWindow != 0) &&
  729. (++Connection->ReceivesWithoutAck >= Device->AckWindow))) {
  730. NbiSendDataAck(
  731. Connection,
  732. NbiAckResponse
  733. NB_LOCK_HANDLE_ARG(LockHandle));
  734. } else {
  735. NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
  736. }
  737. }
  738. } else {
  739. NbiSendDataAck(
  740. Connection,
  741. EndOfMessage ? NbiAckResponse : NbiAckResend
  742. NB_LOCK_HANDLE_ARG(LockHandle));
  743. }
  744. ++Connection->ConnectionInfo.ReceivedTsdus;
  745. //
  746. // This will complete the request inside ReceiveComplete,
  747. // dereference the connection, and set the state to IDLE.
  748. //
  749. if ( Request->Type != 0x6 ) {
  750. DbgPrint("NB: tracking pool corruption in rcv irp %lx \n", Request );
  751. DbgBreakPoint();
  752. }
  753. NB_INSERT_TAIL_LIST(
  754. &Device->ReceiveCompletionQueue,
  755. REQUEST_LINKAGE (Request),
  756. &Device->Lock);
  757. }
  758. } /* NbiCompleteReceive */
  759. NTSTATUS
  760. NbiTdiReceive(
  761. IN PDEVICE Device,
  762. IN PREQUEST Request
  763. )
  764. /*++
  765. Routine Description:
  766. This routine does a receive on an active connection.
  767. Arguments:
  768. Device - The netbios device.
  769. Request - The request describing the receive.
  770. Return Value:
  771. NTSTATUS - status of operation.
  772. --*/
  773. {
  774. PCONNECTION Connection;
  775. NB_DEFINE_SYNC_CONTEXT (SyncContext)
  776. NB_DEFINE_LOCK_HANDLE (LockHandle)
  777. CTELockHandle CancelLH;
  778. //
  779. // Check that the file type is valid
  780. //
  781. if (REQUEST_OPEN_TYPE(Request) != (PVOID)TDI_CONNECTION_FILE)
  782. {
  783. CTEAssert(FALSE);
  784. return (STATUS_INVALID_ADDRESS_COMPONENT);
  785. }
  786. //
  787. // First make sure the connection is valid.
  788. //
  789. Connection = (PCONNECTION)REQUEST_OPEN_CONTEXT(Request);
  790. if (Connection->Type == NB_CONNECTION_SIGNATURE) {
  791. NB_GET_CANCEL_LOCK( &CancelLH );
  792. NB_BEGIN_SYNC (&SyncContext);
  793. NB_SYNC_GET_LOCK (&Connection->Lock, &LockHandle);
  794. //
  795. // Make sure the connection is in a good state.
  796. //
  797. if (Connection->State == CONNECTION_STATE_ACTIVE) {
  798. //
  799. // If the connection is idle then send it now, otherwise
  800. // queue it.
  801. //
  802. if (!Request->Cancel) {
  803. IoSetCancelRoutine (Request, NbiCancelReceive);
  804. NB_SYNC_SWAP_IRQL( CancelLH, LockHandle );
  805. NB_FREE_CANCEL_LOCK( CancelLH );
  806. NbiReferenceConnectionSync (Connection, CREF_RECEIVE);
  807. //
  808. // Insert this in our queue, then see if we need
  809. // to wake up the remote.
  810. //
  811. REQUEST_SINGLE_LINKAGE(Request) = NULL;
  812. REQUEST_LIST_INSERT_TAIL(&Connection->ReceiveQueue, Request);
  813. if (Connection->ReceiveState != CONNECTION_RECEIVE_W_RCV) {
  814. NB_DEBUG2 (RECEIVE, ("Receive %lx, connection %lx idle\n", Request, Connection));
  815. NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
  816. } else {
  817. NB_DEBUG2 (RECEIVE, ("Receive %lx, connection %lx awakened\n", Request, Connection));
  818. Connection->ReceiveState = CONNECTION_RECEIVE_IDLE;
  819. //
  820. // This releases the lock.
  821. //
  822. if (Connection->NewNetbios) {
  823. Connection->LocalRcvSequenceMax = (USHORT)
  824. (Connection->ReceiveSequence + Connection->ReceiveWindowSize - 1);
  825. }
  826. NbiSendDataAck(
  827. Connection,
  828. NbiAckResend
  829. NB_LOCK_HANDLE_ARG(LockHandle));
  830. }
  831. NB_END_SYNC (&SyncContext);
  832. return STATUS_PENDING;
  833. } else {
  834. NB_DEBUG2 (RECEIVE, ("Receive %lx, connection %lx cancelled\n", Request, Connection));
  835. NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
  836. NB_END_SYNC (&SyncContext);
  837. NB_FREE_CANCEL_LOCK( CancelLH );
  838. return STATUS_CANCELLED;
  839. }
  840. } else {
  841. NB_DEBUG2 (RECEIVE, ("Receive connection %lx state is %d\n", Connection, Connection->State));
  842. NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
  843. NB_END_SYNC (&SyncContext);
  844. NB_FREE_CANCEL_LOCK( CancelLH );
  845. return STATUS_INVALID_CONNECTION;
  846. }
  847. } else {
  848. NB_DEBUG (RECEIVE, ("Receive connection %lx has bad signature\n", Connection));
  849. return STATUS_INVALID_CONNECTION;
  850. }
  851. } /* NbiTdiReceive */
  852. VOID
  853. NbiCancelReceive(
  854. IN PDEVICE_OBJECT DeviceObject,
  855. IN PIRP Irp
  856. )
  857. /*++
  858. Routine Description:
  859. This routine is called by the I/O system to cancel a receive.
  860. The request is found on the connection's receive queue.
  861. NOTE: This routine is called with the CancelSpinLock held and
  862. is responsible for releasing it.
  863. Arguments:
  864. DeviceObject - Pointer to the device object for this driver.
  865. Irp - Pointer to the request packet representing the I/O request.
  866. Return Value:
  867. none.
  868. --*/
  869. {
  870. PCONNECTION Connection;
  871. PREQUEST Request = (PREQUEST)Irp;
  872. NB_DEFINE_LOCK_HANDLE (LockHandle)
  873. NB_DEFINE_SYNC_CONTEXT (SyncContext)
  874. CTEAssert ((REQUEST_MAJOR_FUNCTION(Request) == IRP_MJ_INTERNAL_DEVICE_CONTROL) &&
  875. (REQUEST_MINOR_FUNCTION(Request) == TDI_RECEIVE));
  876. CTEAssert (REQUEST_OPEN_TYPE(Request) == (PVOID)TDI_CONNECTION_FILE);
  877. Connection = (PCONNECTION)REQUEST_OPEN_CONTEXT(Request);
  878. //
  879. // Just stop the connection, that will tear down any
  880. // receives.
  881. //
  882. // Do we care about cancelling non-active
  883. // receives without stopping the connection??
  884. //
  885. // This routine is the same as NbiCancelSend,
  886. // so if we don't make it more specific, merge the two.
  887. //
  888. NbiReferenceConnectionSync (Connection, CREF_CANCEL);
  889. IoReleaseCancelSpinLock (Irp->CancelIrql);
  890. NB_BEGIN_SYNC (&SyncContext);
  891. NB_SYNC_GET_LOCK (&Connection->Lock, &LockHandle);
  892. //
  893. // This frees the lock, cancels any sends, etc.
  894. //
  895. NbiStopConnection(
  896. Connection,
  897. STATUS_CANCELLED
  898. NB_LOCK_HANDLE_ARG (LockHandle));
  899. NbiDereferenceConnection (Connection, CREF_CANCEL);
  900. NB_END_SYNC (&SyncContext);
  901. } /* NbiCancelReceive */