#define MS_CLASS "RTC::RTCP::FeedbackPsRemb"
#include "RTC/RTCP/FeedbackPsRemb.hpp"
#include "Logger.hpp"
#include "Utils.hpp"
#include <cstring>
namespace RTC
{
namespace RTCP
{
FeedbackPsRembPacket* FeedbackPsRembPacket::Parse(const uint8_t* data, size_t len)
{
MS_TRACE();
if (len < Packet::CommonHeaderSize + FeedbackPacket::HeaderSize + 8u)
{
MS_WARN_TAG(rtcp, "not enough space for Feedback packet, discarded");
return nullptr;
}
auto* commonHeader = const_cast<CommonHeader*>(reinterpret_cast<const CommonHeader*>(data));
std::unique_ptr<FeedbackPsRembPacket> packet(new FeedbackPsRembPacket(commonHeader, len));
if (!packet->IsCorrect())
{
return nullptr;
}
return packet.release();
}
FeedbackPsRembPacket::FeedbackPsRembPacket(CommonHeader* commonHeader, size_t availableLen)
: FeedbackPsAfbPacket(commonHeader, FeedbackPsAfbPacket::Application::REMB)
{
const size_t len = static_cast<size_t>(ntohs(commonHeader->length) + 1) * 4;
if (len > availableLen)
{
MS_WARN_TAG(rtcp, "packet announced length exceeds the available buffer length, discarded");
this->isCorrect = false;
return;
}
auto* data = reinterpret_cast<uint8_t*>(commonHeader) + Packet::CommonHeaderSize +
FeedbackPacket::HeaderSize;
const size_t numSsrcs = data[4];
if (len != Packet::CommonHeaderSize + FeedbackPacket::HeaderSize + 8u + (numSsrcs * 4u))
{
MS_WARN_TAG(
rtcp, "invalid payload size (%zu bytes) for the given number of ssrcs (%zu)", len, numSsrcs);
this->isCorrect = false;
return;
}
if (Utils::Byte::Get4Bytes(data, 0) != FeedbackPsRembPacket::UniqueIdentifier)
{
MS_WARN_TAG(rtcp, "invalid unique indentifier in REMB packet");
this->isCorrect = false;
return;
}
const uint8_t exponent = data[5] >> 2;
const uint64_t mantissa =
(static_cast<uint32_t>(data[5] & 0x03) << 16) | Utils::Byte::Get2Bytes(data, 6);
this->bitrate = (mantissa << exponent);
if ((this->bitrate >> exponent) != mantissa)
{
MS_WARN_TAG(rtcp, "invalid REMB bitrate value: %" PRIu64 " *2^%u", mantissa, exponent);
this->isCorrect = false;
return;
}
size_t index{ 8 };
this->ssrcs.reserve(numSsrcs);
for (size_t n{ 0 }; n < numSsrcs; ++n)
{
this->ssrcs.push_back(Utils::Byte::Get4Bytes(data, index));
index += 4u;
}
}
size_t FeedbackPsRembPacket::Serialize(uint8_t* buffer)
{
MS_TRACE();
size_t offset = FeedbackPsPacket::Serialize(buffer);
uint64_t mantissa = this->bitrate;
uint8_t exponent{ 0u };
while (mantissa > 0x3FFFF )
{
mantissa >>= 1;
++exponent;
}
Utils::Byte::Set4Bytes(buffer, offset, FeedbackPsRembPacket::UniqueIdentifier);
offset += FeedbackPsRembPacket::UniqueIdentifierSize;
buffer[offset] = this->ssrcs.size();
offset += 1;
buffer[offset] = (exponent << 2) | (mantissa >> 16);
offset += 1;
Utils::Byte::Set2Bytes(buffer, offset, mantissa & 0xFFFF);
offset += 2;
for (auto ssrc : this->ssrcs)
{
Utils::Byte::Set4Bytes(buffer, offset, ssrc);
offset += 4u;
}
return offset;
}
void FeedbackPsRembPacket::Dump(int indentation) const
{
MS_TRACE();
MS_DUMP_CLEAN(indentation, "<FeedbackPsRembPacket>");
FeedbackPsPacket::Dump();
MS_DUMP_CLEAN(indentation, " bitrate (bps): %" PRIu64, this->bitrate);
for (auto ssrc : this->ssrcs)
{
MS_DUMP_CLEAN(indentation, " ssrc: %" PRIu32, ssrc);
}
MS_DUMP_CLEAN(indentation, "</FeedbackPsRembPacket>");
}
} }