#define MS_CLASS "RTC::SCTP::DataChunk"
#include "RTC/SCTP/packet/chunks/DataChunk.hpp"
#include "Logger.hpp"
#include "MediaSoupErrors.hpp"
#include "RTC/SCTP/packet/Chunk.hpp"
namespace RTC
{
namespace SCTP
{
DataChunk* DataChunk::Parse(const uint8_t* buffer, size_t bufferLength)
{
MS_TRACE();
Chunk::ChunkType chunkType;
uint16_t chunkLength;
uint8_t padding;
if (!Chunk::IsChunk(buffer, bufferLength, chunkType, chunkLength, padding))
{
return nullptr;
}
if (chunkType != Chunk::ChunkType::DATA)
{
MS_WARN_DEV("invalid Chunk type");
return nullptr;
}
return DataChunk::ParseStrict(buffer, bufferLength, chunkLength, padding);
}
DataChunk* DataChunk::Factory(uint8_t* buffer, size_t bufferLength)
{
MS_TRACE();
if (bufferLength < DataChunk::DataChunkHeaderLength)
{
MS_THROW_TYPE_ERROR("buffer too small");
}
auto* chunk = new DataChunk(buffer, bufferLength);
chunk->InitializeHeader(Chunk::ChunkType::DATA, 0, DataChunk::DataChunkHeaderLength);
chunk->SetTsn(0);
chunk->SetStreamId(0);
chunk->SetStreamSequenceNumber(0);
chunk->SetPayloadProtocolId(0);
return chunk;
}
DataChunk* DataChunk::ParseStrict(
const uint8_t* buffer, size_t bufferLength, uint16_t chunkLength, uint8_t padding)
{
MS_TRACE();
if (chunkLength < DataChunk::DataChunkHeaderLength)
{
MS_WARN_TAG(
sctp,
"DataChunk Length field must be equal or greater than %zu",
DataChunk::DataChunkHeaderLength);
return nullptr;
}
auto* chunk = new DataChunk(const_cast<uint8_t*>(buffer), bufferLength);
chunk->SetLength(chunkLength + padding);
return chunk;
}
DataChunk::DataChunk(uint8_t* buffer, size_t bufferLength) : AnyDataChunk(buffer, bufferLength)
{
MS_TRACE();
SetLength(DataChunk::DataChunkHeaderLength);
}
DataChunk::~DataChunk()
{
MS_TRACE();
}
void DataChunk::Dump(int indentation) const
{
MS_TRACE();
MS_DUMP_CLEAN(indentation, "<SCTP::DataChunk>");
DumpCommon(indentation);
MS_DUMP_CLEAN(indentation, " flag I: %" PRIu8, GetI());
MS_DUMP_CLEAN(indentation, " flag U: %" PRIu8, GetU());
MS_DUMP_CLEAN(indentation, " flag B: %" PRIu8, GetB());
MS_DUMP_CLEAN(indentation, " flag E: %" PRIu8, GetE());
MS_DUMP_CLEAN(indentation, " tsn: %" PRIu32, GetTsn());
MS_DUMP_CLEAN(indentation, " stream id: %" PRIu16, GetStreamId());
MS_DUMP_CLEAN(indentation, " stream sequence number: %" PRIu16, GetStreamSequenceNumber());
MS_DUMP_CLEAN(indentation, " payload protocol id (PPID): %" PRIu32, GetPayloadProtocolId());
MS_DUMP_CLEAN(
indentation,
" user data length: %" PRIu16 " (has user data: %s)",
GetUserDataPayloadLength(),
HasUserDataPayload() ? "yes" : "no");
MS_DUMP_CLEAN(indentation, "</SCTP::DataChunk>");
}
DataChunk* DataChunk::Clone(uint8_t* buffer, size_t bufferLength) const
{
MS_TRACE();
auto* clonedChunk = new DataChunk(buffer, bufferLength);
CloneInto(clonedChunk);
SoftCloneInto(clonedChunk);
return clonedChunk;
}
void DataChunk::SetI(bool flag)
{
MS_TRACE();
SetBit3(flag);
}
void DataChunk::SetU(bool flag)
{
MS_TRACE();
SetBit2(flag);
}
void DataChunk::SetB(bool flag)
{
MS_TRACE();
SetBit1(flag);
}
void DataChunk::SetE(bool flag)
{
MS_TRACE();
SetBit0(flag);
}
void DataChunk::SetTsn(uint32_t value)
{
MS_TRACE();
Utils::Byte::Set4Bytes(const_cast<uint8_t*>(GetBuffer()), 4, value);
}
void DataChunk::SetStreamId(uint16_t value)
{
MS_TRACE();
Utils::Byte::Set2Bytes(const_cast<uint8_t*>(GetBuffer()), 8, value);
}
void DataChunk::SetStreamSequenceNumber(uint16_t value)
{
MS_TRACE();
Utils::Byte::Set2Bytes(const_cast<uint8_t*>(GetBuffer()), 10, value);
}
void DataChunk::SetPayloadProtocolId(uint32_t value)
{
MS_TRACE();
Utils::Byte::Set4Bytes(const_cast<uint8_t*>(GetBuffer()), 12, value);
}
void DataChunk::SetUserDataPayload(const uint8_t* userDataPayload, uint16_t userDataPayloadLength)
{
MS_TRACE();
SetVariableLengthValue(userDataPayload, userDataPayloadLength);
}
DataChunk* DataChunk::SoftClone(const uint8_t* buffer) const
{
MS_TRACE();
auto* softClonedChunk = new DataChunk(const_cast<uint8_t*>(buffer), GetLength());
SoftCloneInto(softClonedChunk);
return softClonedChunk;
}
} }