/*************************************************************************** * * File: h245wsrx.c * * INTEL Corporation Proprietary Information * Copyright (c) 1996 Intel Corporation. * * This listing is supplied under the terms of a license agreement * with INTEL Corporation and may not be used, copied, nor disclosed * except in accordance with the terms of that agreement. * *************************************************************************** * * $Workfile: h245wsrx.cpp $ * $Revision: 2.4 $ * $Modtime: 30 Jan 1997 17:15:58 $ * $Log: S:/STURGEON/SRC/H245WS/VCS/h245wsrx.cpv $ * * Rev 2.4 30 Jan 1997 17:17:16 EHOWARDX * Fixed bug in trace message - need to do trace before * calling shutdown() sent shutdown clears error retrieved * by WSAGetLastError(). * * Rev 2.3 14 Jan 1997 15:48:04 EHOWARDX * Changed TryRecv() and TrySend() to check for WSAECONNRESET and * WSAECONNABORT return from recv() and send() and act accordingly. * * Rev 2.2 19 Dec 1996 18:54:54 SBELL1 * took out tag comments * * Rev 2.1 Dec 13 1996 17:31:00 plantz * moved #ifdef _cplusplus to after include files // // Rev 1.1 13 Dec 1996 12:11:34 SBELL1 // moved #ifdef _cplusplus to after include files // // Rev 1.0 11 Dec 1996 13:41:52 SBELL1 // Initial revision. * * Rev 1.19 08 Jul 1996 19:27:44 unknown * Second experiment to try to fix Q.931 shutdown problem. * * Rev 1.18 01 Jul 1996 16:45:12 EHOWARDX * * Moved Call to SocketCloseEvent from TryRecv() to ProcessQueuedRecvs(). * TryRecv() now returns LINK_RECV_CLOSED to trigger ProcessQueuedRecvs() * to call SocketCloseEvent(). * * Rev 1.17 May 28 1996 18:14:36 plantz * Change error codes to use HRESULT. Propogate Winsock errors where appropriate * * Rev 1.16 17 May 1996 16:49:32 EHOWARDX * Shutdown fix. * * Rev 1.15 09 May 1996 18:33:16 EHOWARDX * * Changes to build with new LINKAPI.H. * * Rev 1.14 29 Apr 1996 16:53:16 EHOWARDX * * Added trace statement. * * Rev 1.13 Apr 29 1996 14:04:20 plantz * Call NotifyRead instead of ProcessQueuedRecvs. * * Rev 1.12 Apr 29 1996 12:14:06 plantz * Change tpkt header to include header size in packet length. * Assert that message length does not exceed INT_MAX. * . * * Rev 1.11 27 Apr 1996 14:07:32 EHOWARDX * Parenthesized return from TryRecv(). * * Rev 1.10 Apr 25 1996 21:15:12 plantz * Check state of connection before attemting to call recv. * * Rev 1.9 Apr 24 1996 16:39:34 plantz * Merge 1.5.1.0 with 1.8 (changes for winsock 1) * * Rev 1.5.1.0 Apr 24 1996 16:23:00 plantz * Change to not use overlapped I/O (for winsock 1). * * Rev 1.5 01 Apr 1996 14:20:12 unknown * Shutdown redesign. * * Rev 1.4 19 Mar 1996 20:18:16 EHOWARDX * * Redesigned shutdown. * * Rev 1.3 18 Mar 1996 19:08:32 EHOWARDX * Fixed shutdown; eliminated TPKT/WSCB dependencies. * Define TPKT to put TPKT/WSCB dependencies back in. * * Rev 1.2 14 Mar 1996 17:01:58 EHOWARDX * * NT4.0 testing; got rid of HwsAssert(); got rid of TPKT/WSCB. * * Rev 1.1 09 Mar 1996 21:12:02 EHOWARDX * Fixes as result of testing. * * Rev 1.0 08 Mar 1996 20:20:18 unknown * Initial revision. * ***************************************************************************/ #define LINKDLL_EXPORT #pragma warning ( disable : 4115 4201 4214 4514 ) #undef _WIN32_WINNT // override bogus platform definition in our common build environment #include "precomp.h" #include //#include #include "queue.h" #include "linkapi.h" #include "h245ws.h" #include "tstable.h" #if defined(__cplusplus) extern "C" { #endif // (__cplusplus) // If we are not using the Unicode version of the ISR display utility, then redefine // the __TEXT macro to do nothing. #ifndef UNICODE_TRACE #undef __TEXT #define __TEXT(x) x #endif extern TSTable* gpInstanceTable; // global ptr to the instance table #define GetTpktLength(pReq) (((pReq)->req_TpktHeader[2] << 8) + (pReq)->req_TpktHeader[3]) /*++ Description: Start a receive Arguments: pHws - Pointer to context for "connection" pReq - Pointer to I/O request structure Return Value: SUCCESS - Successfully started receive. LINK_RECV_ERROR_WOULD_BLOCK - Winsock error --*/ static HRESULT TryRecv(IN PHWSINST pHws, IN char *data, IN int length, IN OUT int *total_bytes_done) { int requested_length = length - *total_bytes_done; int recv_result = recv(pHws->hws_Socket, data+*total_bytes_done, requested_length, 0); if (recv_result == SOCKET_ERROR) { int err = WSAGetLastError(); switch (err) { case WSAEWOULDBLOCK: return LINK_RECV_WOULD_BLOCK; case WSAECONNABORTED: case WSAECONNRESET: HWSTRACE1(pHws->hws_dwPhysicalId, HWS_WARNING, __TEXT("TryRecv: recv() returned %s"), SocketErrorText()); if (pHws->hws_uState == HWS_CONNECTED) { HWSTRACE0(pHws->hws_dwPhysicalId, HWS_TRACE, __TEXT("TryRecv: calling shutdown")); if (shutdown(pHws->hws_Socket, 1) == SOCKET_ERROR) { HWSTRACE1(pHws->hws_dwPhysicalId, HWS_WARNING, __TEXT("TryRecv: shutdown() returned %s"), SocketErrorText()); } pHws->hws_uState = HWS_CLOSING; } return MAKE_WINSOCK_ERROR(err); default: HWSTRACE1(pHws->hws_dwPhysicalId, HWS_WARNING, __TEXT("TryRecv: recv() returned %s"), SocketErrorText()); return MAKE_WINSOCK_ERROR(err); } // switch } HWSTRACE1(pHws->hws_dwPhysicalId, HWS_TRACE, __TEXT("TryRecv: recv returned %d"), recv_result); if (recv_result == 0) { return LINK_RECV_CLOSED; } *total_bytes_done += recv_result; return (recv_result == requested_length) ? NOERROR : LINK_RECV_WOULD_BLOCK; } static HRESULT RecvStart(IN PHWSINST pHws, IN PREQUEST pReq) { HRESULT nResult = NOERROR; // Sanity checks HWSASSERT(pHws != NULL); HWSASSERT(pHws->hws_dwMagic == HWSINST_MAGIC); HWSASSERT(pReq != NULL); HWSASSERT(pReq->req_dwMagic == RECV_REQUEST_MAGIC); HWSASSERT(pReq->req_pHws == pHws); // Get the header first; if that succeeds get the client data if (pReq->req_header_bytes_done < TPKT_HEADER_SIZE) { nResult = TryRecv(pHws, (char *)pReq->req_TpktHeader, TPKT_HEADER_SIZE, &pReq->req_header_bytes_done); } if (nResult == NOERROR) { long int tpkt_length = GetTpktLength(pReq) - TPKT_HEADER_SIZE; if (pReq->req_TpktHeader[0] != TPKT_VERSION || tpkt_length <= 0) { // Invalid header version HWSTRACE0(pHws->hws_dwPhysicalId, HWS_CRITICAL, __TEXT("RecvComplete: bad header version; available data discarded")); // Should this be reported to the client?? // Read and discard all available data // The client's buffer is used as a temporary buffer. while (recv(pHws->hws_Socket, (char *)pReq->req_client_data, pReq->req_client_length, 0) > 0) ; // Mark the header for this request as unread; it // will be read again when additional data is received. pReq->req_header_bytes_done = 0; nResult = LINK_RECV_ERROR; } else if (tpkt_length > pReq->req_client_length) { // Packet too large int request_length; int result; HWSTRACE0(pHws->hws_dwPhysicalId, HWS_CRITICAL, __TEXT("RecvComplete: packet too large; packet discarded")); // Should this be reported to the client?? // Read and discard the packet // The client's buffer is used as a temporary buffer. do { request_length = pReq->req_client_length; if (request_length > tpkt_length) request_length = tpkt_length; result = recv(pHws->hws_Socket, (char *)pReq->req_client_data, request_length, 0); } while (result > 0 && (tpkt_length -= result) > 0); if (result == SOCKET_ERROR && WSAGetLastError() == WSAEWOULDBLOCK) { //TODO: packet too large handling // Adjust the header so that the rest of this packet will be read, but // flag it so that it is known to be an error and will not be returned // to the client. } else { // Mark the header for this request as unread; it // will be read again for the next packet received. pReq->req_header_bytes_done = 0; } nResult = LINK_RECV_ERROR; } else { // Normal case // The current implementation of TryRecv requires that the requested // size fit in a signed int (because that is what Winsock supports // in a single recv). This is guaranteed at this point regardless // of the originator of the packets, because we don't allow a buffer // to be posted that is larger than that (see ASSERT below). If the // packet were larger than the buffer, it would have been caught above. // If TryRecv is changed to remove the restriction on buffer size and // accept a parameter of type long int, this ASSERT may be removed. HWSASSERT(tpkt_length <= INT_MAX); nResult = TryRecv(pHws, (char *)pReq->req_client_data, (int)tpkt_length, &pReq->req_client_bytes_done); } } return nResult; } // RecvStart() void ProcessQueuedRecvs(IN PHWSINST pHws) { register PREQUEST pReq; register DWORD dwPhysicalId = pHws->hws_dwPhysicalId; HWSASSERT(pHws != NULL); HWSASSERT(pHws->hws_dwMagic == HWSINST_MAGIC); HWSASSERT(pHws->hws_uState <= HWS_CLOSING); HWSTRACE0(pHws->hws_dwPhysicalId, HWS_TRACE, __TEXT("ProcessQueuedRecvs")); while ((pReq = (PREQUEST) QRemove(pHws->hws_pRecvQueue)) != NULL) { switch (RecvStart(pHws, pReq)) { case NOERROR: // Call Recv callback pHws->hws_h245RecvCallback(pHws->hws_dwH245Instance, LINK_RECV_DATA, pReq->req_client_data, pReq->req_client_bytes_done); // Free the I/O request structure MemFree(pReq); // Check to see if callback deallocated our instance or state changed // Check to see if callback deallocated our instance - this can be done // by attempting a lock - which will now fail if the entry has been marked // for deletion. Thus, if the lock succeeds, then just unlock it (since we // already have a lock on it in a higher level function). if(gpInstanceTable->Lock(dwPhysicalId) == NULL) return; gpInstanceTable->Unlock(dwPhysicalId); if(pHws->hws_uState > HWS_CONNECTED) return; break; default: HWSTRACE0(pHws->hws_dwPhysicalId, HWS_WARNING, __TEXT("ProcessQueuedRecvs: RecvStart() failed")); // Fall-through to next case is intentional case LINK_RECV_WOULD_BLOCK: // The receive would have blocked; we need to requeue the I/O request // and wait for a FD_READ network event. // If any part of the data was received, the bytes_done field has been updated. if (QInsertAtHead(pHws->hws_pRecvQueue, pReq) == FALSE) { HWSTRACE0(pHws->hws_dwPhysicalId, HWS_CRITICAL, __TEXT("ProcessQueuedRecvs: QInsertAtHead() failed")); } return; case LINK_RECV_CLOSED: if (QInsertAtHead(pHws->hws_pRecvQueue, pReq) == FALSE) { HWSTRACE0(pHws->hws_dwPhysicalId, HWS_CRITICAL, __TEXT("ProcessQueuedRecvs: QInsertAtHead() failed")); } SocketCloseEvent(pHws); return; } // switch } // while } // ProcessQueuedRecvs() /************************************************************************** ** Function : datalinkReceiveRequest ** Description : Fills header/tail of buffer and posts buffer to H.223 ***************************************************************************/ LINKDLL HRESULT datalinkReceiveRequest( DWORD dwPhysicalId, PBYTE pbyDataBuf, DWORD dwLength) { register PHWSINST pHws; register PREQUEST pReq; HWSTRACE0(dwPhysicalId, HWS_TRACE, __TEXT("datalinkReceiveRequest")); pHws = gpInstanceTable->Lock(dwPhysicalId); if (pHws == NULL) { HWSTRACE0(dwPhysicalId, HWS_ERROR, __TEXT("datalinkReceiveRequest: dwPhysicalId not found")); return LINK_INVALID_INSTANCE; } if (pHws->hws_uState > HWS_CONNECTED) { HWSTRACE1(dwPhysicalId, HWS_ERROR, __TEXT("datalinkReceiveRequest: state = %d"), pHws->hws_uState); gpInstanceTable->Unlock(dwPhysicalId); return LINK_INVALID_STATE; } // Allocate request structure pReq = (PREQUEST) MemAlloc(sizeof(*pReq)); if (pReq == NULL) { HWSTRACE0(dwPhysicalId, HWS_WARNING, __TEXT("datalinkReceiveRequest: could not allocate request buffer")); gpInstanceTable->Unlock(dwPhysicalId); return LINK_MEM_FAILURE; } // The current implementation requires that the size of each message // fit in a signed int (because that is what Winsock supports in a // single recv). If it is necessary to receive larger messages, // TryRecv and RecvStart must be changed to limit the size in each // recv call, and loop until all the data is received. // This ASSERT could then be removed. HWSASSERT(dwLength <= INT_MAX); pReq->req_pHws = pHws; pReq->req_header_bytes_done= 0; pReq->req_client_data = pbyDataBuf; pReq->req_client_length = (int)dwLength; pReq->req_client_bytes_done= 0; pReq->req_dwMagic = RECV_REQUEST_MAGIC; if (QInsert(pHws->hws_pRecvQueue, pReq) == FALSE) { HWSTRACE0(pHws->hws_dwPhysicalId, HWS_CRITICAL, __TEXT("datalinkReceiveRequest: QInsert() failed")); gpInstanceTable->Unlock(dwPhysicalId); return LINK_RECV_NOBUFF; } if (pHws->hws_uState == HWS_CONNECTED) NotifyRead(pHws); HWSTRACE0(dwPhysicalId, HWS_TRACE, __TEXT("datalinkReceiveRequest: succeeded")); gpInstanceTable->Unlock(dwPhysicalId); return NOERROR; } // datalinkReceiveRequest() #if defined(__cplusplus) } #endif // (__cplusplus)