#define MS_CLASS "RTC::RTCP::FeedbackPsRemb"
#include "RTC/RTCP/FeedbackPsRemb.hpp"
#include "Logger.hpp"
#include "Utils.hpp"
#include <cstring>
namespace RTC
{
namespace RTCP
{
uint32_t FeedbackPsRembPacket::uniqueIdentifier{ 0x52454D42 };
FeedbackPsRembPacket* FeedbackPsRembPacket::Parse(const uint8_t* data, size_t len)
{
MS_TRACE();
if (len < sizeof(CommonHeader) + sizeof(FeedbackPacket::Header) + 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)
{
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) + sizeof(CommonHeader) +
sizeof(FeedbackPacket::Header);
size_t numSsrcs = data[4];
if (len != sizeof(CommonHeader) + sizeof(FeedbackPacket::Header) + 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;
}
uint8_t exponent = data[5] >> 2;
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 += sizeof(FeedbackPsRembPacket::uniqueIdentifier);
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() const
{
MS_TRACE();
MS_DUMP("<FeedbackPsRembPacket>");
FeedbackPsPacket::Dump();
MS_DUMP(" bitrate (bps) : %" PRIu64, this->bitrate);
for (auto ssrc : this->ssrcs)
{
MS_DUMP(" ssrc : %" PRIu32, ssrc);
}
MS_DUMP("</FeedbackPsRembPacket>");
}
} }