#define MS_CLASS "RTC::SCTP::IForwardTsnChunk"
#include "RTC/SCTP/packet/chunks/IForwardTsnChunk.hpp"
#include "Logger.hpp"
#include "MediaSoupErrors.hpp"
#include "RTC/SCTP/packet/Chunk.hpp"
namespace RTC
{
namespace SCTP
{
IForwardTsnChunk* IForwardTsnChunk::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::I_FORWARD_TSN)
{
MS_WARN_DEV("invalid Chunk type");
return nullptr;
}
return IForwardTsnChunk::ParseStrict(buffer, bufferLength, chunkLength, padding);
}
IForwardTsnChunk* IForwardTsnChunk::Factory(uint8_t* buffer, size_t bufferLength)
{
MS_TRACE();
if (bufferLength < IForwardTsnChunk::IForwardTsnChunkHeaderLength)
{
MS_THROW_TYPE_ERROR("buffer too small");
}
auto* chunk = new IForwardTsnChunk(buffer, bufferLength);
chunk->InitializeHeader(
Chunk::ChunkType::I_FORWARD_TSN, 0, IForwardTsnChunk::IForwardTsnChunkHeaderLength);
chunk->SetNewCumulativeTsn(0);
return chunk;
}
IForwardTsnChunk* IForwardTsnChunk::ParseStrict(
const uint8_t* buffer, size_t bufferLength, uint16_t chunkLength, uint8_t padding)
{
MS_TRACE();
if (chunkLength < IForwardTsnChunk::IForwardTsnChunkHeaderLength)
{
MS_WARN_TAG(
sctp,
"IForwardTsnChunk Length field must be equal or greater than %zu",
IForwardTsnChunk::IForwardTsnChunkHeaderLength);
return nullptr;
}
if (chunkLength % 8 != 0)
{
MS_WARN_TAG(sctp, "wrong length (not multiple of 4)");
return nullptr;
}
auto* chunk = new IForwardTsnChunk(const_cast<uint8_t*>(buffer), bufferLength);
chunk->SetLength(chunkLength + padding);
return chunk;
}
IForwardTsnChunk::IForwardTsnChunk(uint8_t* buffer, size_t bufferLength)
: AnyForwardTsnChunk(buffer, bufferLength)
{
MS_TRACE();
SetLength(IForwardTsnChunk::IForwardTsnChunkHeaderLength);
}
IForwardTsnChunk::~IForwardTsnChunk()
{
MS_TRACE();
}
void IForwardTsnChunk::Dump(int indentation) const
{
MS_TRACE();
MS_DUMP_CLEAN(indentation, "<SCTP::IForwardTsnChunk>");
DumpCommon(indentation);
MS_DUMP_CLEAN(indentation, " new cumulative tsn: %" PRIu32, GetNewCumulativeTsn());
MS_DUMP_CLEAN(indentation, " number of skipped streams: %" PRIu16, GetNumberOfSkippedStreams());
MS_DUMP_CLEAN(indentation, " skipped streams:");
for (auto& skippedStream : GetSkippedStreams())
{
MS_DUMP_CLEAN(
indentation,
" - unordered:%s, stream id: %" PRIu16 ", mid:%" PRIu32,
skippedStream.unordered ? "yes" : "no",
skippedStream.streamId,
skippedStream.mid);
}
MS_DUMP_CLEAN(indentation, "</SCTP::IForwardTsnChunk>");
}
IForwardTsnChunk* IForwardTsnChunk::Clone(uint8_t* buffer, size_t bufferLength) const
{
MS_TRACE();
auto* clonedChunk = new IForwardTsnChunk(buffer, bufferLength);
CloneInto(clonedChunk);
SoftCloneInto(clonedChunk);
return clonedChunk;
}
void IForwardTsnChunk::SetNewCumulativeTsn(uint32_t value)
{
MS_TRACE();
Utils::Byte::Set4Bytes(const_cast<uint8_t*>(GetBuffer()), 4, value);
}
std::vector<AnyForwardTsnChunk::SkippedStream> IForwardTsnChunk::GetSkippedStreams() const
{
MS_TRACE();
std::vector<AnyForwardTsnChunk::SkippedStream> skippedStreams;
const uint16_t numSkippedStreams = GetNumberOfSkippedStreams();
skippedStreams.reserve(numSkippedStreams);
for (uint16_t idx{ 0 }; idx < numSkippedStreams; ++idx)
{
skippedStreams.emplace_back(
GetUFlagAt(idx), GetSkippedStreamIdAt(idx), GetMessageIdentifierAt(idx));
}
return skippedStreams;
}
void IForwardTsnChunk::AddSkippedStream(const AnyForwardTsnChunk::SkippedStream& skippedStream)
{
MS_TRACE();
auto previousVariableLengthValueLength = GetVariableLengthValueLength();
SetVariableLengthValueLength(previousVariableLengthValueLength + 8);
Utils::Byte::Set2Bytes(
GetVariableLengthValuePointer(), previousVariableLengthValueLength, skippedStream.streamId);
Utils::Byte::Set2Bytes(
GetVariableLengthValuePointer(),
previousVariableLengthValueLength + 2,
skippedStream.unordered);
Utils::Byte::Set4Bytes(
GetVariableLengthValuePointer(), previousVariableLengthValueLength + 4, skippedStream.mid);
}
IForwardTsnChunk* IForwardTsnChunk::SoftClone(const uint8_t* buffer) const
{
MS_TRACE();
auto* softClonedChunk = new IForwardTsnChunk(const_cast<uint8_t*>(buffer), GetLength());
SoftCloneInto(softClonedChunk);
return softClonedChunk;
}
} }