Files
ZLMediaKit/src/Rtcp/Rtcp.cpp

883 lines
30 KiB
C++
Raw Normal View History

2021-01-31 20:46:55 +08:00
/*
2023-12-09 16:23:51 +08:00
* Copyright (c) 2016-present The ZLMediaKit project authors. All Rights Reserved.
2021-01-31 19:18:17 +08:00
*
2023-12-09 16:23:51 +08:00
* This file is part of ZLMediaKit(https://github.com/ZLMediaKit/ZLMediaKit).
2021-01-31 19:18:17 +08:00
*
2023-12-09 16:23:51 +08:00
* Use of this source code is governed by MIT-like license that can be found in the
2021-01-31 19:18:17 +08:00
* LICENSE file in the root of the source tree. All contributing project authors
* may be found in the AUTHORS file in the root of the source tree.
*/
#include "Rtcp.h"
2021-04-26 21:03:04 +08:00
#include "RtcpFCI.h"
2022-07-14 22:49:31 +08:00
#include "Util/logger.h"
#include <assert.h>
#include <stddef.h>
2021-01-31 19:18:17 +08:00
using namespace std;
using namespace toolkit;
2021-01-31 19:18:17 +08:00
namespace mediakit {
2022-07-14 22:49:31 +08:00
const char *rtcpTypeToStr(RtcpType type) {
switch (type) {
#define SWITCH_CASE(key, value) \
2022-08-05 17:55:48 +08:00
case RtcpType::key: return #value "(" #key ")";
2021-01-31 19:18:17 +08:00
RTCP_PT_MAP(SWITCH_CASE)
#undef SWITCH_CASE
2022-08-05 17:55:48 +08:00
default: return "unknown rtcp pt";
2021-01-31 19:18:17 +08:00
}
}
2022-07-14 22:49:31 +08:00
const char *sdesTypeToStr(SdesType type) {
switch (type) {
#define SWITCH_CASE(key, value) \
2022-08-05 17:55:48 +08:00
case SdesType::key: return #value "(" #key ")";
2021-01-31 19:18:17 +08:00
SDES_TYPE_MAP(SWITCH_CASE)
#undef SWITCH_CASE
2022-08-05 17:55:48 +08:00
default: return "unknown source description type";
2021-01-31 19:18:17 +08:00
}
}
2021-04-13 18:50:20 +08:00
const char *psfbTypeToStr(PSFBType type) {
2022-07-14 22:49:31 +08:00
switch (type) {
#define SWITCH_CASE(key, value) \
2022-08-05 17:55:48 +08:00
case PSFBType::key: return #value "(" #key ")";
2021-04-13 18:50:20 +08:00
PSFB_TYPE_MAP(SWITCH_CASE)
#undef SWITCH_CASE
2022-08-05 17:55:48 +08:00
default: return "unknown payload-specific fb message fmt type";
2021-04-13 18:50:20 +08:00
}
2021-04-23 15:25:33 +08:00
}
const char *rtpfbTypeToStr(RTPFBType type) {
2022-07-14 22:49:31 +08:00
switch (type) {
#define SWITCH_CASE(key, value) \
2022-08-05 17:55:48 +08:00
case RTPFBType::key: return #value "(" #key ")";
2021-04-23 15:25:33 +08:00
RTPFB_TYPE_MAP(SWITCH_CASE)
#undef SWITCH_CASE
2022-08-05 17:55:48 +08:00
default: return "unknown transport layer feedback messages fmt type";
2021-04-23 15:25:33 +08:00
}
2021-04-13 18:50:20 +08:00
}
2021-01-31 19:18:17 +08:00
static size_t alignSize(size_t bytes) {
2022-07-14 22:49:31 +08:00
return (size_t)((bytes + 3) >> 2) << 2;
2021-01-31 19:18:17 +08:00
}
static void setupHeader(RtcpHeader *rtcp, RtcpType type, size_t report_count, size_t total_bytes) {
rtcp->version = 2;
rtcp->padding = 0;
if (report_count > 0x1F) {
throw std::invalid_argument(StrPrinter << "rtcp report_count最大赋值为31,当前为:" << report_count);
}
// items总个数 [AUTO-TRANSLATED:2d40d010]
// Total number of items
2021-01-31 19:18:17 +08:00
rtcp->report_count = report_count;
2022-07-14 22:49:31 +08:00
rtcp->pt = (uint8_t)type;
2021-04-23 18:27:47 +08:00
rtcp->setSize(total_bytes);
2021-01-31 19:18:17 +08:00
}
2021-04-26 21:50:10 +08:00
static void setupPadding(RtcpHeader *rtcp, size_t padding_size) {
if (padding_size) {
rtcp->padding = 1;
2022-07-14 22:49:31 +08:00
((uint8_t *)rtcp)[rtcp->getSize() - 1] = padding_size & 0xFF;
2021-04-26 21:50:10 +08:00
} else {
rtcp->padding = 0;
}
}
2021-01-31 19:18:17 +08:00
/////////////////////////////////////////////////////////////////////////////
2021-12-30 15:28:02 +08:00
string RtcpHeader::dumpHeader() const {
2021-01-31 19:18:17 +08:00
_StrPrinter printer;
printer << "version:" << version << "\r\n";
2021-04-26 21:50:10 +08:00
if (padding) {
printer << "padding:" << padding << " " << getPaddingSize() << "\r\n";
} else {
printer << "padding:" << padding << "\r\n";
}
2022-07-14 22:49:31 +08:00
switch ((RtcpType)pt) {
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_RTPFB: {
printer << "report_count:" << rtpfbTypeToStr((RTPFBType)report_count) << "\r\n";
break;
}
case RtcpType::RTCP_PSFB: {
printer << "report_count:" << psfbTypeToStr((PSFBType)report_count) << "\r\n";
break;
}
default: {
printer << "report_count:" << report_count << "\r\n";
break;
}
2021-04-23 18:27:47 +08:00
}
2022-07-14 22:49:31 +08:00
printer << "pt:" << rtcpTypeToStr((RtcpType)pt) << "\r\n";
2021-04-23 18:27:47 +08:00
printer << "size:" << getSize() << "\r\n";
2021-01-31 19:18:17 +08:00
printer << "--------\r\n";
2021-01-31 19:18:50 +08:00
return std::move(printer);
2021-01-31 19:18:17 +08:00
}
string RtcpHeader::dumpString() const {
2022-07-14 22:49:31 +08:00
switch ((RtcpType)pt) {
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_SR: {
RtcpSR *rtcp = (RtcpSR *)this;
return rtcp->dumpString();
}
2021-01-31 19:18:17 +08:00
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_RR: {
RtcpRR *rtcp = (RtcpRR *)this;
return rtcp->dumpString();
}
2021-01-31 19:18:17 +08:00
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_SDES: {
RtcpSdes *rtcp = (RtcpSdes *)this;
return rtcp->dumpString();
}
2021-04-03 09:34:49 +08:00
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_RTPFB:
case RtcpType::RTCP_PSFB: {
RtcpFB *rtcp = (RtcpFB *)this;
return rtcp->dumpString();
}
2021-04-07 16:45:50 +08:00
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_BYE: {
RtcpBye *rtcp = (RtcpBye *)this;
return rtcp->dumpString();
}
2021-04-07 16:45:50 +08:00
2022-08-05 17:55:48 +08:00
default: return StrPrinter << dumpHeader() << hexdump((char *)this + sizeof(*this), getSize() - sizeof(*this));
2021-01-31 19:18:17 +08:00
}
}
2021-04-07 16:45:50 +08:00
size_t RtcpHeader::getSize() const {
// 加上rtcp头长度 [AUTO-TRANSLATED:21a40b4b]
// Add rtcp header length
2021-04-23 18:27:47 +08:00
return (1 + ntohs(length)) << 2;
}
2022-07-14 22:49:31 +08:00
size_t RtcpHeader::getPaddingSize() const {
2021-04-26 21:50:10 +08:00
if (!padding) {
return 0;
}
2022-07-14 22:49:31 +08:00
return ((uint8_t *)this)[getSize() - 1];
2021-04-26 21:50:10 +08:00
}
2021-04-23 18:27:47 +08:00
void RtcpHeader::setSize(size_t size) {
// 不包含rtcp头的长度 [AUTO-TRANSLATED:b26ad8ef]
// Length excluding rtcp header
2022-07-14 22:49:31 +08:00
length = htons((uint16_t)((size >> 2) - 1));
2021-04-07 16:45:50 +08:00
}
2021-12-30 15:28:02 +08:00
void RtcpHeader::net2Host(size_t len) {
2022-07-14 22:49:31 +08:00
switch ((RtcpType)pt) {
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_SR: {
RtcpSR *sr = (RtcpSR *)this;
sr->net2Host(len);
break;
}
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_RR: {
RtcpRR *rr = (RtcpRR *)this;
rr->net2Host(len);
break;
}
case RtcpType::RTCP_SDES: {
RtcpSdes *sdes = (RtcpSdes *)this;
sdes->net2Host(len);
break;
}
case RtcpType::RTCP_RTPFB:
case RtcpType::RTCP_PSFB: {
RtcpFB *fb = (RtcpFB *)this;
fb->net2Host(len);
break;
}
case RtcpType::RTCP_BYE: {
RtcpBye *bye = (RtcpBye *)this;
bye->net2Host(len);
break;
}
case RtcpType::RTCP_XR: {
RtcpXRRRTR *xr = (RtcpXRRRTR *)this;
if (xr->bt == 4) {
xr->net2Host(len);
// TraceL<<xr->dumpString();
} else if (xr->bt == 5) {
RtcpXRDLRR *dlrr = (RtcpXRDLRR *)this;
dlrr->net2Host(len);
TraceL << dlrr->dumpString();
} else if (xr->bt == 42){
//当有浏览器将屏幕推流到服务器时会发生这个, 暂时没发现什么作用,先解析出来,不做处理
RtcpXRTargetBitrate* tb = (RtcpXRTargetBitrate *)this;
tb->net2Host(len);
//TraceL << tb->dumpString();
2022-08-05 17:55:48 +08:00
} else {
throw std::runtime_error(StrPrinter << "rtcp xr bt " << (int)xr->bt << " not support");
2022-08-05 17:55:48 +08:00
}
break;
}
default: throw std::runtime_error(StrPrinter << "未处理的rtcp包:" << rtcpTypeToStr((RtcpType)this->pt));
2021-01-31 19:18:17 +08:00
}
}
2021-12-30 15:28:02 +08:00
vector<RtcpHeader *> RtcpHeader::loadFromBytes(char *data, size_t len) {
2021-01-31 19:18:17 +08:00
vector<RtcpHeader *> ret;
ssize_t remain = len;
char *ptr = data;
2022-07-14 22:49:31 +08:00
while (remain > (ssize_t)sizeof(RtcpHeader)) {
RtcpHeader *rtcp = (RtcpHeader *)ptr;
2021-04-23 18:27:47 +08:00
auto rtcp_len = rtcp->getSize();
2022-07-14 22:49:31 +08:00
if (remain < (ssize_t)rtcp_len) {
2021-04-07 16:45:50 +08:00
WarnL << "非法的rtcp包,声明的长度超过实际数据长度";
break;
}
2021-01-31 19:18:17 +08:00
try {
rtcp->net2Host(rtcp_len);
ret.emplace_back(rtcp);
} catch (std::exception &ex) {
// 不能处理的rtcp包或者无法解析的rtcp包忽略掉 [AUTO-TRANSLATED:752ec400]
// Ignore unprocessable rtcp packets or rtcp packets that cannot be parsed
2021-04-03 09:34:49 +08:00
WarnL << ex.what() << ",长度为:" << rtcp_len;
2021-01-31 19:18:17 +08:00
}
ptr += rtcp_len;
remain -= rtcp_len;
}
return ret;
}
class BufferRtcp : public Buffer {
public:
2022-07-14 22:49:31 +08:00
BufferRtcp(std::shared_ptr<RtcpHeader> rtcp) { _rtcp = std::move(rtcp); }
char *data() const override { return (char *)_rtcp.get(); }
size_t size() const override { return _rtcp->getSize(); }
2021-01-31 19:18:17 +08:00
private:
std::shared_ptr<RtcpHeader> _rtcp;
};
Buffer::Ptr RtcpHeader::toBuffer(std::shared_ptr<RtcpHeader> rtcp) {
return std::make_shared<BufferRtcp>(std::move(rtcp));
}
/////////////////////////////////////////////////////////////////////////////
std::shared_ptr<RtcpSR> RtcpSR::create(size_t item_count) {
2021-04-26 21:50:10 +08:00
auto real_size = sizeof(RtcpSR) - sizeof(ReportItem) + item_count * sizeof(ReportItem);
auto bytes = alignSize(real_size);
2022-07-14 22:49:31 +08:00
auto ptr = (RtcpSR *)new char[bytes];
2021-01-31 19:18:17 +08:00
setupHeader(ptr, RtcpType::RTCP_SR, item_count, bytes);
2021-04-26 21:50:10 +08:00
setupPadding(ptr, bytes - real_size);
2022-08-05 17:55:48 +08:00
return std::shared_ptr<RtcpSR>(ptr, [](RtcpSR *ptr) { delete[](char *) ptr; });
2021-01-31 19:18:17 +08:00
}
2021-12-30 15:28:02 +08:00
string RtcpSR::getNtpStamp() const {
2021-01-31 19:18:17 +08:00
struct timeval tv;
tv.tv_sec = ntpmsw - 0x83AA7E80;
2022-07-14 22:49:31 +08:00
tv.tv_usec = (decltype(tv.tv_usec))(ntplsw / ((double)(((uint64_t)1) << 32) * 1.0e-6));
2021-01-31 19:18:17 +08:00
return LogChannel::printTime(tv);
}
uint64_t RtcpSR::getNtpUnixStampMS() const {
2021-10-20 11:54:05 +08:00
if (ntpmsw < 0x83AA7E80) {
// ntp时间戳起始时间为1900年但是utc时间戳起始时间为1970年两者相差0x83AA7E80秒 [AUTO-TRANSLATED:6b3ac2fa]
// The ntp timestamp starts from 1900, but the utc timestamp starts from 1970, with a difference of 0x83AA7E80 seconds
// ntp时间戳不得早于1970年否则无法转换为utc时间戳 [AUTO-TRANSLATED:d70fc88c]
// The ntp timestamp must not be earlier than 1970, otherwise it cannot be converted to utc timestamp
2021-10-20 11:54:05 +08:00
return 0;
}
struct timeval tv;
tv.tv_sec = ntpmsw - 0x83AA7E80;
2022-07-14 22:49:31 +08:00
tv.tv_usec = (decltype(tv.tv_usec))(ntplsw / ((double)(((uint64_t)1) << 32) * 1.0e-6));
return (uint64_t)1000 * tv.tv_sec + tv.tv_usec / 1000;
}
2021-01-31 19:18:17 +08:00
void RtcpSR::setNtpStamp(struct timeval tv) {
ntpmsw = htonl(tv.tv_sec + 0x83AA7E80); /* 0x83AA7E80 is the number of seconds from 1900 to 1970 */
2022-07-14 22:49:31 +08:00
ntplsw = htonl((uint32_t)((double)tv.tv_usec * (double)(((uint64_t)1) << 32) * 1.0e-6));
2021-01-31 19:18:17 +08:00
}
void RtcpSR::setNtpStamp(uint64_t unix_stamp_ms) {
struct timeval tv;
tv.tv_sec = unix_stamp_ms / 1000;
tv.tv_usec = (unix_stamp_ms % 1000) * 1000;
setNtpStamp(tv);
}
2021-12-30 15:28:02 +08:00
string RtcpSR::dumpString() const {
2021-01-31 19:18:17 +08:00
_StrPrinter printer;
printer << RtcpHeader::dumpHeader();
printer << "ssrc:" << ssrc << "\r\n";
printer << "ntpmsw:" << ntpmsw << "\r\n";
printer << "ntplsw:" << ntplsw << "\r\n";
printer << "ntp time:" << getNtpStamp() << "\r\n";
printer << "rtpts:" << rtpts << "\r\n";
printer << "packet_count:" << packet_count << "\r\n";
printer << "octet_count:" << octet_count << "\r\n";
2022-07-14 22:49:31 +08:00
auto items = ((RtcpSR *)this)->getItemList();
2021-01-31 19:18:17 +08:00
auto i = 0;
for (auto &item : items) {
printer << "---- item:" << i++ << " ----\r\n";
printer << item->dumpString();
}
2021-01-31 19:18:50 +08:00
return std::move(printer);
2021-01-31 19:18:17 +08:00
}
2022-07-14 22:49:31 +08:00
#define CHECK_MIN_SIZE(size, kMinSize) \
if (size < kMinSize) { \
throw std::out_of_range( \
StrPrinter << rtcpTypeToStr((RtcpType)pt) << " 长度不足:" << size << " < " << kMinSize); \
}
2021-01-31 19:18:17 +08:00
2022-07-14 22:49:31 +08:00
#define CHECK_REPORT_COUNT(item_count) \
/*修正个数防止getItemList时内存越界
/*Correct the number to prevent memory overflow when getItemList
* [AUTO-TRANSLATED:852bd70e]
*/ \
2022-07-14 22:49:31 +08:00
if (report_count != item_count) { \
WarnL << rtcpTypeToStr((RtcpType)pt) << " report_count 字段不正确,已修正为:" << (int)report_count << " -> " \
<< item_count; \
report_count = item_count; \
}
2021-01-31 19:18:17 +08:00
void RtcpSR::net2Host(size_t size) {
static const size_t kMinSize = sizeof(RtcpSR) - sizeof(items);
CHECK_MIN_SIZE(size, kMinSize);
ssrc = ntohl(ssrc);
ntpmsw = ntohl(ntpmsw);
ntplsw = ntohl(ntplsw);
rtpts = ntohl(rtpts);
packet_count = ntohl(packet_count);
octet_count = ntohl(octet_count);
ReportItem *ptr = &items;
int item_count = 0;
2022-07-14 22:49:31 +08:00
for (int i = 0; i < (int)report_count && (char *)(ptr) + sizeof(ReportItem) <= (char *)(this) + size; ++i) {
2021-01-31 19:18:17 +08:00
ptr->net2Host();
++ptr;
++item_count;
}
2021-04-23 18:27:47 +08:00
CHECK_REPORT_COUNT(item_count);
2021-01-31 19:18:17 +08:00
}
2021-12-30 15:28:02 +08:00
vector<ReportItem *> RtcpSR::getItemList() {
2021-01-31 19:18:17 +08:00
vector<ReportItem *> ret;
ReportItem *ptr = &items;
2022-07-14 22:49:31 +08:00
for (int i = 0; i < (int)report_count; ++i) {
2021-01-31 19:18:17 +08:00
ret.emplace_back(ptr);
++ptr;
}
return ret;
}
/////////////////////////////////////////////////////////////////////////////
2021-12-30 15:28:02 +08:00
string ReportItem::dumpString() const {
2021-01-31 19:18:17 +08:00
_StrPrinter printer;
printer << "ssrc:" << ssrc << "\r\n";
printer << "fraction:" << fraction << "\r\n";
printer << "cumulative:" << cumulative << "\r\n";
printer << "seq_cycles:" << seq_cycles << "\r\n";
printer << "seq_max:" << seq_max << "\r\n";
printer << "jitter:" << jitter << "\r\n";
printer << "last_sr_stamp:" << last_sr_stamp << "\r\n";
printer << "delay_since_last_sr:" << delay_since_last_sr << "\r\n";
2021-01-31 19:18:50 +08:00
return std::move(printer);
2021-01-31 19:18:17 +08:00
}
void ReportItem::net2Host() {
ssrc = ntohl(ssrc);
2021-12-30 15:28:02 +08:00
cumulative = ntohl(cumulative) >> 8;
2021-01-31 19:18:17 +08:00
seq_cycles = ntohs(seq_cycles);
seq_max = ntohs(seq_max);
jitter = ntohl(jitter);
last_sr_stamp = ntohl(last_sr_stamp);
delay_since_last_sr = ntohl(delay_since_last_sr);
}
/////////////////////////////////////////////////////////////////////////////
std::shared_ptr<RtcpRR> RtcpRR::create(size_t item_count) {
2021-04-26 21:50:10 +08:00
auto real_size = sizeof(RtcpRR) - sizeof(ReportItem) + item_count * sizeof(ReportItem);
auto bytes = alignSize(real_size);
2022-07-14 22:49:31 +08:00
auto ptr = (RtcpRR *)new char[bytes];
2021-01-31 19:18:17 +08:00
setupHeader(ptr, RtcpType::RTCP_RR, item_count, bytes);
2021-04-26 21:50:10 +08:00
setupPadding(ptr, bytes - real_size);
2022-08-05 17:55:48 +08:00
return std::shared_ptr<RtcpRR>(ptr, [](RtcpRR *ptr) { delete[](char *) ptr; });
2021-01-31 19:18:17 +08:00
}
2021-12-30 15:28:02 +08:00
string RtcpRR::dumpString() const {
2021-01-31 19:18:17 +08:00
_StrPrinter printer;
printer << RtcpHeader::dumpHeader();
printer << "ssrc:" << ssrc << "\r\n";
2022-07-14 22:49:31 +08:00
auto items = ((RtcpRR *)this)->getItemList();
2021-01-31 19:18:17 +08:00
auto i = 0;
for (auto &item : items) {
printer << "---- item:" << i++ << " ----\r\n";
printer << item->dumpString();
}
2021-01-31 19:18:50 +08:00
return std::move(printer);
2021-01-31 19:18:17 +08:00
}
void RtcpRR::net2Host(size_t size) {
static const size_t kMinSize = sizeof(RtcpRR) - sizeof(items);
CHECK_MIN_SIZE(size, kMinSize);
ssrc = ntohl(ssrc);
ReportItem *ptr = &items;
int item_count = 0;
2022-07-14 22:49:31 +08:00
for (int i = 0; i < (int)report_count && (char *)(ptr) + sizeof(ReportItem) <= (char *)(this) + size; ++i) {
2021-01-31 19:18:17 +08:00
ptr->net2Host();
++ptr;
++item_count;
}
2021-04-23 18:27:47 +08:00
CHECK_REPORT_COUNT(item_count);
2021-01-31 19:18:17 +08:00
}
2021-12-30 15:28:02 +08:00
vector<ReportItem *> RtcpRR::getItemList() {
2021-01-31 19:18:17 +08:00
vector<ReportItem *> ret;
ReportItem *ptr = &items;
2022-07-14 22:49:31 +08:00
for (int i = 0; i < (int)report_count; ++i) {
2021-01-31 19:18:17 +08:00
ret.emplace_back(ptr);
++ptr;
}
return ret;
}
/////////////////////////////////////////////////////////////////////////////
2021-07-15 17:38:04 +08:00
void SdesChunk::net2Host() {
2021-01-31 19:18:17 +08:00
ssrc = ntohl(ssrc);
}
2021-12-30 15:28:02 +08:00
size_t SdesChunk::totalBytes() const {
2021-04-23 18:27:47 +08:00
return alignSize(minSize() + txt_len);
2021-01-31 19:18:17 +08:00
}
2021-07-15 17:38:04 +08:00
size_t SdesChunk::minSize() {
return sizeof(SdesChunk) - sizeof(text);
2021-01-31 19:18:17 +08:00
}
2021-12-30 15:28:02 +08:00
string SdesChunk::dumpString() const {
2021-01-31 19:18:17 +08:00
_StrPrinter printer;
printer << "ssrc:" << ssrc << "\r\n";
2022-07-14 22:49:31 +08:00
printer << "type:" << sdesTypeToStr((SdesType)type) << "\r\n";
printer << "txt_len:" << (int)txt_len << "\r\n";
2021-04-23 18:27:47 +08:00
printer << "text:" << (txt_len ? string(text, txt_len) : "") << "\r\n";
2021-01-31 19:18:50 +08:00
return std::move(printer);
2021-01-31 19:18:17 +08:00
}
/////////////////////////////////////////////////////////////////////////////
2021-04-23 18:27:47 +08:00
std::shared_ptr<RtcpSdes> RtcpSdes::create(const std::vector<string> &item_text) {
2021-01-31 19:18:17 +08:00
size_t item_total_size = 0;
for (auto &text : item_text) {
// 统计所有SdesChunk对象占用的空间 [AUTO-TRANSLATED:87871205]
// Count the space occupied by all SdesChunk objects
2021-07-15 17:38:04 +08:00
item_total_size += alignSize(SdesChunk::minSize() + (0xFF & text.size()));
2021-01-31 19:18:17 +08:00
}
2021-07-15 17:38:04 +08:00
auto real_size = sizeof(RtcpSdes) - sizeof(SdesChunk) + item_total_size;
2021-04-26 21:50:10 +08:00
auto bytes = alignSize(real_size);
2022-07-14 22:49:31 +08:00
auto ptr = (RtcpSdes *)new char[bytes];
2021-10-15 14:09:32 +08:00
memset(ptr, 0x00, bytes);
2021-07-15 17:38:04 +08:00
auto item_ptr = &ptr->chunks;
2021-01-31 19:18:17 +08:00
for (auto &text : item_text) {
2021-04-23 18:27:47 +08:00
item_ptr->txt_len = (0xFF & text.size());
// 确保赋值\0为RTCP_SDES_END [AUTO-TRANSLATED:316be0a3]
// Ensure that the assignment \0 is RTCP_SDES_END
2021-04-23 18:27:47 +08:00
memcpy(item_ptr->text, text.data(), item_ptr->txt_len + 1);
2022-07-14 22:49:31 +08:00
item_ptr = (SdesChunk *)((char *)item_ptr + item_ptr->totalBytes());
2021-01-31 19:18:17 +08:00
}
setupHeader(ptr, RtcpType::RTCP_SDES, item_text.size(), bytes);
2021-04-26 21:50:10 +08:00
setupPadding(ptr, bytes - real_size);
2022-08-05 17:55:48 +08:00
return std::shared_ptr<RtcpSdes>(ptr, [](RtcpSdes *ptr) { delete[](char *) ptr; });
2021-01-31 19:18:17 +08:00
}
string RtcpSdes::dumpString() const {
_StrPrinter printer;
printer << RtcpHeader::dumpHeader();
2022-07-14 22:49:31 +08:00
auto items = ((RtcpSdes *)this)->getChunkList();
2021-01-31 19:18:17 +08:00
auto i = 0;
for (auto &item : items) {
printer << "---- item:" << i++ << " ----\r\n";
printer << item->dumpString();
}
2021-01-31 19:18:50 +08:00
return std::move(printer);
2021-01-31 19:18:17 +08:00
}
void RtcpSdes::net2Host(size_t size) {
2021-07-15 17:38:04 +08:00
static const size_t kMinSize = sizeof(RtcpSdes) - sizeof(chunks);
2021-01-31 19:18:17 +08:00
CHECK_MIN_SIZE(size, kMinSize);
2021-07-15 17:38:04 +08:00
SdesChunk *ptr = &chunks;
2021-01-31 19:18:17 +08:00
int item_count = 0;
2022-07-14 22:49:31 +08:00
for (int i = 0; i < (int)report_count && (char *)(ptr) + SdesChunk::minSize() <= (char *)(this) + size; ++i) {
2021-01-31 19:18:17 +08:00
ptr->net2Host();
2022-07-14 22:49:31 +08:00
ptr = (SdesChunk *)((char *)ptr + ptr->totalBytes());
2021-01-31 19:18:17 +08:00
++item_count;
}
2021-04-23 18:27:47 +08:00
CHECK_REPORT_COUNT(item_count);
2021-01-31 19:18:17 +08:00
}
2021-07-15 17:38:04 +08:00
vector<SdesChunk *> RtcpSdes::getChunkList() {
vector<SdesChunk *> ret;
SdesChunk *ptr = &chunks;
2022-07-14 22:49:31 +08:00
for (int i = 0; i < (int)report_count; ++i) {
2021-01-31 19:18:17 +08:00
ret.emplace_back(ptr);
2022-07-14 22:49:31 +08:00
ptr = (SdesChunk *)((char *)ptr + ptr->totalBytes());
2021-01-31 19:18:17 +08:00
}
return ret;
}
2021-04-03 09:34:49 +08:00
////////////////////////////////////////////////////////////////////
2021-04-23 18:27:47 +08:00
std::shared_ptr<RtcpFB> RtcpFB::create_l(RtcpType type, int fmt, const void *fci, size_t fci_len) {
if (!fci) {
fci_len = 0;
}
2021-04-26 21:50:10 +08:00
auto real_size = sizeof(RtcpFB) + fci_len;
auto bytes = alignSize(real_size);
2022-07-14 22:49:31 +08:00
auto ptr = (RtcpFB *)new char[bytes];
2021-04-23 18:27:47 +08:00
if (fci && fci_len) {
2022-07-14 22:49:31 +08:00
memcpy((char *)ptr + sizeof(RtcpFB), fci, fci_len);
2021-04-23 18:27:47 +08:00
}
setupHeader(ptr, type, fmt, bytes);
2021-04-26 21:50:10 +08:00
setupPadding(ptr, bytes - real_size);
2022-08-05 17:55:48 +08:00
return std::shared_ptr<RtcpFB>((RtcpFB *)ptr, [](RtcpFB *ptr) { delete[](char *) ptr; });
2021-04-03 09:34:49 +08:00
}
2021-04-23 18:27:47 +08:00
std::shared_ptr<RtcpFB> RtcpFB::create(PSFBType fmt, const void *fci, size_t fci_len) {
2022-07-14 22:49:31 +08:00
return RtcpFB::create_l(RtcpType::RTCP_PSFB, (int)fmt, fci, fci_len);
2021-04-23 18:27:47 +08:00
}
std::shared_ptr<RtcpFB> RtcpFB::create(RTPFBType fmt, const void *fci, size_t fci_len) {
2022-07-14 22:49:31 +08:00
return RtcpFB::create_l(RtcpType::RTCP_RTPFB, (int)fmt, fci, fci_len);
2021-04-23 18:27:47 +08:00
}
2021-05-10 23:27:11 +08:00
const void *RtcpFB::getFciPtr() const {
2022-07-14 22:49:31 +08:00
return (uint8_t *)&ssrc_media + sizeof(ssrc_media);
2021-05-10 23:27:11 +08:00
}
size_t RtcpFB::getFciSize() const {
2022-07-14 22:49:31 +08:00
auto fci_len = (ssize_t)getSize() - getPaddingSize() - sizeof(RtcpFB);
2023-04-21 21:12:42 +08:00
CHECK(getSize() >= getPaddingSize() + sizeof(RtcpFB));
2021-05-10 23:27:11 +08:00
return fci_len;
}
2021-04-23 18:27:47 +08:00
string RtcpFB::dumpString() const {
2021-04-03 09:34:49 +08:00
_StrPrinter printer;
printer << RtcpHeader::dumpHeader();
printer << "ssrc:" << ssrc << "\r\n";
2021-04-23 18:27:47 +08:00
printer << "ssrc_media:" << ssrc_media << "\r\n";
2022-07-14 22:49:31 +08:00
switch ((RtcpType)pt) {
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_PSFB: {
switch ((PSFBType)report_count) {
case PSFBType::RTCP_PSFB_SLI: {
auto &fci = getFci<FCI_SLI>();
printer << "fci:" << psfbTypeToStr((PSFBType)report_count) << " " << fci.dumpString();
break;
}
case PSFBType::RTCP_PSFB_PLI: {
getFciSize();
printer << "fci:" << psfbTypeToStr((PSFBType)report_count);
break;
}
case PSFBType::RTCP_PSFB_FIR: {
auto &fci = getFci<FCI_FIR>();
printer << "fci:" << psfbTypeToStr((PSFBType)report_count) << " " << fci.dumpString();
break;
}
case PSFBType::RTCP_PSFB_REMB: {
auto &fci = getFci<FCI_REMB>();
printer << "fci:" << psfbTypeToStr((PSFBType)report_count) << " " << fci.dumpString();
break;
}
default: {
printer << "fci:" << psfbTypeToStr((PSFBType)report_count) << " "
<< hexdump(getFciPtr(), getFciSize());
break;
}
}
2022-07-14 22:49:31 +08:00
break;
}
2022-08-05 17:55:48 +08:00
case RtcpType::RTCP_RTPFB: {
switch ((RTPFBType)report_count) {
case RTPFBType::RTCP_RTPFB_NACK: {
auto &fci = getFci<FCI_NACK>();
printer << "fci:" << rtpfbTypeToStr((RTPFBType)report_count) << " " << fci.dumpString();
break;
}
case RTPFBType::RTCP_RTPFB_TWCC: {
auto &fci = getFci<FCI_TWCC>();
printer << "fci:" << rtpfbTypeToStr((RTPFBType)report_count) << " " << fci.dumpString(getFciSize());
break;
}
default: {
printer << "fci:" << rtpfbTypeToStr((RTPFBType)report_count) << " "
<< hexdump(getFciPtr(), getFciSize());
break;
}
}
2022-07-14 22:49:31 +08:00
break;
}
2022-08-05 17:55:48 +08:00
default: /*不可达*/ assert(0); break;
2021-04-23 18:27:47 +08:00
}
2021-04-03 09:34:49 +08:00
return std::move(printer);
}
2021-04-23 18:27:47 +08:00
void RtcpFB::net2Host(size_t size) {
static const size_t kMinSize = sizeof(RtcpFB);
2021-04-03 09:34:49 +08:00
CHECK_MIN_SIZE(size, kMinSize);
ssrc = ntohl(ssrc);
ssrc_media = ntohl(ssrc_media);
}
2021-04-07 16:45:50 +08:00
////////////////////////////////////////////////////////////////////
2021-04-23 18:27:47 +08:00
std::shared_ptr<RtcpBye> RtcpBye::create(const std::vector<uint32_t> &ssrcs, const string &reason) {
2021-04-07 16:45:50 +08:00
assert(reason.size() <= 0xFF);
2021-04-26 21:50:10 +08:00
auto real_size = sizeof(RtcpHeader) + sizeof(uint32_t) * ssrcs.size() + 1 + reason.size();
auto bytes = alignSize(real_size);
2022-07-14 22:49:31 +08:00
auto ptr = (RtcpBye *)new char[bytes];
2021-04-07 16:45:50 +08:00
setupHeader(ptr, RtcpType::RTCP_BYE, ssrcs.size(), bytes);
2021-04-26 21:50:10 +08:00
setupPadding(ptr, bytes - real_size);
2021-04-07 16:45:50 +08:00
2021-12-30 15:28:02 +08:00
int i = 0;
2021-04-07 16:45:50 +08:00
for (auto ssrc : ssrcs) {
2022-07-14 22:49:31 +08:00
((RtcpBye *)ptr)->ssrc[i++] = htonl(ssrc);
2021-04-07 16:45:50 +08:00
}
if (!reason.empty()) {
2022-07-14 22:49:31 +08:00
uint8_t *reason_len_ptr = (uint8_t *)ptr + sizeof(RtcpHeader) + sizeof(uint32_t) * ssrcs.size();
2021-04-07 16:45:50 +08:00
*reason_len_ptr = reason.size() & 0xFF;
memcpy(reason_len_ptr + 1, reason.data(), *reason_len_ptr);
}
2022-08-05 17:55:48 +08:00
return std::shared_ptr<RtcpBye>(ptr, [](RtcpBye *ptr) { delete[](char *) ptr; });
2021-04-07 16:45:50 +08:00
}
2021-12-30 15:28:02 +08:00
vector<uint32_t *> RtcpBye::getSSRC() {
2021-04-07 16:45:50 +08:00
vector<uint32_t *> ret;
for (size_t i = 0; i < report_count; ++i) {
2021-12-30 15:28:02 +08:00
ret.emplace_back(&(ssrc[i]));
2021-04-07 16:45:50 +08:00
}
return ret;
}
string RtcpBye::getReason() const {
auto *reason_len_ptr = &reason_len + sizeof(ssrc) * (report_count - 1);
2022-07-14 22:49:31 +08:00
if (reason_len_ptr + 1 >= (uint8_t *)this + getSize()) {
2021-04-07 16:45:50 +08:00
return "";
}
2022-07-14 22:49:31 +08:00
return string((char *)reason_len_ptr + 1, *reason_len_ptr);
2021-04-07 16:45:50 +08:00
}
string RtcpBye::dumpString() const {
_StrPrinter printer;
printer << RtcpHeader::dumpHeader();
2022-07-14 22:49:31 +08:00
for (auto ssrc : ((RtcpBye *)this)->getSSRC()) {
2021-04-07 16:45:50 +08:00
printer << "ssrc:" << *ssrc << "\r\n";
}
printer << "reason:" << getReason();
return std::move(printer);
}
void RtcpBye::net2Host(size_t size) {
static const size_t kMinSize = sizeof(RtcpHeader);
CHECK_MIN_SIZE(size, kMinSize);
size_t offset = kMinSize;
size_t i = 0;
for (; i < report_count && offset + sizeof(ssrc) <= size; ++i) {
2021-12-30 15:28:02 +08:00
ssrc[i] = ntohl(ssrc[i]);
2021-04-07 16:45:50 +08:00
offset += sizeof(ssrc);
}
// 修正ssrc个数 [AUTO-TRANSLATED:57c74f58]
// Correct the number of ssrcs
2021-04-23 18:27:47 +08:00
CHECK_REPORT_COUNT(i);
2021-04-07 16:45:50 +08:00
if (offset < size) {
uint8_t *reason_len_ptr = &reason_len + sizeof(ssrc) * (report_count - 1);
2022-07-14 22:49:31 +08:00
if (reason_len_ptr + 1 + *reason_len_ptr > (uint8_t *)this + size) {
2021-04-07 16:45:50 +08:00
WarnL << "invalid rtcp bye reason length";
// 修正reason_len长度 [AUTO-TRANSLATED:1c0c9645]
// Correct the length of reason_len
2022-07-14 22:49:31 +08:00
*reason_len_ptr = ((uint8_t *)this + size - reason_len_ptr - 1) & 0xFF;
2021-04-07 16:45:50 +08:00
}
}
}
2022-07-14 22:49:31 +08:00
////////////////////////////////////////////
string RtcpXRRRTR::dumpString() const {
_StrPrinter printer;
printer << RtcpHeader::dumpHeader();
2022-07-14 22:49:31 +08:00
printer << "ssrc :" << ssrc << "\r\n";
printer << "bt :" << (int)bt << "\r\n";
printer << "block_length : " << block_length << "\r\n";
printer << "ntp msw : " << ntpmsw << "\r\n";
printer << "ntp lsw : " << ntplsw << "\r\n";
return std::move(printer);
}
void RtcpXRRRTR::net2Host(size_t size) {
static const size_t kMinSize = sizeof(RtcpHeader);
CHECK_MIN_SIZE(size, kMinSize);
2022-07-14 22:49:31 +08:00
if (size != sizeof(RtcpXRRRTR)) {
throw std::invalid_argument(
StrPrinter << "rtcp xr Receiver Reference Time Report Block must is " << sizeof(RtcpXRRRTR)
<< " actual size " << size);
}
ssrc = ntohl(ssrc);
block_length = ntohs(block_length);
ntpmsw = ntohl(ntpmsw);
ntplsw = ntohl(ntplsw);
}
string RtcpXRDLRRReportItem::dumpString() const {
_StrPrinter printer;
2022-07-14 22:49:31 +08:00
printer << "ssrc :" << ssrc << "\r\n";
printer << "last RR (lrr) :" << lrr << "\r\n";
printer << "delay since last RR (dlrr): " << dlrr << "\r\n";
return std::move(printer);
}
void RtcpXRDLRRReportItem::net2Host() {
ssrc = ntohl(ssrc);
lrr = ntohl(lrr);
dlrr = ntohl(dlrr);
}
2022-07-14 22:49:31 +08:00
std::vector<RtcpXRDLRRReportItem *> RtcpXRDLRR::getItemList() {
auto count = block_length / 3;
RtcpXRDLRRReportItem *ptr = &items;
vector<RtcpXRDLRRReportItem *> ret;
2022-07-14 22:49:31 +08:00
for (int i = 0; i < (int)count; ++i) {
ret.emplace_back(ptr);
++ptr;
}
return ret;
}
string RtcpXRDLRR::dumpString() const {
_StrPrinter printer;
printer << RtcpHeader::dumpHeader();
2022-07-14 22:49:31 +08:00
printer << "ssrc :" << ssrc << "\r\n";
printer << "bt :" << (int)bt << "\r\n";
printer << "block_length : " << block_length << "\r\n";
auto items_list = ((RtcpXRDLRR *)this)->getItemList();
auto i = 0;
for (auto &item : items_list) {
printer << "---- item:" << i++ << " ----\r\n";
printer << item->dumpString();
}
return std::move(printer);
}
void RtcpXRDLRR::net2Host(size_t size) {
static const size_t kMinSize = sizeof(RtcpHeader);
CHECK_MIN_SIZE(size, kMinSize);
ssrc = ntohl(ssrc);
block_length = ntohs(block_length);
2022-07-14 22:49:31 +08:00
auto count = block_length / 3;
for (int i = 0; i < (int)count; ++i) {
RtcpXRDLRRReportItem *ptr = &items;
ptr->net2Host();
ptr++;
}
}
2022-07-14 22:49:31 +08:00
std::shared_ptr<RtcpXRDLRR> RtcpXRDLRR::create(size_t item_count) {
auto real_size = sizeof(RtcpXRDLRR) - sizeof(RtcpXRDLRRReportItem) + item_count * sizeof(RtcpXRDLRRReportItem);
auto bytes = alignSize(real_size);
2022-07-14 22:49:31 +08:00
auto ptr = (RtcpXRDLRR *)new char[bytes];
setupHeader(ptr, RtcpType::RTCP_XR, 0, bytes);
setupPadding(ptr, bytes - real_size);
2022-08-05 17:55:48 +08:00
return std::shared_ptr<RtcpXRDLRR>(ptr, [](RtcpXRDLRR *ptr) { delete[](char *) ptr; });
}
2021-04-07 16:45:50 +08:00
////////////////////////////////////
string RtcpXRTargetBitrateItem::dumpString() const {
_StrPrinter printer;
printer << "Spatial Layer :" << spatial_layer << "\r\n";
printer << "Temporal Layer :" << temporal_layer << "\r\n";
printer << "Target Bitrate: " << target_bitrate << "\r\n";
return std::move(printer);
}
void RtcpXRTargetBitrateItem::net2Host() {
target_bitrate = ntohl(target_bitrate) >> 8;
}
std::vector<RtcpXRTargetBitrateItem *> RtcpXRTargetBitrate::getItemList() {
auto count = block_length;
RtcpXRTargetBitrateItem *ptr = &items;
vector<RtcpXRTargetBitrateItem *> ret;
for (int i = 0; i < (int)count; ++i) {
ret.emplace_back(ptr);
++ptr;
}
return ret;
}
string RtcpXRTargetBitrate::dumpString() const {
_StrPrinter printer;
printer << RtcpHeader::dumpHeader();
printer << "ssrc :" << ssrc << "\r\n";
printer << "bt :" << (int)bt << "\r\n";
printer << "block_length : " << block_length << "\r\n";
auto items_list = ((RtcpXRTargetBitrate *)this)->getItemList();
auto i = 0;
for (auto &item : items_list) {
printer << "---- item:" << i++ << " ----\r\n";
printer << item->dumpString();
}
return std::move(printer);
}
void RtcpXRTargetBitrate::net2Host(size_t size) {
static const size_t kMinSize = sizeof(RtcpHeader);
CHECK_MIN_SIZE(size, kMinSize);
ssrc = ntohl(ssrc);
block_length = ntohs(block_length);
auto count = block_length;
for (int i = 0; i < (int)count; ++i) {
RtcpXRTargetBitrateItem *ptr = &items;
ptr->net2Host();
ptr++;
}
}
std::shared_ptr<RtcpXRTargetBitrate> RtcpXRTargetBitrate::create(size_t item_count) {
auto real_size = sizeof(RtcpXRTargetBitrate) - sizeof(RtcpXRTargetBitrateItem) + item_count * sizeof(RtcpXRTargetBitrateItem);
auto bytes = alignSize(real_size);
auto ptr = (RtcpXRTargetBitrate *)new char[bytes];
setupHeader(ptr, RtcpType::RTCP_XR, 0, bytes);
setupPadding(ptr, bytes - real_size);
return std::shared_ptr<RtcpXRTargetBitrate>(ptr, [](RtcpXRTargetBitrate *ptr) { delete[](char *) ptr; });
}
2021-04-15 19:35:49 +08:00
#if 0
2021-04-07 16:45:50 +08:00
#include "Util/onceToken.h"
static toolkit::onceToken token([](){
auto bye = RtcpBye::create({1,2,3,4,5,6}, "this is a bye reason");
auto buffer = RtcpHeader::toBuffer(bye);
auto rtcps = RtcpHeader::loadFromBytes(buffer->data(), buffer->size());
for(auto rtcp : rtcps){
std::cout << rtcp->dumpString() << std::endl;
}
});
#endif
2022-07-14 22:49:31 +08:00
} // namespace mediakit