2021-07-13 21:02:53 +02:00
|
|
|
#include "fsfw/platform.h"
|
2021-09-28 15:01:01 +02:00
|
|
|
#include "fsfw/FSFW.h"
|
|
|
|
|
|
|
|
#include "TcpTmTcServer.h"
|
|
|
|
#include "TcpTmTcBridge.h"
|
|
|
|
#include "tcpipHelpers.h"
|
|
|
|
|
2021-09-28 15:09:56 +02:00
|
|
|
#include "fsfw/tmtcservices/SpacePacketParser.h"
|
2021-09-28 15:01:01 +02:00
|
|
|
#include "fsfw/tasks/TaskFactory.h"
|
2021-09-28 15:09:56 +02:00
|
|
|
#include "fsfw/globalfunctions/arrayprinter.h"
|
2021-07-13 21:02:53 +02:00
|
|
|
#include "fsfw/container/SharedRingBuffer.h"
|
|
|
|
#include "fsfw/ipc/MessageQueueSenderIF.h"
|
|
|
|
#include "fsfw/ipc/MutexGuard.h"
|
|
|
|
#include "fsfw/objectmanager/ObjectManager.h"
|
|
|
|
#include "fsfw/serviceinterface/ServiceInterface.h"
|
|
|
|
#include "fsfw/tmtcservices/TmTcMessage.h"
|
2021-03-11 14:47:47 +01:00
|
|
|
|
2021-05-12 16:47:53 +02:00
|
|
|
#ifdef PLATFORM_WIN
|
2021-03-11 14:47:47 +01:00
|
|
|
#include <winsock2.h>
|
|
|
|
#include <ws2tcpip.h>
|
2021-05-12 16:47:53 +02:00
|
|
|
#elif defined(PLATFORM_UNIX)
|
2021-03-21 12:51:28 +01:00
|
|
|
#include <netdb.h>
|
2021-03-21 00:30:33 +01:00
|
|
|
#endif
|
2021-05-05 12:59:42 +02:00
|
|
|
|
2021-08-09 18:12:25 +02:00
|
|
|
const std::string TcpTmTcServer::DEFAULT_SERVER_PORT = tcpip::DEFAULT_SERVER_PORT;
|
2021-03-11 14:47:47 +01:00
|
|
|
|
2021-05-05 12:59:42 +02:00
|
|
|
TcpTmTcServer::TcpTmTcServer(object_id_t objectId, object_id_t tmtcTcpBridge,
|
2021-09-28 15:01:01 +02:00
|
|
|
size_t receptionBufferSize, size_t ringBufferSize, std::string customTcpServerPort,
|
|
|
|
ReceptionModes receptionMode):
|
|
|
|
SystemObject(objectId), tmtcBridgeId(tmtcTcpBridge), receptionMode(receptionMode),
|
|
|
|
tcpConfig(customTcpServerPort), receptionBuffer(receptionBufferSize),
|
|
|
|
ringBuffer(ringBufferSize, true), validPacketIds() {
|
2021-03-11 14:47:47 +01:00
|
|
|
}
|
|
|
|
|
2021-03-21 00:30:33 +01:00
|
|
|
ReturnValue_t TcpTmTcServer::initialize() {
|
2021-03-12 02:15:21 +01:00
|
|
|
using namespace tcpip;
|
2021-03-21 00:30:33 +01:00
|
|
|
|
|
|
|
ReturnValue_t result = TcpIpBase::initialize();
|
|
|
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-09-28 15:01:01 +02:00
|
|
|
switch(receptionMode) {
|
|
|
|
case(ReceptionModes::SPACE_PACKETS): {
|
|
|
|
spacePacketParser = new SpacePacketParser(validPacketIds);
|
|
|
|
if(spacePacketParser == nullptr) {
|
|
|
|
return HasReturnvaluesIF::RETURN_FAILED;
|
|
|
|
}
|
|
|
|
tcpConfig.tcpFlags |= MSG_DONTWAIT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-05 19:52:38 +02:00
|
|
|
tcStore = ObjectManager::instance()->get<StorageManagerIF>(objects::TC_STORE);
|
2021-05-05 12:59:42 +02:00
|
|
|
if (tcStore == nullptr) {
|
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
|
|
|
sif::error << "TcpTmTcServer::initialize: TC store uninitialized!" << std::endl;
|
|
|
|
#else
|
|
|
|
sif::printError("TcpTmTcServer::initialize: TC store uninitialized!\n");
|
|
|
|
#endif
|
|
|
|
return ObjectManagerIF::CHILD_INIT_FAILED;
|
|
|
|
}
|
|
|
|
|
2021-06-05 19:52:38 +02:00
|
|
|
tmtcBridge = ObjectManager::instance()->get<TcpTmTcBridge>(tmtcBridgeId);
|
2021-05-05 12:59:42 +02:00
|
|
|
|
2021-03-11 14:47:47 +01:00
|
|
|
int retval = 0;
|
|
|
|
struct addrinfo *addrResult = nullptr;
|
2021-03-23 11:35:23 +01:00
|
|
|
struct addrinfo hints = {};
|
2021-03-11 14:47:47 +01:00
|
|
|
|
|
|
|
hints.ai_family = AF_INET;
|
|
|
|
hints.ai_socktype = SOCK_STREAM;
|
|
|
|
hints.ai_protocol = IPPROTO_TCP;
|
|
|
|
hints.ai_flags = AI_PASSIVE;
|
|
|
|
|
2021-05-05 15:59:41 +02:00
|
|
|
// Listen to all addresses (0.0.0.0) by using AI_PASSIVE in the hint flags
|
2021-09-28 15:01:01 +02:00
|
|
|
retval = getaddrinfo(nullptr, tcpConfig.tcpPort.c_str(), &hints, &addrResult);
|
2021-03-11 14:47:47 +01:00
|
|
|
if (retval != 0) {
|
2021-03-12 02:15:21 +01:00
|
|
|
handleError(Protocol::TCP, ErrorSources::GETADDRINFO_CALL);
|
2021-03-11 14:47:47 +01:00
|
|
|
return HasReturnvaluesIF::RETURN_FAILED;
|
|
|
|
}
|
|
|
|
|
2021-05-05 15:59:41 +02:00
|
|
|
// Open TCP (stream) socket
|
2021-03-11 14:47:47 +01:00
|
|
|
listenerTcpSocket = socket(addrResult->ai_family, addrResult->ai_socktype,
|
|
|
|
addrResult->ai_protocol);
|
|
|
|
if(listenerTcpSocket == INVALID_SOCKET) {
|
|
|
|
freeaddrinfo(addrResult);
|
2021-03-12 02:15:21 +01:00
|
|
|
handleError(Protocol::TCP, ErrorSources::SOCKET_CALL);
|
2021-03-11 14:47:47 +01:00
|
|
|
return HasReturnvaluesIF::RETURN_FAILED;
|
|
|
|
}
|
|
|
|
|
2021-05-05 15:59:41 +02:00
|
|
|
// Bind to the address found by getaddrinfo
|
2021-03-12 00:34:30 +01:00
|
|
|
retval = bind(listenerTcpSocket, addrResult->ai_addr, static_cast<int>(addrResult->ai_addrlen));
|
2021-03-11 14:47:47 +01:00
|
|
|
if(retval == SOCKET_ERROR) {
|
|
|
|
freeaddrinfo(addrResult);
|
2021-03-12 02:15:21 +01:00
|
|
|
handleError(Protocol::TCP, ErrorSources::BIND_CALL);
|
2021-04-08 18:48:44 +02:00
|
|
|
return HasReturnvaluesIF::RETURN_FAILED;
|
2021-03-11 14:47:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
freeaddrinfo(addrResult);
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-21 00:30:33 +01:00
|
|
|
TcpTmTcServer::~TcpTmTcServer() {
|
2021-03-21 12:51:28 +01:00
|
|
|
closeSocket(listenerTcpSocket);
|
2021-03-11 14:47:47 +01:00
|
|
|
}
|
|
|
|
|
2021-03-21 00:30:33 +01:00
|
|
|
ReturnValue_t TcpTmTcServer::performOperation(uint8_t opCode) {
|
2021-03-12 02:15:21 +01:00
|
|
|
using namespace tcpip;
|
2021-05-05 15:59:41 +02:00
|
|
|
// If a connection is accepted, the corresponding socket will be assigned to the new socket
|
2021-05-05 12:59:42 +02:00
|
|
|
socket_t connSocket = 0;
|
2021-05-13 22:17:21 +02:00
|
|
|
// sockaddr clientSockAddr = {};
|
|
|
|
// socklen_t connectorSockAddrLen = 0;
|
2021-03-11 14:47:47 +01:00
|
|
|
int retval = 0;
|
2021-03-12 02:15:21 +01:00
|
|
|
|
2021-05-05 15:59:41 +02:00
|
|
|
// Listen for connection requests permanently for lifetime of program
|
2021-03-11 14:47:47 +01:00
|
|
|
while(true) {
|
2021-09-28 15:01:01 +02:00
|
|
|
retval = listen(listenerTcpSocket, tcpConfig.tcpBacklog);
|
2021-03-11 14:47:47 +01:00
|
|
|
if(retval == SOCKET_ERROR) {
|
2021-03-12 02:15:21 +01:00
|
|
|
handleError(Protocol::TCP, ErrorSources::LISTEN_CALL, 500);
|
2021-03-11 14:47:47 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-05-13 22:17:21 +02:00
|
|
|
//connSocket = accept(listenerTcpSocket, &clientSockAddr, &connectorSockAddrLen);
|
|
|
|
connSocket = accept(listenerTcpSocket, nullptr, nullptr);
|
2021-03-11 14:47:47 +01:00
|
|
|
|
2021-05-05 12:59:42 +02:00
|
|
|
if(connSocket == INVALID_SOCKET) {
|
2021-03-12 02:15:21 +01:00
|
|
|
handleError(Protocol::TCP, ErrorSources::ACCEPT_CALL, 500);
|
2021-05-05 12:59:42 +02:00
|
|
|
closeSocket(connSocket);
|
2021-03-11 14:47:47 +01:00
|
|
|
continue;
|
|
|
|
};
|
|
|
|
|
2021-05-05 12:59:42 +02:00
|
|
|
handleServerOperation(connSocket);
|
|
|
|
|
2021-05-05 15:59:41 +02:00
|
|
|
// Done, shut down connection and go back to listening for client requests
|
2021-09-28 15:01:01 +02:00
|
|
|
retval = shutdown(connSocket, SHUT_BOTH);
|
2021-05-05 12:59:42 +02:00
|
|
|
if(retval != 0) {
|
|
|
|
handleError(Protocol::TCP, ErrorSources::SHUTDOWN_CALL);
|
2021-03-11 14:47:47 +01:00
|
|
|
}
|
2021-05-05 12:59:42 +02:00
|
|
|
closeSocket(connSocket);
|
2021-09-28 15:01:01 +02:00
|
|
|
connSocket = 0;
|
2021-05-05 12:59:42 +02:00
|
|
|
}
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
|
|
|
}
|
2021-03-11 14:47:47 +01:00
|
|
|
|
2021-05-05 12:59:42 +02:00
|
|
|
ReturnValue_t TcpTmTcServer::initializeAfterTaskCreation() {
|
2021-05-05 15:59:41 +02:00
|
|
|
if(tmtcBridge == nullptr) {
|
|
|
|
return ObjectManagerIF::CHILD_INIT_FAILED;
|
|
|
|
}
|
2021-05-05 12:59:42 +02:00
|
|
|
/* Initialize the destination after task creation. This ensures
|
|
|
|
that the destination has already been set in the TMTC bridge. */
|
|
|
|
targetTcDestination = tmtcBridge->getRequestQueue();
|
2021-05-05 15:59:41 +02:00
|
|
|
tcStore = tmtcBridge->tcStore;
|
|
|
|
tmStore = tmtcBridge->tmStore;
|
2021-05-05 12:59:42 +02:00
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
|
|
|
}
|
|
|
|
|
2021-09-28 15:01:01 +02:00
|
|
|
void TcpTmTcServer::handleServerOperation(socket_t& connSocket) {
|
|
|
|
while (true) {
|
|
|
|
int retval = recv(
|
|
|
|
connSocket,
|
2021-05-05 12:59:42 +02:00
|
|
|
reinterpret_cast<char*>(receptionBuffer.data()),
|
|
|
|
receptionBuffer.capacity(),
|
2021-09-28 15:01:01 +02:00
|
|
|
tcpConfig.tcpFlags
|
|
|
|
);
|
|
|
|
if(retval == 0) {
|
|
|
|
// Client closed connection
|
|
|
|
return;
|
2021-05-05 12:59:42 +02:00
|
|
|
}
|
2021-09-28 15:01:01 +02:00
|
|
|
else if(retval > 0) {
|
|
|
|
// The ring buffer was configured for overwrite, so the returnvalue does not need to
|
|
|
|
// be checked for now
|
|
|
|
ringBuffer.writeData(receptionBuffer.data(), retval);
|
2021-03-11 14:47:47 +01:00
|
|
|
}
|
2021-09-28 15:01:01 +02:00
|
|
|
else if(retval < 0) {
|
|
|
|
if(errno == EAGAIN) {
|
|
|
|
// No data available. Check whether any packets have been read, then send back
|
|
|
|
// telemetry if available
|
|
|
|
bool tcAvailable = false;
|
|
|
|
bool tmSent = false;
|
|
|
|
size_t availableReadData = ringBuffer.getAvailableReadData();
|
|
|
|
if(availableReadData > lastRingBufferSize) {
|
|
|
|
tcAvailable = true;
|
|
|
|
handleTcRingBufferData(availableReadData);
|
|
|
|
}
|
|
|
|
ReturnValue_t result = handleTmSending(connSocket, tmSent);
|
|
|
|
if(result == CONN_BROKEN) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(not tcAvailable and not tmSent) {
|
|
|
|
TaskFactory::delayTask(DEFAULT_LOOP_DELAY_MS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
tcpip::handleError(tcpip::Protocol::TCP, tcpip::ErrorSources::RECV_CALL);
|
|
|
|
}
|
2021-03-11 14:47:47 +01:00
|
|
|
}
|
2021-09-28 15:01:01 +02:00
|
|
|
}
|
2021-05-05 12:59:42 +02:00
|
|
|
}
|
2021-03-11 14:47:47 +01:00
|
|
|
|
2021-09-28 15:01:01 +02:00
|
|
|
ReturnValue_t TcpTmTcServer::handleTcReception(uint8_t* spacePacket, size_t packetSize) {
|
2021-09-28 15:09:56 +02:00
|
|
|
if(wiretappingEnabled) {
|
|
|
|
sif::info << "Received TC:" << std::endl;
|
|
|
|
arrayprinter::print(spacePacket, packetSize);
|
|
|
|
}
|
|
|
|
|
2021-09-28 15:01:01 +02:00
|
|
|
if(spacePacket == nullptr or packetSize == 0) {
|
|
|
|
return HasReturnvaluesIF::RETURN_FAILED;
|
|
|
|
}
|
2021-05-05 12:59:42 +02:00
|
|
|
store_address_t storeId;
|
2021-09-28 15:01:01 +02:00
|
|
|
ReturnValue_t result = tcStore->addData(&storeId, spacePacket, packetSize);
|
2021-05-05 12:59:42 +02:00
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
#if FSFW_VERBOSE_LEVEL >= 1
|
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
2021-09-28 15:01:01 +02:00
|
|
|
sif::warning << "TcpTmTcServer::handleServerOperation: Data storage with packet size" <<
|
|
|
|
packetSize << " failed" << std::endl;
|
|
|
|
#else
|
|
|
|
sif::printWarning("TcpTmTcServer::handleServerOperation: Data storage with packet size %d "
|
|
|
|
"failed\n", packetSize);
|
2021-05-05 12:59:42 +02:00
|
|
|
#endif /* FSFW_CPP_OSTREAM_ENABLED == 1 */
|
|
|
|
#endif /* FSFW_VERBOSE_LEVEL >= 1 */
|
2021-09-28 15:01:01 +02:00
|
|
|
return result;
|
2021-03-11 14:47:47 +01:00
|
|
|
}
|
|
|
|
|
2021-05-05 12:59:42 +02:00
|
|
|
TmTcMessage message(storeId);
|
2021-03-11 14:47:47 +01:00
|
|
|
|
2021-09-28 15:01:01 +02:00
|
|
|
result = MessageQueueSenderIF::sendMessage(targetTcDestination, &message);
|
2021-05-05 12:59:42 +02:00
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
#if FSFW_VERBOSE_LEVEL >= 1
|
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
2021-09-28 15:01:01 +02:00
|
|
|
sif::warning << "TcpTmTcServer::handleServerOperation: "
|
2021-05-05 12:59:42 +02:00
|
|
|
" Sending message to queue failed" << std::endl;
|
2021-09-28 15:01:01 +02:00
|
|
|
#else
|
|
|
|
sif::printWarning("TcpTmTcServer::handleServerOperation: "
|
|
|
|
" Sending message to queue failed\n");
|
2021-05-05 12:59:42 +02:00
|
|
|
#endif /* FSFW_CPP_OSTREAM_ENABLED == 1 */
|
|
|
|
#endif /* FSFW_VERBOSE_LEVEL >= 1 */
|
|
|
|
tcStore->deleteData(storeId);
|
|
|
|
}
|
|
|
|
return result;
|
2021-03-11 14:47:47 +01:00
|
|
|
}
|
|
|
|
|
2021-09-28 15:01:01 +02:00
|
|
|
std::string TcpTmTcServer::getTcpPort() const {
|
|
|
|
return tcpConfig.tcpPort;
|
2021-05-05 15:59:41 +02:00
|
|
|
}
|
2021-03-11 14:47:47 +01:00
|
|
|
|
2021-09-28 15:01:01 +02:00
|
|
|
void TcpTmTcServer::setSpacePacketParsingOptions(std::vector<uint16_t> validPacketIds) {
|
|
|
|
this->validPacketIds = validPacketIds;
|
|
|
|
}
|
|
|
|
|
|
|
|
TcpTmTcServer::TcpConfig& TcpTmTcServer::getTcpConfigStruct() {
|
|
|
|
return tcpConfig;
|
2021-08-09 18:12:25 +02:00
|
|
|
}
|
|
|
|
|
2021-09-28 15:01:01 +02:00
|
|
|
ReturnValue_t TcpTmTcServer::handleTmSending(socket_t connSocket, bool& tmSent) {
|
2021-05-05 15:59:41 +02:00
|
|
|
// Access to the FIFO is mutex protected because it is filled by the bridge
|
|
|
|
MutexGuard(tmtcBridge->mutex, tmtcBridge->timeoutType, tmtcBridge->mutexTimeoutMs);
|
|
|
|
store_address_t storeId;
|
|
|
|
while((not tmtcBridge->tmFifo->empty()) and
|
|
|
|
(tmtcBridge->packetSentCounter < tmtcBridge->sentPacketsPerCycle)) {
|
2021-09-28 15:01:01 +02:00
|
|
|
// Send can fail, so only peek from the FIFO
|
|
|
|
tmtcBridge->tmFifo->peek(&storeId);
|
2021-05-05 15:59:41 +02:00
|
|
|
|
|
|
|
// Using the store accessor will take care of deleting TM from the store automatically
|
|
|
|
ConstStorageAccessor storeAccessor(storeId);
|
|
|
|
ReturnValue_t result = tmStore->getData(storeId, storeAccessor);
|
|
|
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
2021-09-28 15:09:56 +02:00
|
|
|
if(wiretappingEnabled) {
|
|
|
|
sif::info << "Sending TM:" << std::endl;
|
|
|
|
arrayprinter::print(storeAccessor.data(), storeAccessor.size());
|
|
|
|
}
|
2021-05-05 15:59:41 +02:00
|
|
|
int retval = send(connSocket,
|
|
|
|
reinterpret_cast<const char*>(storeAccessor.data()),
|
|
|
|
storeAccessor.size(),
|
2021-09-28 15:01:01 +02:00
|
|
|
tcpConfig.tcpTmFlags);
|
|
|
|
if(retval == static_cast<int>(storeAccessor.size())) {
|
|
|
|
// Packet sent, clear FIFO entry
|
|
|
|
tmtcBridge->tmFifo->pop();
|
|
|
|
tmSent = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
else if(retval <= 0) {
|
|
|
|
// Assume that the client has closed the connection here for now
|
|
|
|
handleSocketError(storeAccessor);
|
|
|
|
return CONN_BROKEN;
|
2021-05-05 15:59:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
|
|
|
}
|
2021-09-28 15:01:01 +02:00
|
|
|
|
|
|
|
ReturnValue_t TcpTmTcServer::handleTcRingBufferData(size_t availableReadData) {
|
|
|
|
ReturnValue_t status = HasReturnvaluesIF::RETURN_OK;
|
|
|
|
ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
|
|
|
|
size_t readAmount = availableReadData;
|
|
|
|
lastRingBufferSize = availableReadData;
|
|
|
|
if(readAmount >= ringBuffer.getMaxSize()) {
|
|
|
|
#if FSFW_VERBOSE_LEVEL >= 1
|
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
|
|
|
// Possible configuration error, too much data or/and data coming in too fast,
|
|
|
|
// requiring larger buffers
|
|
|
|
sif::warning << "TcpTmTcServer::handleServerOperation: Ring buffer reached " <<
|
|
|
|
"fill count" << std::endl;
|
|
|
|
#else
|
|
|
|
sif::printWarning("TcpTmTcServer::handleServerOperation: Ring buffer reached "
|
|
|
|
"fill count");
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
if(readAmount >= receptionBuffer.size()) {
|
|
|
|
#if FSFW_VERBOSE_LEVEL >= 1
|
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
|
|
|
// Possible configuration error, too much data or/and data coming in too fast,
|
|
|
|
// requiring larger buffers
|
|
|
|
sif::warning << "TcpTmTcServer::handleServerOperation: "
|
|
|
|
"Reception buffer too small " << std::endl;
|
|
|
|
#else
|
|
|
|
sif::printWarning("TcpTmTcServer::handleServerOperation: Reception buffer too small\n");
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
readAmount = receptionBuffer.size();
|
|
|
|
}
|
|
|
|
ringBuffer.readData(receptionBuffer.data(), readAmount, true);
|
|
|
|
const uint8_t* bufPtr = receptionBuffer.data();
|
|
|
|
const uint8_t** bufPtrPtr = &bufPtr;
|
|
|
|
size_t startIdx = 0;
|
|
|
|
size_t foundSize = 0;
|
|
|
|
size_t readLen = 0;
|
|
|
|
while(readLen < readAmount) {
|
|
|
|
result = spacePacketParser->parseSpacePackets(bufPtrPtr, readAmount,
|
|
|
|
startIdx, foundSize, readLen);
|
|
|
|
switch(result) {
|
|
|
|
case(SpacePacketParser::NO_PACKET_FOUND):
|
|
|
|
case(SpacePacketParser::SPLIT_PACKET): {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case(HasReturnvaluesIF::RETURN_OK): {
|
|
|
|
result = handleTcReception(receptionBuffer.data() + startIdx, foundSize);
|
|
|
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
status = result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ringBuffer.deleteData(foundSize);
|
|
|
|
lastRingBufferSize = ringBuffer.getAvailableReadData();
|
|
|
|
std::memset(receptionBuffer.data() + startIdx, 0, foundSize);
|
|
|
|
}
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2021-09-28 15:09:56 +02:00
|
|
|
void TcpTmTcServer::enableWiretapping(bool enable) {
|
|
|
|
this->wiretappingEnabled = enable;
|
|
|
|
}
|
|
|
|
|
2021-09-28 15:01:01 +02:00
|
|
|
void TcpTmTcServer::handleSocketError(ConstStorageAccessor &accessor) {
|
|
|
|
// Don't delete data
|
|
|
|
accessor.release();
|
|
|
|
auto socketError = getLastSocketError();
|
|
|
|
switch(socketError) {
|
|
|
|
#if defined PLATFORM_WIN
|
|
|
|
case(WSAECONNRESET): {
|
|
|
|
// See https://docs.microsoft.com/en-us/windows/win32/api/winsock2/nf-winsock2-send
|
|
|
|
// Remote client might have shut down connection
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
case(EPIPE): {
|
|
|
|
// See https://man7.org/linux/man-pages/man2/send.2.html
|
|
|
|
// Remote client might have shut down connection
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
default: {
|
|
|
|
tcpip::handleError(tcpip::Protocol::TCP, tcpip::ErrorSources::SEND_CALL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|