#if RTC_ENABLE_MEDIA
#include "rtcpsrreporter.hpp"
#include <cassert>
#include <chrono>
#include <cmath>
namespace {
uint64_t ntp_time() {
const auto now = std::chrono::system_clock::now();
const double secs = std::chrono::duration<double>(now.time_since_epoch()).count();
return uint64_t(std::floor((secs + 2208988800.) * double(uint64_t(1) << 32)));
}
}
namespace rtc {
RtcpSrReporter::RtcpSrReporter(shared_ptr<RtpPacketizationConfig> rtpConfig)
: rtpConfig(rtpConfig) {
mLastReportedTimestamp = rtpConfig->timestamp;
}
void RtcpSrReporter::setNeedsToReport() { mNeedsToReport = true; }
uint32_t RtcpSrReporter::lastReportedTimestamp() const { return mLastReportedTimestamp; }
void RtcpSrReporter::outgoing(message_vector &messages, const message_callback &send) {
for (const auto &message : messages) {
if (message->type == Message::Control)
continue;
if (message->size() < sizeof(RtpHeader))
continue;
auto rtp = reinterpret_cast<RtpHeader *>(message->data());
addToReport(rtp, uint32_t(message->size()));
}
if (std::exchange(mNeedsToReport, false)) {
auto timestamp = rtpConfig->timestamp;
auto sr = getSenderReport(timestamp);
send(sr);
}
}
void RtcpSrReporter::addToReport(RtpHeader *rtp, uint32_t rtpSize) {
mPacketCount += 1;
assert(!rtp->padding());
mPayloadOctets += rtpSize - uint32_t(rtp->getSize());
}
message_ptr RtcpSrReporter::getSenderReport(uint32_t timestamp) {
auto srSize = RtcpSr::Size(0);
auto msg = make_message(srSize + RtcpSdes::Size({{uint8_t(rtpConfig->cname.size())}}),
Message::Control);
auto sr = reinterpret_cast<RtcpSr *>(msg->data());
sr->setNtpTimestamp(ntp_time());
sr->setRtpTimestamp(timestamp);
sr->setPacketCount(mPacketCount);
sr->setOctetCount(mPayloadOctets);
sr->preparePacket(rtpConfig->ssrc, 0);
auto sdes = reinterpret_cast<RtcpSdes *>(msg->data() + srSize);
auto chunk = sdes->getChunk(0);
chunk->setSSRC(rtpConfig->ssrc);
auto item = chunk->getItem(0);
item->type = 1;
item->setText(rtpConfig->cname);
sdes->preparePacket(1);
mLastReportedTimestamp = timestamp;
return msg;
}
}
#endif