This PR adds the packet stack for the CCSDS File Delivery Protocol. It also refactors the existing TMTC infastructure to allow sending of CFDP packets to the CCSDS handlers. This includes the whole PDU (Protocol Data Unit) stack: - File Data PDUs and all file directive PDUs - ACK PDU - NAK PDU - Metadata PDU - Finished PDU - Prompt PDU - Keep Alive PDU - EOF PDU The PR includes a full set of unittests for the packet stack with a coverage of 90+ %. The refactoring of the existing TMTC infastructure includes non-ideal solutions like diamond inheritance. Avoiding this solution would require refactoring the packet stack. This would be a good idea anyway because the existing stack is tightly coupled to the FSFW, making reuse more difficult if only the stack is planned to be used without the store functionalities etc. The PDU implementation provided here is only weakly coupled to the FSFW, only using components like returnvalues or the Serialization modules. There are dedicated serializers and deserializers, which also helps in creating small focused modules which are easy to test. Some of the modules here were provied by Matthias Tompert.
197 lines
9.6 KiB
C++
197 lines
9.6 KiB
C++
#include <catch2/catch_test_macros.hpp>
|
|
|
|
#include "fsfw/globalfunctions/arrayprinter.h"
|
|
#include "fsfw/cfdp/pdu/FinishedPduDeserializer.h"
|
|
#include "fsfw/cfdp/pdu/FinishedPduSerializer.h"
|
|
|
|
#include <array>
|
|
|
|
TEST_CASE( "Finished PDU" , "[FinishedPdu]") {
|
|
using namespace cfdp;
|
|
ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
|
|
std::array<uint8_t, 256> fnBuffer = {};
|
|
uint8_t* buffer = fnBuffer.data();
|
|
size_t sz = 0;
|
|
EntityId destId(WidthInBytes::TWO_BYTES, 2);
|
|
TransactionSeqNum seqNum(WidthInBytes::TWO_BYTES, 15);
|
|
EntityId sourceId(WidthInBytes::TWO_BYTES, 1);
|
|
PduConfig pduConf(TransmissionModes::ACKNOWLEDGED, seqNum, sourceId, destId);
|
|
|
|
cfdp::Lv emptyFsMsg;
|
|
FinishedInfo info(cfdp::ConditionCode::INACTIVITY_DETECTED,
|
|
cfdp::FinishedDeliveryCode::DATA_INCOMPLETE,
|
|
cfdp::FinishedFileStatus::DISCARDED_DELIBERATELY);
|
|
|
|
SECTION("Serialize") {
|
|
FinishPduSerializer serializer(pduConf, info);
|
|
result = serializer.serialize(&buffer, &sz, fnBuffer.size(),
|
|
SerializeIF::Endianness::NETWORK);
|
|
REQUIRE(result == HasReturnvaluesIF::RETURN_OK);
|
|
REQUIRE(serializer.getSerializedSize() == 12);
|
|
REQUIRE(((fnBuffer[1] << 8) | fnBuffer[2]) == 2);
|
|
REQUIRE(fnBuffer[10] == cfdp::FileDirectives::FINISH);
|
|
REQUIRE(((fnBuffer[sz - 1] >> 4) & 0x0f) == cfdp::ConditionCode::INACTIVITY_DETECTED);
|
|
REQUIRE(((fnBuffer[sz - 1] >> 2) & 0x01) == cfdp::FinishedDeliveryCode::DATA_INCOMPLETE);
|
|
REQUIRE((fnBuffer[sz - 1] & 0b11) == cfdp::FinishedFileStatus::DISCARDED_DELIBERATELY);
|
|
REQUIRE(sz == 12);
|
|
|
|
// Add a filestore response
|
|
std::string firstName = "hello.txt";
|
|
cfdp::Lv firstNameLv(reinterpret_cast<uint8_t*>(firstName.data()), firstName.size());
|
|
FilestoreResponseTlv response(cfdp::FilestoreActionCode::DELETE_FILE,
|
|
cfdp::FSR_APPEND_FILE_1_NOT_EXISTS, firstNameLv, nullptr);
|
|
FilestoreResponseTlv* responsePtr = &response;
|
|
REQUIRE(response.getSerializedSize() == 14);
|
|
size_t len = 1;
|
|
info.setFilestoreResponsesArray(&responsePtr, &len, &len);
|
|
serializer.updateDirectiveFieldLen();
|
|
REQUIRE(serializer.getSerializedSize() == 12 + 14);
|
|
sz = 0;
|
|
buffer = fnBuffer.data();
|
|
result = serializer.serialize(&buffer, &sz, fnBuffer.size(),
|
|
SerializeIF::Endianness::NETWORK);
|
|
REQUIRE(result == HasReturnvaluesIF::RETURN_OK);
|
|
REQUIRE(serializer.getSerializedSize() == 12 + 14);
|
|
REQUIRE(serializer.getPduDataFieldLen() == 16);
|
|
|
|
// Add two filestore responses and a fault location parameter
|
|
std::string secondName = "hello2.txt";
|
|
cfdp::Lv secondNameLv(reinterpret_cast<uint8_t*>(secondName.data()), secondName.size());
|
|
FilestoreResponseTlv response2(cfdp::FilestoreActionCode::DENY_FILE ,
|
|
cfdp::FSR_SUCCESS, secondNameLv, nullptr);
|
|
REQUIRE(response2.getSerializedSize() == 15);
|
|
len = 2;
|
|
std::array<FilestoreResponseTlv*, 2> responses {&response, &response2};
|
|
info.setFilestoreResponsesArray(responses.data(), &len, &len);
|
|
serializer.updateDirectiveFieldLen();
|
|
|
|
EntityIdTlv faultLoc(destId);
|
|
REQUIRE(faultLoc.getSerializedSize() == 4);
|
|
info.setFaultLocation(&faultLoc);
|
|
serializer.updateDirectiveFieldLen();
|
|
sz = 0;
|
|
buffer = fnBuffer.data();
|
|
result = serializer.serialize(&buffer, &sz, fnBuffer.size(),
|
|
SerializeIF::Endianness::NETWORK);
|
|
REQUIRE(result == HasReturnvaluesIF::RETURN_OK);
|
|
info.setConditionCode(cfdp::ConditionCode::FILESTORE_REJECTION);
|
|
REQUIRE(serializer.getSerializedSize() == 12 + 14 + 15 + 4);
|
|
REQUIRE(sz == 12 + 14 + 15 + 4);
|
|
info.setFileStatus(cfdp::FinishedFileStatus::DISCARDED_FILESTORE_REJECTION);
|
|
REQUIRE(info.getFileStatus() == cfdp::FinishedFileStatus::DISCARDED_FILESTORE_REJECTION);
|
|
info.setDeliveryCode(cfdp::FinishedDeliveryCode::DATA_INCOMPLETE);
|
|
REQUIRE(info.getDeliveryCode() == cfdp::FinishedDeliveryCode::DATA_INCOMPLETE);
|
|
for(size_t maxSz = 0; maxSz < 45; maxSz++) {
|
|
sz = 0;
|
|
buffer = fnBuffer.data();
|
|
result = serializer.serialize(&buffer, &sz, maxSz, SerializeIF::Endianness::NETWORK);
|
|
REQUIRE(result != HasReturnvaluesIF::RETURN_OK);
|
|
}
|
|
}
|
|
|
|
SECTION("Deserialize") {
|
|
FinishedInfo emptyInfo;
|
|
FinishPduSerializer serializer(pduConf, info);
|
|
result = serializer.serialize(&buffer, &sz, fnBuffer.size(),
|
|
SerializeIF::Endianness::NETWORK);
|
|
REQUIRE(result == HasReturnvaluesIF::RETURN_OK);
|
|
FinishPduDeserializer deserializer(fnBuffer.data(), fnBuffer.size(), emptyInfo);
|
|
result = deserializer.parseData();
|
|
REQUIRE(result == HasReturnvaluesIF::RETURN_OK);
|
|
REQUIRE(emptyInfo.getFileStatus() == cfdp::FinishedFileStatus::DISCARDED_DELIBERATELY);
|
|
REQUIRE(emptyInfo.getConditionCode() == cfdp::ConditionCode::INACTIVITY_DETECTED);
|
|
REQUIRE(emptyInfo.getDeliveryCode() == cfdp::FinishedDeliveryCode::DATA_INCOMPLETE);
|
|
|
|
// Add a filestore response
|
|
sz = 0;
|
|
buffer = fnBuffer.data();
|
|
std::string firstName = "hello.txt";
|
|
cfdp::Lv firstNameLv(reinterpret_cast<uint8_t*>(firstName.data()), firstName.size());
|
|
FilestoreResponseTlv response(cfdp::FilestoreActionCode::DELETE_FILE,
|
|
cfdp::FSR_NOT_PERFORMED, firstNameLv, nullptr);
|
|
FilestoreResponseTlv* responsePtr = &response;
|
|
size_t len = 1;
|
|
info.setFilestoreResponsesArray(&responsePtr, &len, &len);
|
|
serializer.updateDirectiveFieldLen();
|
|
REQUIRE(serializer.getPduDataFieldLen() == 16);
|
|
result = serializer.serialize(&buffer, &sz, fnBuffer.size(),
|
|
SerializeIF::Endianness::NETWORK);
|
|
REQUIRE(result == HasReturnvaluesIF::RETURN_OK);
|
|
FilestoreResponseTlv emptyResponse(firstNameLv, nullptr);
|
|
responsePtr = &emptyResponse;
|
|
emptyInfo.setFilestoreResponsesArray(&responsePtr, nullptr, &len);
|
|
FinishPduDeserializer deserializer2(fnBuffer.data(), fnBuffer.size(), emptyInfo);
|
|
result = deserializer2.parseData();
|
|
REQUIRE(result == HasReturnvaluesIF::RETURN_OK);
|
|
REQUIRE(emptyInfo.getFsResponsesLen() == 1);
|
|
FilestoreResponseTlv** responseArray = nullptr;
|
|
emptyInfo.getFilestoreResonses(&responseArray, nullptr, nullptr);
|
|
REQUIRE(responseArray[0]->getActionCode() == cfdp::FilestoreActionCode::DELETE_FILE);
|
|
REQUIRE(responseArray[0]->getStatusCode() == cfdp::FSR_NOT_PERFORMED);
|
|
auto& fileNameRef = responseArray[0]->getFirstFileName();
|
|
size_t stringSize = 0;
|
|
const char* string = reinterpret_cast<const char*>(fileNameRef.getValue(&stringSize));
|
|
std::string firstFileName(string, stringSize);
|
|
REQUIRE(firstFileName == "hello.txt");
|
|
|
|
// Add two filestore responses and a fault location parameter
|
|
std::string secondName = "hello2.txt";
|
|
cfdp::Lv secondNameLv(reinterpret_cast<uint8_t*>(secondName.data()), secondName.size());
|
|
FilestoreResponseTlv response2(cfdp::FilestoreActionCode::DENY_FILE ,
|
|
cfdp::FSR_SUCCESS, secondNameLv, nullptr);
|
|
REQUIRE(response2.getSerializedSize() == 15);
|
|
len = 2;
|
|
std::array<FilestoreResponseTlv*, 2> responses {&response, &response2};
|
|
info.setFilestoreResponsesArray(responses.data(), &len, &len);
|
|
serializer.updateDirectiveFieldLen();
|
|
|
|
EntityIdTlv faultLoc(destId);
|
|
info.setFaultLocation(&faultLoc);
|
|
serializer.updateDirectiveFieldLen();
|
|
sz = 0;
|
|
buffer = fnBuffer.data();
|
|
result = serializer.serialize(&buffer, &sz, fnBuffer.size(),
|
|
SerializeIF::Endianness::NETWORK);
|
|
REQUIRE(result == HasReturnvaluesIF::RETURN_OK);
|
|
EntityId emptyId;
|
|
EntityIdTlv emptyFaultLoc(emptyId);
|
|
emptyInfo.setFaultLocation(&emptyFaultLoc);
|
|
response.setFilestoreMessage(&emptyFsMsg);
|
|
emptyInfo.setFilestoreResponsesArray(responses.data(), &len, &len);
|
|
response2.setFilestoreMessage(&emptyFsMsg);
|
|
FinishPduDeserializer deserializer3(fnBuffer.data(), fnBuffer.size(), emptyInfo);
|
|
result = deserializer3.parseData();
|
|
REQUIRE(result == HasReturnvaluesIF::RETURN_OK);
|
|
auto& infoRef = deserializer3.getInfo();
|
|
REQUIRE(deserializer3.getWholePduSize() == 45);
|
|
|
|
size_t invalidMaxLen = 1;
|
|
emptyInfo.setFilestoreResponsesArray(responses.data(), &len, &invalidMaxLen);
|
|
result = deserializer3.parseData();
|
|
REQUIRE(result == cfdp::FINISHED_CANT_PARSE_FS_RESPONSES);
|
|
emptyInfo.setFilestoreResponsesArray(nullptr, nullptr, nullptr);
|
|
result = deserializer3.parseData();
|
|
REQUIRE(result == cfdp::FINISHED_CANT_PARSE_FS_RESPONSES);
|
|
|
|
// Clear condition code
|
|
auto tmp = fnBuffer[11];
|
|
fnBuffer[11] = fnBuffer[11] & ~0xf0;
|
|
fnBuffer[11] = fnBuffer[11] | (cfdp::ConditionCode::NO_ERROR << 4);
|
|
emptyInfo.setFilestoreResponsesArray(responses.data(), &len, &len);
|
|
result = deserializer3.parseData();
|
|
REQUIRE(result == cfdp::INVALID_TLV_TYPE);
|
|
|
|
fnBuffer[11] = tmp;
|
|
// Invalid TLV type, should be entity ID
|
|
fnBuffer[sz - 4] = cfdp::TlvTypes::FILESTORE_REQUEST;
|
|
result = deserializer3.parseData();
|
|
REQUIRE(result == cfdp::INVALID_TLV_TYPE);
|
|
|
|
for(size_t maxSz = 0; maxSz < 45; maxSz++) {
|
|
FinishPduDeserializer faultyDeser(fnBuffer.data(), maxSz, emptyInfo);
|
|
result = faultyDeser.parseData();
|
|
REQUIRE(result != HasReturnvaluesIF::RETURN_OK);
|
|
}
|
|
}
|
|
}
|