169 lines
6.9 KiB
C++
169 lines
6.9 KiB
C++
#include <array>
|
|
#include <catch2/catch_test_macros.hpp>
|
|
|
|
#include "fsfw/storagemanager/LocalPool.h"
|
|
#include "fsfw/tcdistribution/CcsdsDistributor.h"
|
|
#include "fsfw/tmtcpacket/ccsds/SpacePacketCreator.h"
|
|
#include "mock/AcceptsTcMock.h"
|
|
#include "mock/CcsdsCheckerMock.h"
|
|
#include "mock/MessageQueueMock.h"
|
|
|
|
TEST_CASE("CCSDS Distributor", "[ccsds][tmtcdistrib]") {
|
|
LocalPool::LocalPoolConfig cfg = {{5, 32}, {2, 64}};
|
|
LocalPool pool(objects::NO_OBJECT, cfg);
|
|
auto queue = MessageQueueMock(1, MessageQueueIF::NO_QUEUE);
|
|
auto checkerMock = CcsdsCheckerMock();
|
|
uint16_t unregisteredApid = 0;
|
|
uint16_t defaultApid = 4;
|
|
MessageQueueId_t defaultQueueId = 5;
|
|
auto ccsdsDistrib = CcsdsDistributor(defaultApid, 1, &pool, &queue, &checkerMock);
|
|
uint32_t tcAcceptorApid = 1;
|
|
MessageQueueId_t tcAcceptorQueueId = 3;
|
|
|
|
auto tcAcceptorMock = AcceptsTcMock("TC Receiver Dummy", tcAcceptorApid, tcAcceptorQueueId);
|
|
auto defReceiverMock = AcceptsTcMock("Default Receiver Dummy", defaultApid, defaultQueueId);
|
|
auto packetId = PacketId(ccsds::PacketType::TC, true, 0);
|
|
auto psc = PacketSeqCtrl(ccsds::SequenceFlags::FIRST_SEGMENT, 0x34);
|
|
auto spParams = SpacePacketParams(packetId, psc, 0x16);
|
|
SpacePacketCreator spCreator(spParams);
|
|
std::array<uint8_t, 32> buf{};
|
|
|
|
auto createSpacePacket = [&](uint16_t apid, TmTcMessage& msg, uint8_t* dataField = nullptr,
|
|
size_t dataFieldLen = 1) {
|
|
store_address_t storeId{};
|
|
spCreator.setApid(tcAcceptorApid);
|
|
spCreator.setCcsdsLenFromTotalDataFieldLen(dataFieldLen);
|
|
uint8_t* dataPtr;
|
|
REQUIRE(pool.getFreeElement(&storeId, spCreator.getSerializedSize() + dataFieldLen, &dataPtr) ==
|
|
returnvalue::OK);
|
|
size_t serLen = 0;
|
|
REQUIRE(spCreator.SerializeIF::serializeBe(dataPtr, serLen, ccsds::HEADER_LEN) ==
|
|
returnvalue::OK);
|
|
REQUIRE(spCreator.SerializeIF::serializeBe(buf.data(), serLen, ccsds::HEADER_LEN) ==
|
|
returnvalue::OK);
|
|
if (dataField == nullptr) {
|
|
dataPtr[ccsds::HEADER_LEN] = 0;
|
|
buf[ccsds::HEADER_LEN] = 0;
|
|
} else {
|
|
std::memcpy(dataPtr + ccsds::HEADER_LEN, dataField, dataFieldLen);
|
|
std::memcpy(buf.data() + ccsds::HEADER_LEN, dataField, dataFieldLen);
|
|
}
|
|
msg.setStorageId(storeId);
|
|
};
|
|
|
|
SECTION("State") {
|
|
CHECK(ccsdsDistrib.initialize() == returnvalue::OK);
|
|
CHECK(ccsdsDistrib.getRequestQueue() == 1);
|
|
CHECK(ccsdsDistrib.getIdentifier() == 0);
|
|
CHECK(ccsdsDistrib.getObjectId() == 1);
|
|
REQUIRE(ccsdsDistrib.getName() != nullptr);
|
|
CHECK(std::strcmp(ccsdsDistrib.getName(), "CCSDS Distributor") == 0);
|
|
}
|
|
|
|
SECTION("Basic Forwarding") {
|
|
CcsdsDistributor::DestInfo info(tcAcceptorMock, false);
|
|
REQUIRE(ccsdsDistrib.registerApplication(info) == returnvalue::OK);
|
|
TmTcMessage message;
|
|
createSpacePacket(tcAcceptorApid, message);
|
|
store_address_t storeId = message.getStorageId();
|
|
queue.addReceivedMessage(message);
|
|
REQUIRE(ccsdsDistrib.performOperation(0) == returnvalue::OK);
|
|
CHECK(checkerMock.checkedPacketLen == 7);
|
|
CHECK(checkerMock.checkCallCount == 1);
|
|
CHECK(queue.wasMessageSent());
|
|
CHECK(queue.numberOfSentMessages() == 1);
|
|
// The packet is forwarded, with no need to delete the data
|
|
CHECK(pool.hasDataAtId(storeId));
|
|
TmTcMessage sentMsg;
|
|
CHECK(queue.getNextSentMessage(tcAcceptorQueueId, sentMsg) == returnvalue::OK);
|
|
CHECK(sentMsg.getStorageId() == storeId);
|
|
auto accessor = pool.getData(storeId);
|
|
CHECK(accessor.first == returnvalue::OK);
|
|
CHECK(accessor.second.size() == ccsds::HEADER_LEN + 1);
|
|
for (size_t i = 0; i < ccsds::HEADER_LEN; i++) {
|
|
CHECK(accessor.second.data()[i] == buf[i]);
|
|
}
|
|
}
|
|
|
|
SECTION("Forwarding to Default Destination, but not registered") {
|
|
TmTcMessage message;
|
|
createSpacePacket(unregisteredApid, message);
|
|
store_address_t storeId = message.getStorageId();
|
|
message.setStorageId(storeId);
|
|
queue.addReceivedMessage(message);
|
|
REQUIRE(ccsdsDistrib.performOperation(0) == TcDistributorBase::DESTINATION_NOT_FOUND);
|
|
}
|
|
|
|
SECTION("Forward to Default Handler") {
|
|
CcsdsDistributor::DestInfo info(defReceiverMock, false);
|
|
ccsdsDistrib.registerApplication(info);
|
|
TmTcMessage message;
|
|
createSpacePacket(defaultApid, message);
|
|
store_address_t storeId = message.getStorageId();
|
|
message.setStorageId(storeId);
|
|
queue.addReceivedMessage(message);
|
|
REQUIRE(ccsdsDistrib.performOperation(0) == returnvalue::OK);
|
|
CHECK(checkerMock.checkedPacketLen == 7);
|
|
CHECK(checkerMock.checkCallCount == 1);
|
|
CHECK(queue.wasMessageSent());
|
|
CHECK(queue.numberOfSentMessages() == 1);
|
|
// The packet is forwarded, with no need to delete the data
|
|
CHECK(pool.hasDataAtId(storeId));
|
|
TmTcMessage sentMsg;
|
|
CHECK(queue.getNextSentMessage(defaultQueueId, sentMsg) == returnvalue::OK);
|
|
CHECK(sentMsg.getStorageId() == storeId);
|
|
auto accessor = pool.getData(storeId);
|
|
CHECK(accessor.first == returnvalue::OK);
|
|
CHECK(accessor.second.size() == ccsds::HEADER_LEN + 1);
|
|
for (size_t i = 0; i < ccsds::HEADER_LEN; i++) {
|
|
CHECK(accessor.second.data()[i] == buf[i]);
|
|
}
|
|
}
|
|
|
|
SECTION("Remove CCSDS header") {
|
|
uint16_t tgtApid = 0;
|
|
MessageQueueId_t tgtQueueId = MessageQueueIF::NO_QUEUE;
|
|
SECTION("Default destination") {
|
|
CcsdsDistributor::DestInfo info(defReceiverMock, true);
|
|
tgtApid = defaultApid;
|
|
tgtQueueId = defaultQueueId;
|
|
REQUIRE(ccsdsDistrib.registerApplication(info) == returnvalue::OK);
|
|
}
|
|
SECTION("Specific destination") {
|
|
CcsdsDistributor::DestInfo info(tcAcceptorMock, true);
|
|
tgtApid = tcAcceptorApid;
|
|
tgtQueueId = tcAcceptorQueueId;
|
|
REQUIRE(ccsdsDistrib.registerApplication(info) == returnvalue::OK);
|
|
}
|
|
TmTcMessage message;
|
|
std::array<uint8_t, 5> dataField = {0, 1, 2, 3, 4};
|
|
createSpacePacket(tgtApid, message, dataField.data(), 5);
|
|
store_address_t storeId = message.getStorageId();
|
|
message.setStorageId(storeId);
|
|
queue.addReceivedMessage(message);
|
|
REQUIRE(ccsdsDistrib.performOperation(0) == returnvalue::OK);
|
|
CHECK(checkerMock.checkedPacketLen == 11);
|
|
CHECK(checkerMock.checkCallCount == 1);
|
|
// Data was deleted from old slot to re-store without the header
|
|
CHECK(not pool.hasDataAtId(storeId));
|
|
TmTcMessage sentMsg;
|
|
CHECK(queue.getNextSentMessage(tgtQueueId, sentMsg) == returnvalue::OK);
|
|
CHECK(sentMsg.getStorageId() != storeId);
|
|
auto accessor = pool.getData(sentMsg.getStorageId());
|
|
CHECK(accessor.first == returnvalue::OK);
|
|
CHECK(accessor.second.size() == 5);
|
|
// Verify correctness of data field
|
|
for (size_t i = 0; i < 5; i++) {
|
|
CHECK(accessor.second.data()[i] == i);
|
|
}
|
|
}
|
|
|
|
SECTION("Invalid Space Packet, Too Short") {
|
|
store_address_t storeId{};
|
|
std::array<uint8_t, 4> data = {1, 2, 3, 4};
|
|
pool.addData(&storeId, data.data(), data.size());
|
|
TmTcMessage message(storeId);
|
|
queue.addReceivedMessage(message);
|
|
REQUIRE(ccsdsDistrib.performOperation(0) == SerializeIF::STREAM_TOO_SHORT);
|
|
}
|
|
} |