fsfw/src/fsfw/tmtcpacket/pus/tc/TcPacketPus.cpp

100 lines
3.5 KiB
C++
Raw Normal View History

2021-10-05 13:13:36 +02:00
#include "TcPacketPus.h"
2021-06-13 12:34:06 +02:00
2021-06-13 16:29:13 +02:00
#include <cstring>
2021-06-13 12:34:06 +02:00
2022-02-02 10:29:30 +01:00
#include "fsfw/globalfunctions/CRC.h"
TcPacketPus::TcPacketPus(const uint8_t *setData) : TcPacketPusBase(setData) {
tcData = reinterpret_cast<TcPacketPointer *>(const_cast<uint8_t *>(setData));
2021-06-13 16:29:13 +02:00
}
2022-02-02 10:29:30 +01:00
void TcPacketPus::initializeTcPacket(uint16_t apid, uint16_t sequenceCount, uint8_t ack,
uint8_t service, uint8_t subservice,
pus::PusVersion pusVersion, uint16_t sourceId) {
initSpacePacketHeader(true, true, apid, sequenceCount);
std::memset(&tcData->dataField, 0, sizeof(tcData->dataField));
setPacketDataLength(sizeof(PUSTcDataFieldHeader) + CRC_SIZE - 1);
// Data Field Header. For PUS A, the first bit (CCSDS Secondary Header Flag) is zero
tcData->dataField.versionTypeAck = pusVersion << 4 | (ack & 0x0F);
tcData->dataField.serviceType = service;
tcData->dataField.serviceSubtype = subservice;
2021-06-14 15:24:26 +02:00
#if FSFW_USE_PUS_C_TELECOMMANDS == 1
2022-02-02 10:29:30 +01:00
tcData->dataField.sourceIdH = (sourceId >> 8) | 0xff;
tcData->dataField.sourceIdL = sourceId & 0xff;
2021-06-14 15:24:26 +02:00
#else
2022-02-02 10:29:30 +01:00
tcData->dataField.sourceId = sourceId;
2021-06-14 15:24:26 +02:00
#endif
2021-06-14 11:16:56 +02:00
}
2022-02-02 10:29:30 +01:00
uint8_t TcPacketPus::getService() const { return tcData->dataField.serviceType; }
2021-06-13 16:29:13 +02:00
2022-02-02 10:29:30 +01:00
uint8_t TcPacketPus::getSubService() const { return tcData->dataField.serviceSubtype; }
2021-06-13 16:29:13 +02:00
2021-06-14 11:16:56 +02:00
uint8_t TcPacketPus::getAcknowledgeFlags() const {
2022-02-02 10:29:30 +01:00
return tcData->dataField.versionTypeAck & 0b00001111;
2021-06-13 16:29:13 +02:00
}
2022-02-02 10:29:30 +01:00
const uint8_t *TcPacketPus::getApplicationData() const { return &tcData->appData; }
2021-06-13 16:29:13 +02:00
2021-06-14 11:16:56 +02:00
uint16_t TcPacketPus::getApplicationDataSize() const {
2022-07-18 10:20:26 +02:00
return SpacePacketBase::getPacketDataLen() - sizeof(tcData->dataField) - CRC_SIZE + 1;
2021-06-13 16:29:13 +02:00
}
2021-06-14 11:16:56 +02:00
uint16_t TcPacketPus::getErrorControl() const {
2022-02-02 10:29:30 +01:00
uint16_t size = getApplicationDataSize() + CRC_SIZE;
uint8_t *p_to_buffer = &tcData->appData;
return (p_to_buffer[size - 2] << 8) + p_to_buffer[size - 1];
2021-06-13 16:29:13 +02:00
}
2021-06-14 10:19:01 +02:00
void TcPacketPus::setErrorControl() {
2022-02-02 10:29:30 +01:00
uint32_t full_size = getFullSize();
uint16_t crc = CRC::crc16ccitt(getWholeData(), full_size - CRC_SIZE);
uint32_t size = getApplicationDataSize();
(&tcData->appData)[size] = (crc & 0XFF00) >> 8; // CRCH
(&tcData->appData)[size + 1] = (crc)&0X00FF; // CRCL
2021-06-13 16:29:13 +02:00
}
2021-06-14 11:16:56 +02:00
uint8_t TcPacketPus::getSecondaryHeaderFlag() const {
2021-06-14 11:44:39 +02:00
#if FSFW_USE_PUS_C_TELECOMMANDS == 1
2022-02-02 10:29:30 +01:00
// Does not exist for PUS C
return 0;
2021-06-14 11:44:39 +02:00
#else
2022-02-02 10:29:30 +01:00
return (tcData->dataField.versionTypeAck & 0b10000000) >> 7;
2021-06-14 11:44:39 +02:00
#endif
2021-06-13 16:29:13 +02:00
}
2021-06-14 11:16:56 +02:00
uint8_t TcPacketPus::getPusVersionNumber() const {
2021-06-14 11:44:39 +02:00
#if FSFW_USE_PUS_C_TELECOMMANDS == 1
2022-02-02 10:29:30 +01:00
return (tcData->dataField.versionTypeAck & 0b11110000) >> 4;
2021-06-14 11:44:39 +02:00
#else
2022-02-02 10:29:30 +01:00
return (tcData->dataField.versionTypeAck & 0b01110000) >> 4;
2021-06-14 11:44:39 +02:00
#endif
2021-06-13 16:29:13 +02:00
}
2021-06-14 11:16:56 +02:00
uint16_t TcPacketPus::getSourceId() const {
#if FSFW_USE_PUS_C_TELECOMMANDS == 1
2022-02-02 10:29:30 +01:00
return (tcData->dataField.sourceIdH << 8) | tcData->dataField.sourceIdL;
2021-06-14 11:16:56 +02:00
#else
2022-02-02 10:29:30 +01:00
return tcData->dataField.sourceId;
2021-06-14 11:16:56 +02:00
#endif
2021-06-13 16:29:13 +02:00
}
2021-06-14 11:16:56 +02:00
size_t TcPacketPus::calculateFullPacketLength(size_t appDataLen) const {
2022-02-02 10:29:30 +01:00
return sizeof(CCSDSPrimaryHeader) + sizeof(PUSTcDataFieldHeader) + appDataLen +
TcPacketPusBase::CRC_SIZE;
2021-06-13 16:29:13 +02:00
}
ReturnValue_t TcPacketPus::setData(uint8_t *dataPtr, size_t maxSize, void *args) {
2022-07-18 10:20:26 +02:00
ReturnValue_t result = SpacePacketBase::setData(dataPtr, maxSize, args);
2022-02-02 10:29:30 +01:00
if (result != HasReturnvaluesIF::RETURN_OK) {
return result;
}
if (maxSize < sizeof(TcPacketPointer)) {
return HasReturnvaluesIF::RETURN_FAILED;
}
// This function is const-correct, but it was decided to keep the pointer non-const
// for convenience. Therefore, cast away constness here and then cast to packet type.
tcData = reinterpret_cast<TcPacketPointer *>(const_cast<uint8_t *>(dataPtr));
return HasReturnvaluesIF::RETURN_OK;
}