📄 porttcp.c
字号:
{ vMBPortLog( MB_LOG_DEBUG, _T( "TCP-POLL" ), _T( "FD_READ event\r\n" ) ); } /* Process part of the Modbus TCP frame. In case of an I/O error we have to drop * the client connection. */ if( prvMBTCPGetFrame( ) != TRUE ) { prvvMBPortReleaseClient( ); } } else if( xNetworkEvents.lNetworkEvents & FD_CLOSE ) { if( MB_TCP_DEBUG ) { vMBPortLog( MB_LOG_DEBUG, _T( "TCP-POLL" ), _T( "FD_CLOSE event\r\n" ) ); } prvvMBPortReleaseClient( ); } else { vMBPortLog( MB_LOG_WARN, _T( "TCP-POLL" ), _T( "unknown EV_CLIENT event\r\n" ) ); } } else { /* Error - Log a warning. */ } } return bOkay;}/*! * \ingroup port_win32tcp * \brief Receives parts of a Modbus TCP frame and if complete notifies * the protocol stack. * \internal * * This function reads a complete Modbus TCP frame from the protocol stack. * It starts by reading the header with an initial request size for * usTCPFrameBytesLeft = MB_TCP_FUNC. If the header is complete the * number of bytes left can be calculated from it (See Length in MBAP header). * Further read calls are issued until the frame is complete. * * \return \c TRUE if part of a Modbus TCP frame could be processed. In case * of a communication error the function returns \c FALSE. */BOOLprvMBTCPGetFrame( ){ BOOL bOkay = TRUE; USHORT usLength; int iRes; LPTSTR szFrameAsStr; /* Make sure that we can safely process the next read request. If there * is an overflow drop the client. */ if( ( usTCPBufPos + usTCPFrameBytesLeft ) >= MB_TCP_BUF_SIZE ) { vMBPortLog( MB_LOG_WARN, _T( "MBTCP-RCV" ), _T( "Detected buffer overrun. Dropping client.\r\n" ) ); return FALSE; } iRes = recv( xClientSocket, &aucTCPBuf[usTCPBufPos], usTCPFrameBytesLeft, 0 ); switch ( iRes ) { case SOCKET_ERROR: vMBPortLog( MB_LOG_WARN, _T( "MBTCP-RCV" ), _T( "recv failed: %s\r\n" ), WsaError2String( WSAGetLastError( ) ) ); if( WSAGetLastError( ) != WSAEWOULDBLOCK ) { bOkay = FALSE; } break; case 0: bOkay = FALSE; break; default: usTCPBufPos += iRes; usTCPFrameBytesLeft -= iRes; } /* If we have received the MBAP header we can analyze it and calculate * the number of bytes left to complete the current request. If complete * notify the protocol stack. */ if( usTCPBufPos >= MB_TCP_FUNC ) { /* Length is a byte count of Modbus PDU (function code + data) and the * unit identifier. */ usLength = aucTCPBuf[MB_TCP_LEN] << 8U; usLength |= aucTCPBuf[MB_TCP_LEN + 1]; /* Is the frame already complete. */ if( usTCPBufPos < ( MB_TCP_UID + usLength ) ) { usTCPFrameBytesLeft = usLength + MB_TCP_UID - usTCPBufPos; } /* The frame is complete. */ else if( usTCPBufPos == ( MB_TCP_UID + usLength ) ) { if( MB_TCP_DEBUG ) { szFrameAsStr = prvMBTCPPortFrameToString( aucTCPBuf, usTCPBufPos ); if( szFrameAsStr != NULL ) { vMBPortLog( MB_LOG_DEBUG, _T( "MBTCP-RCV" ), _T( "Received: %s\r\n" ), szFrameAsStr ); free( szFrameAsStr ); } } ( void )xMBPortEventPost( EV_FRAME_RECEIVED ); } /* This can not happend because we always calculate the number of bytes * to receive. */ else { assert( usTCPBufPos <= ( MB_TCP_UID + usLength ) ); } } return bOkay;}BOOLxMBTCPPortGetRequest( UCHAR ** ppucMBTCPFrame, USHORT * usTCPLength ){ *ppucMBTCPFrame = &aucTCPBuf[0]; *usTCPLength = usTCPBufPos; /* Reset the buffer. */ usTCPBufPos = 0; usTCPFrameBytesLeft = MB_TCP_FUNC; return TRUE;}BOOLxMBTCPPortSendResponse( const UCHAR * pucMBTCPFrame, USHORT usTCPLength ){ BOOL bFrameSent = FALSE; BOOL bAbort = FALSE; int res; int iBytesSent = 0; int iTimeOut = MB_TCP_READ_TIMEOUT; LPTSTR szFrameAsStr; if( MB_TCP_DEBUG ) { szFrameAsStr = prvMBTCPPortFrameToString( aucTCPBuf, usTCPLength ); if( szFrameAsStr != NULL ) { vMBPortLog( MB_LOG_DEBUG, _T( "MBTCP-SND" ), _T( "Snd: %s\r\n" ), szFrameAsStr ); free( szFrameAsStr ); } } do { res = send( xClientSocket, &pucMBTCPFrame[iBytesSent], usTCPLength - iBytesSent, 0 ); switch ( res ) { case SOCKET_ERROR: if( ( WSAGetLastError( ) == WSAEWOULDBLOCK ) && ( iTimeOut > 0 ) ) { iTimeOut -= MB_TCP_READ_CYCLE; Sleep( MB_TCP_READ_CYCLE ); } else { bAbort = TRUE; } break; case 0: prvvMBPortReleaseClient( ); bAbort = TRUE; break; default: iBytesSent += res; break; } } while( ( iBytesSent != usTCPLength ) && !bAbort ); bFrameSent = iBytesSent == usTCPLength ? TRUE : FALSE; return bFrameSent;}voidprvvMBPortReleaseClient( ){ TCHAR szIPAddr[32]; if( prvMBTCPPortAddressToString( xClientSocket, szIPAddr, _countof( szIPAddr ) ) == TRUE ) { vMBPortLog( MB_LOG_INFO, _T( "MBTCP-CMGT" ), _T( "client %s disconnected.\r\n" ), szIPAddr ); } else { vMBPortLog( MB_LOG_INFO, _T( "MBTCP-CMGT" ), _T( "unknown client disconnected.\r\n" ) ); } /* Disable event notification for this client socket. */ if( WSAEventSelect( xClientSocket, xEvents[EV_CLIENT], 0 ) == SOCKET_ERROR ) { vMBPortLog( MB_LOG_ERROR, _T( "MBTCP-CMGT" ), _T( "can't disable events for disconnecting client socket: %s\r\n" ), WsaError2String( WSAGetLastError( ) ) ); } /* Reset event object in case an event was still pending. */ if( WSAResetEvent( xEvents[EV_CLIENT] ) == SOCKET_ERROR ) { vMBPortLog( MB_LOG_ERROR, _T( "MBTCP-CMGT" ), _T( "can't disable events for disconnecting client socket: %s\r\n" ), WsaError2String( WSAGetLastError( ) ) ); } /* Disallow the sender side. This tells the other side that we have finished. */ if( shutdown( xClientSocket, SD_SEND ) == SOCKET_ERROR ) { vMBPortLog( MB_LOG_ERROR, _T( "MBTCP-CMGT" ), _T( "shutdown failed: %s\r\n" ), WsaError2String( WSAGetLastError( ) ) ); } /* Read any unread data from the socket. Note that this is not the strictly * correct way to do it because our sockets are non blocking and therefore * some bytes could remain. */ ( void )recv( xClientSocket, &aucTCPBuf[0], MB_TCP_BUF_SIZE, 0 ); ( void )closesocket( xClientSocket ); xClientSocket = INVALID_SOCKET;}BOOLprvbMBPortAcceptClient( ){ SOCKET xNewSocket; BOOL bOkay; TCHAR szIPAddr[32]; /* Check if we can handle a new connection. */ if( xClientSocket != INVALID_SOCKET ) { vMBPortLog( MB_LOG_ERROR, _T( "MBTCP-CMGT" ), _T( "can't accept new client. all connections in use.\r\n" ) ); bOkay = FALSE; } else if( ( xNewSocket = accept( xListenSocket, NULL, NULL ) ) == INVALID_SOCKET ) { bOkay = FALSE; } /* Register READ events on the socket file descriptor. */ else if( WSAEventSelect( xNewSocket, xEvents[EV_CLIENT], FD_READ | FD_CLOSE ) == SOCKET_ERROR ) { bOkay = FALSE; ( void )closesocket( xNewSocket ); } /* Everything okay - Register the client connection. */ else { xClientSocket = xNewSocket; usTCPBufPos = 0; usTCPFrameBytesLeft = MB_TCP_FUNC; if( prvMBTCPPortAddressToString( xClientSocket, szIPAddr, _countof( szIPAddr ) ) == TRUE ) { vMBPortLog( MB_LOG_INFO, _T( "MBTCP-CMGT" ), _T( "accepted new client %s.\r\n" ), szIPAddr ); } else { vMBPortLog( MB_LOG_INFO, _T( "MBTCP-CMGT" ), _T( "accepted unknown client.\r\n" ) ); } bOkay = TRUE; } return bOkay;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -