mirror of
https://github.com/nasa/fprime.git
synced 2025-12-10 00:44:37 -06:00
* First pass at Svc + TcpClient implementation * Revert FileUplink changes * Add copy (with allocation/deallocation) to FprimeRouter to simplify buffer management * Update FprimeRouter UTs * Update FprimeDeframer UTs * Update FrameAccumulator UTs * Update ComStub UTs * Update missing Drv and UTs * Update ComInterface to use ComDataWithContext on output * Update Ref/RPI topology * Fix spelling * Fix test typo * Update Udp component and UTs * Rename data ports and standardize "Return" naming pattern * Fix variable name * Adapt UTs * Update Communication Adapter Interface docs * Full SDD updates * Spelling & nits and details * Put formatting back to original * Update Deframer interface to include bufferReturn * Address review comments
221 lines
9.3 KiB
C++
221 lines
9.3 KiB
C++
// ======================================================================
|
|
// \title FrameAccumulatorTester.cpp
|
|
// \author thomas-bc
|
|
// \brief cpp file for FrameAccumulator component test harness implementation class
|
|
// ======================================================================
|
|
|
|
#include "FrameAccumulatorTester.hpp"
|
|
#include "STest/Random/Random.hpp"
|
|
|
|
|
|
|
|
namespace Svc {
|
|
|
|
// ----------------------------------------------------------------------
|
|
// Construction and destruction
|
|
// ----------------------------------------------------------------------
|
|
|
|
FrameAccumulatorTester ::FrameAccumulatorTester()
|
|
: FrameAccumulatorGTestBase("FrameAccumulatorTester", FrameAccumulatorTester::MAX_HISTORY_SIZE),
|
|
component("FrameAccumulator") {
|
|
component.configure(this->mockDetector, 1, this->mallocator, 2048);
|
|
this->initComponents();
|
|
this->connectPorts();
|
|
}
|
|
|
|
|
|
FrameAccumulatorTester ::~FrameAccumulatorTester() {
|
|
component.cleanup();
|
|
}
|
|
|
|
// ----------------------------------------------------------------------
|
|
// Tests
|
|
// ----------------------------------------------------------------------
|
|
|
|
void FrameAccumulatorTester ::testFrameDetected() {
|
|
// Prepare a random size buffer
|
|
U32 buffer_size = STest::Random::lowerUpper(1, 1024);
|
|
U8 data[buffer_size];
|
|
Fw::Buffer buffer(data, buffer_size);
|
|
ComCfg::FrameContext context;
|
|
// Set the mock detector to report success of size_out = buffer_size
|
|
this->mockDetector.set_next_result(FrameDetector::Status::FRAME_DETECTED, buffer_size);
|
|
// Receive the buffer on dataIn
|
|
this->invoke_to_dataIn(0, buffer, context);
|
|
// Checks
|
|
ASSERT_from_dataReturnOut_SIZE(1); // input buffer ownership was returned
|
|
ASSERT_from_dataOut_SIZE(1); // frame was sent
|
|
ASSERT_EQ(this->component.m_inRing.get_allocated_size(), 0); // no data left in ring buffer
|
|
ASSERT_EQ(this->fromPortHistory_dataOut->at(0).data.getSize(), buffer_size); // all data was sent out
|
|
}
|
|
|
|
void FrameAccumulatorTester ::testMoreDataNeeded() {
|
|
// Prepare a random size buffer
|
|
U32 buffer_size = STest::Random::lowerUpper(1, 1024);
|
|
U8 data[buffer_size];
|
|
Fw::Buffer buffer(data, buffer_size);
|
|
ComCfg::FrameContext context;
|
|
// Set the mock detector to report more data needed
|
|
this->mockDetector.set_next_result(FrameDetector::Status::MORE_DATA_NEEDED, buffer_size + 1);
|
|
// Receive the buffer on dataIn
|
|
this->invoke_to_dataIn(0, buffer, context);
|
|
// Checks
|
|
ASSERT_from_dataReturnOut_SIZE(1); // input buffer ownership was returned
|
|
ASSERT_from_dataOut_SIZE(0); // frame was not sent (waiting on more data)
|
|
ASSERT_EQ(this->component.m_inRing.get_allocated_size(), buffer_size); // data left in ring buffer
|
|
}
|
|
|
|
void FrameAccumulatorTester ::testNoFrameDetected() {
|
|
// Prepare a random size buffer
|
|
U32 buffer_size = STest::Random::lowerUpper(1, 1024);
|
|
U8 data[buffer_size];
|
|
Fw::Buffer buffer(data, buffer_size);
|
|
ComCfg::FrameContext context;
|
|
// Set the mock detector
|
|
this->mockDetector.set_next_result(FrameDetector::Status::NO_FRAME_DETECTED, 0);
|
|
// Receive the buffer on dataIn
|
|
this->invoke_to_dataIn(0, buffer, context);
|
|
// Checks
|
|
ASSERT_from_dataReturnOut_SIZE(1); // input buffer ownership was returned
|
|
ASSERT_from_dataOut_SIZE(0); // No frame was sent out
|
|
ASSERT_EQ(this->component.m_inRing.get_allocated_size(), 0); // all data was consumed and discarded
|
|
}
|
|
|
|
void FrameAccumulatorTester ::testReceiveZeroSizeBuffer() {
|
|
|
|
// Prepare a zero size buffer
|
|
U8 data[1] = {0};
|
|
Fw::Buffer buffer(data, 0);
|
|
ComCfg::FrameContext context;
|
|
// Receive the buffer on dataIn
|
|
this->invoke_to_dataIn(0, buffer, context);
|
|
// Checks
|
|
ASSERT_from_dataReturnOut_SIZE(1); // input buffer ownership was returned
|
|
ASSERT_from_dataOut_SIZE(0); // No frame was sent out
|
|
ASSERT_EQ(this->component.m_inRing.get_allocated_size(), 0); // No data in ring buffer
|
|
ASSERT_EQ(this->component.m_inRing.m_head_idx, 0);
|
|
}
|
|
|
|
void FrameAccumulatorTester ::testAccumulateTwoBuffers() {
|
|
Fw::Buffer::SizeType buffer1_size = 10;
|
|
Fw::Buffer::SizeType buffer2_size = 20;
|
|
U8 data1[buffer1_size];
|
|
U8 data2[buffer2_size];
|
|
Fw::Buffer buffer1(data1, buffer1_size);
|
|
Fw::Buffer buffer2(data2, buffer2_size);
|
|
ComCfg::FrameContext context;
|
|
|
|
this->mockDetector.set_next_result(FrameDetector::Status::MORE_DATA_NEEDED, buffer2_size);
|
|
// Receive the buffer on dataIn
|
|
this->invoke_to_dataIn(0, buffer1, context);
|
|
// Next result is detection of a full frame, size = buffer1_size + buffer2_size
|
|
this->mockDetector.set_next_result(FrameDetector::Status::FRAME_DETECTED, buffer1_size + buffer2_size );
|
|
// Receive the buffer on dataIn
|
|
this->invoke_to_dataIn(0, buffer2, context);
|
|
|
|
// Checks
|
|
ASSERT_from_dataReturnOut_SIZE(2); // both input buffers ownership were returned
|
|
ASSERT_from_dataOut_SIZE(1); // Exactly one frame was sent out
|
|
ASSERT_EQ(this->component.m_inRing.get_allocated_size(), 0); // No data in ring buffer
|
|
}
|
|
|
|
void FrameAccumulatorTester ::testAccumulateBuffersEmitFrame() {
|
|
U32 frame_size = 0;
|
|
U32 buffer_count = 0;
|
|
this->mockAccumulateFullFrame(frame_size, buffer_count);
|
|
// Checks
|
|
ASSERT_from_dataReturnOut_SIZE(buffer_count); // all input buffers ownership were returned
|
|
ASSERT_from_dataOut_SIZE(1); // Exactly one frame was sent out
|
|
ASSERT_EQ(this->component.m_inRing.get_allocated_size(), 0); // No data left in ring buffer
|
|
ASSERT_EQ(this->fromPortHistory_dataOut->at(0).data.getSize(), frame_size); // accumulated buffer size
|
|
}
|
|
|
|
void FrameAccumulatorTester ::testAccumulateBuffersEmitManyFrames() {
|
|
U32 max_iters = STest::Random::lowerUpper(1, 10);
|
|
U32 total_buffer_received = 0;
|
|
|
|
U32 frame_size = 0;
|
|
U32 buffer_count = 0;
|
|
|
|
// Send frames successively, perform some checks after each frame
|
|
for (U32 i = 0; i < max_iters; i++) {
|
|
this->mockAccumulateFullFrame(frame_size, buffer_count);
|
|
total_buffer_received += buffer_count;
|
|
|
|
ASSERT_from_dataReturnOut_SIZE(total_buffer_received); // all input buffers returned
|
|
ASSERT_from_dataOut_SIZE(i+1); // Exactly one frame was sent out
|
|
ASSERT_EQ(this->component.m_inRing.get_allocated_size(), 0); // No data left in ring buffer
|
|
ASSERT_EQ(this->fromPortHistory_dataOut->at(i).data.getSize(), frame_size); // accumulated buffer size
|
|
}
|
|
// Final checks
|
|
ASSERT_from_dataReturnOut_SIZE(total_buffer_received); // all input buffers returned
|
|
ASSERT_from_dataOut_SIZE(max_iters); // Exactly max_iters frames were sent out
|
|
ASSERT_EQ(this->component.m_inRing.get_allocated_size(), 0); // No data left in ring buffer
|
|
}
|
|
|
|
void FrameAccumulatorTester ::testBufferReturnDeallocation() {
|
|
U8 data[1];
|
|
Fw::Buffer buffer(data, sizeof(data));
|
|
ComCfg::FrameContext context;
|
|
this->invoke_to_dataReturnIn(0, buffer, context);
|
|
ASSERT_from_bufferDeallocate_SIZE(1); // incoming buffer should be deallocated
|
|
ASSERT_EQ(this->fromPortHistory_bufferDeallocate->at(0).fwBuffer.getData(), data);
|
|
ASSERT_EQ(this->fromPortHistory_bufferDeallocate->at(0).fwBuffer.getSize(), sizeof(data));
|
|
}
|
|
|
|
// ----------------------------------------------------------------------
|
|
// Helper functions
|
|
// ----------------------------------------------------------------------
|
|
|
|
void FrameAccumulatorTester ::mockAccumulateFullFrame(U32& frame_size, U32& buffer_count) {
|
|
// Constants need to be picked so that:
|
|
// - the worst case doesn't overflow the ring buffer size (max_size * iters < 2048)
|
|
// - iters < MAX_HISTORY_SIZE
|
|
const U32 buffer_max_size = 64;
|
|
const U32 iters = STest::Random::lowerUpper(0, 10);
|
|
|
|
U8 data[buffer_max_size];
|
|
U32 buffer_size;
|
|
Fw::Buffer buffer(data, 0);
|
|
U32 accumulated_size = 0;
|
|
ComCfg::FrameContext context;
|
|
|
|
// Send multiple buffers with MORE_DATA_NEEDED
|
|
for (U32 i = 0; i < iters; i++) {
|
|
buffer_size = STest::Random::lowerUpper(1, buffer_max_size);
|
|
accumulated_size += buffer_size;
|
|
buffer.setSize(buffer_size);
|
|
// Detector reports MORE_DATA_NEEDED and size needed bigger than accumulated size so far
|
|
this->mockDetector.set_next_result(FrameDetector::Status::MORE_DATA_NEEDED, accumulated_size + 1);
|
|
this->invoke_to_dataIn(0, buffer, context);
|
|
}
|
|
|
|
// Send last buffer with FRAME_DETECTED
|
|
buffer_size = STest::Random::lowerUpper(1, buffer_max_size);
|
|
buffer.setSize(buffer_size);
|
|
accumulated_size += buffer_size; // accumulate once more (sending last buffer below)
|
|
// Send last buffer with finally FRAME_DETECTED and total accumulated + last buffer
|
|
this->mockDetector.set_next_result(FrameDetector::Status::FRAME_DETECTED, accumulated_size);
|
|
// Receive the last buffer on dataIn
|
|
this->invoke_to_dataIn(0, buffer, context);
|
|
frame_size = accumulated_size;
|
|
buffer_count = iters + 1;
|
|
}
|
|
|
|
// ----------------------------------------------------------------------
|
|
// Port handler overrides
|
|
// ----------------------------------------------------------------------
|
|
Fw::Buffer FrameAccumulatorTester ::from_bufferAllocate_handler(
|
|
FwIndexType portNum,
|
|
U32 size
|
|
)
|
|
{
|
|
this->pushFromPortEntry_bufferAllocate(size);
|
|
this->m_buffer.setData(this->m_buffer_slot);
|
|
this->m_buffer.setSize(size);
|
|
::memset(this->m_buffer.getData(), 0, size);
|
|
return this->m_buffer;
|
|
}
|
|
|
|
} // namespace Svc
|