mirror of
https://github.com/nasa/fprime.git
synced 2025-12-11 03:38:57 -06:00
* Initial FprimeFramer and FprimePacketizer * Code clarity + set up UTs * Rework ComQueue and ComStub to use DataWithContext * Add packets to RefPackets.fppi * Fix ComQueue tests * Add hotfix to FileDownlink instead of ComQueue * Fix cancelPacket as well * Fix ComQueue UTs by removing hotfix * Refactor DataWithContext to use an FPP object for context instead of Fw.Buffer * Touch up testing * Add docs * more docs * More docs * Rework buffer deallocation pattern to pass-through ComQueue * Update ComStub UTs * Restore original FileDownlink.cpp * Formatting tweak * Update deprecated getSerializeRepr() calls * deserialization methods * Fix spelling * add cast for safety * CMakefile change * Bump ComQueue depth * Update RPI deployment with new Downlink stack * Rename comQueueIn port to comPktQueueIn * Fix comQueueIn to comPktQueueIn change * Remove legacy Svc.Framer * Fix CMake UTs * Fix RPI topology config * Fix FprimeProtocol.fpp module * Fix namespacing * Use const reference for FrameContext port * Review comments EXCEPT port passback refactor * Rework ComStub with new ByteStream * New ByteStream - ComInterface model * Rework TcpClient / TcpServer with new bytestream * Adapt UDP component for new ByteStream * Adapt FrameAccumulator for new ByteStream * Adapt FprimeFramer for new ByteStream * Update Ref topology with new ByteStream model * Remove all legacy deallocates from Drivers; reintroduce DEPRECATED model types * Fix spelling and include error * More spelling.... * RPI and RpiDemo fixes * Fix conversion warning on RPI * static_cast for short int on RPI * Standardize port names * Remove legacy Drv types and merge RECV/SEND enum type, delete StreamCrossover * Update SDDs * Update SDDs * Fix ComInterface <-> Framer interfaction, clarify comments and fix annotations * Switch ComStub from ASSERT to log failure and return buffer * Add history size check + clarify test handler overrides * Fix RPI topology to wire comStub on Uplink * Rename comm to comDriver in RPI topology * Update communication adapter interface docs
146 lines
5.1 KiB
C++
146 lines
5.1 KiB
C++
// ======================================================================
|
|
// \title TcpServerComponentImpl.cpp
|
|
// \author mstarch
|
|
// \brief cpp file for TcpServerComponentImpl component implementation class
|
|
//
|
|
// \copyright
|
|
// Copyright 2009-2020, by the California Institute of Technology.
|
|
// ALL RIGHTS RESERVED. United States Government Sponsorship
|
|
// acknowledged.
|
|
//
|
|
// ======================================================================
|
|
|
|
#include <limits>
|
|
#include <Drv/TcpServer/TcpServerComponentImpl.hpp>
|
|
#include <Fw/FPrimeBasicTypes.hpp>
|
|
#include "Fw/Types/Assert.hpp"
|
|
#include "Fw/Logger/Logger.hpp"
|
|
|
|
namespace Drv {
|
|
|
|
// ----------------------------------------------------------------------
|
|
// Construction, initialization, and destruction
|
|
// ----------------------------------------------------------------------
|
|
|
|
TcpServerComponentImpl::TcpServerComponentImpl(const char* const compName)
|
|
: TcpServerComponentBase(compName) {}
|
|
|
|
SocketIpStatus TcpServerComponentImpl::configure(const char* hostname,
|
|
const U16 port,
|
|
const U32 send_timeout_seconds,
|
|
const U32 send_timeout_microseconds,
|
|
FwSizeType buffer_size) {
|
|
|
|
// Check that ensures the configured buffer size fits within the limits fixed-width type, U32
|
|
|
|
FW_ASSERT(buffer_size <= std::numeric_limits<U32>::max(), static_cast<FwAssertArgType>(buffer_size));
|
|
m_allocation_size = buffer_size; // Store the buffer size
|
|
//
|
|
(void)m_socket.configure(hostname, port, send_timeout_seconds, send_timeout_microseconds);
|
|
return startup();
|
|
}
|
|
|
|
TcpServerComponentImpl::~TcpServerComponentImpl() {}
|
|
|
|
// ----------------------------------------------------------------------
|
|
// Implementations for socket read task virtual methods
|
|
// ----------------------------------------------------------------------
|
|
|
|
U16 TcpServerComponentImpl::getListenPort() {
|
|
return m_socket.getListenPort();
|
|
}
|
|
|
|
IpSocket& TcpServerComponentImpl::getSocketHandler() {
|
|
return m_socket;
|
|
}
|
|
|
|
Fw::Buffer TcpServerComponentImpl::getBuffer() {
|
|
return allocate_out(0, static_cast<U32>(m_allocation_size));
|
|
}
|
|
|
|
void TcpServerComponentImpl::sendBuffer(Fw::Buffer buffer, SocketIpStatus status) {
|
|
Drv::ByteStreamStatus recvStatus = ByteStreamStatus::OTHER_ERROR;
|
|
if (status == SOCK_SUCCESS) {
|
|
recvStatus = ByteStreamStatus::OP_OK;
|
|
}
|
|
else if (status == SOCK_NO_DATA_AVAILABLE) {
|
|
recvStatus = ByteStreamStatus::RECV_NO_DATA;
|
|
}
|
|
else {
|
|
recvStatus = ByteStreamStatus::OTHER_ERROR;
|
|
}
|
|
this->recv_out(0, buffer, recvStatus);
|
|
}
|
|
|
|
void TcpServerComponentImpl::connected() {
|
|
if (isConnected_ready_OutputPort(0)) {
|
|
this->ready_out(0);
|
|
}
|
|
}
|
|
|
|
bool TcpServerComponentImpl::isStarted() {
|
|
Os::ScopeLock scopedLock(this->m_lock);
|
|
return this->m_descriptor.serverFd != -1;
|
|
}
|
|
|
|
SocketIpStatus TcpServerComponentImpl::startup() {
|
|
Os::ScopeLock scopedLock(this->m_lock);
|
|
Drv::SocketIpStatus status = SOCK_SUCCESS;
|
|
// Prevent multiple startup attempts
|
|
if (this->m_descriptor.serverFd == -1) {
|
|
status = this->m_socket.startup(this->m_descriptor);
|
|
}
|
|
return status;
|
|
}
|
|
|
|
void TcpServerComponentImpl::terminate() {
|
|
Os::ScopeLock scopedLock(this->m_lock);
|
|
this->m_socket.terminate(this->m_descriptor);
|
|
this->m_descriptor.serverFd = -1;
|
|
}
|
|
|
|
void TcpServerComponentImpl::readLoop() {
|
|
Drv::SocketIpStatus status = Drv::SocketIpStatus::SOCK_NOT_STARTED;
|
|
// Keep trying to reconnect until the status is good, told to stop, or reconnection is turned off
|
|
do {
|
|
status = this->startup();
|
|
if (status != SOCK_SUCCESS) {
|
|
Fw::Logger::log("[WARNING] Failed to listen on port %hu with status %d\n", this->getListenPort(), status);
|
|
(void)Os::Task::delay(SOCKET_RETRY_INTERVAL);
|
|
continue;
|
|
}
|
|
}
|
|
while (this->running() && status != SOCK_SUCCESS && this->m_reopen);
|
|
// If start up was successful then perform normal operations
|
|
if (this->running() && status == SOCK_SUCCESS) {
|
|
// Perform the nominal read loop
|
|
SocketComponentHelper::readLoop();
|
|
}
|
|
// Terminate the server
|
|
this->terminate();
|
|
}
|
|
|
|
// ----------------------------------------------------------------------
|
|
// Handler implementations for user-defined typed input ports
|
|
// ----------------------------------------------------------------------
|
|
|
|
void TcpServerComponentImpl::send_handler(const FwIndexType portNum, Fw::Buffer& fwBuffer) {
|
|
Drv::SocketIpStatus status = this->send(fwBuffer.getData(), fwBuffer.getSize());
|
|
Drv::ByteStreamStatus returnStatus;
|
|
switch (status) {
|
|
case SOCK_INTERRUPTED_TRY_AGAIN:
|
|
returnStatus = ByteStreamStatus::SEND_RETRY;
|
|
break;
|
|
case SOCK_SUCCESS:
|
|
returnStatus = ByteStreamStatus::OP_OK;
|
|
break;
|
|
default:
|
|
returnStatus = ByteStreamStatus::OTHER_ERROR;
|
|
break;
|
|
}
|
|
// Return the buffer and status to the caller
|
|
this->dataReturnOut_out(0, fwBuffer, returnStatus);
|
|
}
|
|
|
|
} // end namespace Drv
|