mirror of
https://github.com/ZLMediaKit/ZLMediaKit.git
synced 2024-11-26 12:37:09 +08:00
初步完成TWCC包的解析
This commit is contained in:
parent
5db6154707
commit
90ad90cb78
@ -144,8 +144,8 @@ void RtcpHeader::net2Host(size_t len){
|
|||||||
}
|
}
|
||||||
|
|
||||||
case RtcpType::RTCP_RTPFB: {
|
case RtcpType::RTCP_RTPFB: {
|
||||||
//todo 支持rtcp-fb相关功能
|
RtcpPli *pli = (RtcpPli *)this;
|
||||||
net2Host();
|
pli->net2Host(len);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
default: throw std::runtime_error(StrPrinter << "未处理的rtcp包:" << rtcpTypeToStr((RtcpType) this->pt));
|
default: throw std::runtime_error(StrPrinter << "未处理的rtcp包:" << rtcpTypeToStr((RtcpType) this->pt));
|
||||||
|
@ -246,4 +246,92 @@ string StatusVecChunk::dumpString() const {
|
|||||||
return std::move(printer);
|
return std::move(printer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void FCI_TWCC::net2Host(size_t total_size) {
|
||||||
|
CHECK(total_size >= kSize);
|
||||||
|
base_seq = ntohs(base_seq);
|
||||||
|
pkt_status_count = ntohs(pkt_status_count);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t FCI_TWCC::getReferenceTime() const {
|
||||||
|
uint32_t ret = 0;
|
||||||
|
ret |= ref_time[0] << 16;
|
||||||
|
ret |= ref_time[1] << 8;
|
||||||
|
ret |= ref_time[2];
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint16_t getRecvDelta(SymbolStatus status, uint8_t *&ptr, const uint8_t *end){
|
||||||
|
uint16_t delta = 0;
|
||||||
|
switch (status) {
|
||||||
|
case SymbolStatus::not_received : {
|
||||||
|
//丢包, recv delta为0个字节
|
||||||
|
delta = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case SymbolStatus::small_delta : {
|
||||||
|
CHECK(ptr + 1 <= end);
|
||||||
|
//时间戳增量小于256, recv delta为1个字节
|
||||||
|
delta = *ptr;
|
||||||
|
ptr += 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case SymbolStatus::large_delta : {
|
||||||
|
CHECK(ptr + 2 <= end);
|
||||||
|
//时间戳增量256~65535间,recv delta为2个字节
|
||||||
|
delta = *ptr << 8 | *(ptr + 1);
|
||||||
|
ptr += 2;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
//这个逻辑分支不可达到
|
||||||
|
CHECK(0);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return delta;
|
||||||
|
}
|
||||||
|
|
||||||
|
map<uint16_t, std::pair<SymbolStatus, uint32_t/*stamp*/> > FCI_TWCC::getPacketChunkList(size_t total_size) const {
|
||||||
|
map<uint16_t, std::pair<SymbolStatus, uint32_t> > ret;
|
||||||
|
auto ptr = (uint8_t *) this + kSize;
|
||||||
|
auto end = (uint8_t *) this + total_size;
|
||||||
|
CHECK(ptr < end);
|
||||||
|
auto seq = base_seq;
|
||||||
|
auto stamp = getReferenceTime();
|
||||||
|
|
||||||
|
for (uint8_t i = 0; i < pkt_status_count;) {
|
||||||
|
CHECK(ptr + RunLengthChunk::kSize <= end)
|
||||||
|
RunLengthChunk *chunk = (RunLengthChunk *) ptr;
|
||||||
|
if (!chunk->type) {
|
||||||
|
//RunLengthChunk
|
||||||
|
ptr += RunLengthChunk::kSize;
|
||||||
|
for (auto j = 0; j < chunk->getRunLength(); ++j) {
|
||||||
|
ret.emplace(seq++, std::make_pair((SymbolStatus) chunk->symbol, stamp));
|
||||||
|
stamp += getRecvDelta((SymbolStatus) chunk->symbol, ptr, end);
|
||||||
|
++i;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
//StatusVecChunk
|
||||||
|
StatusVecChunk *chunk = (StatusVecChunk *) ptr;
|
||||||
|
ptr += StatusVecChunk::kSize;
|
||||||
|
for (auto &symbol : chunk->getSymbolList()) {
|
||||||
|
ret.emplace(seq++, std::make_pair(symbol, stamp));
|
||||||
|
stamp += getRecvDelta(symbol, ptr, end);
|
||||||
|
++i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
string FCI_TWCC::dumpString(size_t total_size) const {
|
||||||
|
_StrPrinter printer;
|
||||||
|
auto map = getPacketChunkList(total_size);
|
||||||
|
for (auto &pr : map) {
|
||||||
|
printer << " seq:" << pr.first <<", packet status:" << (int)(pr.second.first) << ", stamp:" << pr.second.second << "\n";
|
||||||
|
}
|
||||||
|
return std::move(printer);
|
||||||
|
}
|
||||||
|
|
||||||
}//namespace mediakit
|
}//namespace mediakit
|
@ -296,29 +296,6 @@ public:
|
|||||||
|
|
||||||
} PACKED;
|
} PACKED;
|
||||||
|
|
||||||
//RTPFB fmt = 15
|
|
||||||
//https://tools.ietf.org/html/draft-holmer-rmcat-transport-wide-cc-extensions-01#section-3.1
|
|
||||||
//https://zhuanlan.zhihu.com/p/206656654
|
|
||||||
//0 1 2 3
|
|
||||||
// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
|
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
||||||
// | base sequence number | packet status count |
|
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
||||||
// | reference time | fb pkt. count |
|
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
||||||
// | packet chunk | packet chunk |
|
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
||||||
// . .
|
|
||||||
// . .
|
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
||||||
// | packet chunk | recv delta | recv delta |
|
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
||||||
// . .
|
|
||||||
// . .
|
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
||||||
// | recv delta | recv delta | zero padding |
|
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
||||||
|
|
||||||
enum class SymbolStatus : uint8_t{
|
enum class SymbolStatus : uint8_t{
|
||||||
//Packet not received
|
//Packet not received
|
||||||
not_received = 0,
|
not_received = 0,
|
||||||
@ -367,7 +344,7 @@ public:
|
|||||||
// 0 1
|
// 0 1
|
||||||
// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
|
// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
// |T| S | Run Length |
|
// |T|S| symbol list |
|
||||||
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
#if __BYTE_ORDER == __BIG_ENDIAN
|
#if __BYTE_ORDER == __BIG_ENDIAN
|
||||||
uint16_t type: 1;
|
uint16_t type: 1;
|
||||||
@ -392,9 +369,31 @@ public:
|
|||||||
string dumpString() const;
|
string dumpString() const;
|
||||||
} PACKED;
|
} PACKED;
|
||||||
|
|
||||||
|
//RTPFB fmt = 15
|
||||||
|
//https://tools.ietf.org/html/draft-holmer-rmcat-transport-wide-cc-extensions-01#section-3.1
|
||||||
|
//https://zhuanlan.zhihu.com/p/206656654
|
||||||
|
// 0 1 2 3
|
||||||
|
// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
|
||||||
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
// | base sequence number | packet status count |
|
||||||
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
// | reference time | fb pkt. count |
|
||||||
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
// | packet chunk | packet chunk |
|
||||||
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
// . .
|
||||||
|
// . .
|
||||||
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
// | packet chunk | recv delta | recv delta |
|
||||||
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
// . .
|
||||||
|
// . .
|
||||||
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
// | recv delta | recv delta | zero padding |
|
||||||
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
class FCI_TWCC{
|
class FCI_TWCC{
|
||||||
public:
|
public:
|
||||||
static size_t constexpr kSize = 12;
|
static size_t constexpr kSize = 8;
|
||||||
|
|
||||||
//base sequence number,基础序号,本次反馈的第一个包的序号;也就是RTP扩展头的序列号
|
//base sequence number,基础序号,本次反馈的第一个包的序号;也就是RTP扩展头的序列号
|
||||||
uint16_t base_seq;
|
uint16_t base_seq;
|
||||||
@ -405,6 +404,11 @@ public:
|
|||||||
//feedback packet count,反馈包号,本包是第几个transport-cc包,每次加1 |
|
//feedback packet count,反馈包号,本包是第几个transport-cc包,每次加1 |
|
||||||
uint8_t fb_pkt_count;
|
uint8_t fb_pkt_count;
|
||||||
|
|
||||||
|
void net2Host(size_t total_size);
|
||||||
|
uint32_t getReferenceTime() const;
|
||||||
|
map<uint16_t, std::pair<SymbolStatus, uint32_t/*stamp*/> > getPacketChunkList(size_t total_size) const;
|
||||||
|
string dumpString(size_t total_size) const;
|
||||||
|
|
||||||
} PACKED;
|
} PACKED;
|
||||||
|
|
||||||
} //namespace mediakit
|
} //namespace mediakit
|
||||||
|
@ -501,6 +501,8 @@ private:
|
|||||||
function<void(const RtpPacket::Ptr &rtp)> _on_before_sort;
|
function<void(const RtpPacket::Ptr &rtp)> _on_before_sort;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#include "RTCP/RtcpFCI.h"
|
||||||
|
|
||||||
void WebRtcTransportImp::onRtcp(const char *buf, size_t len) {
|
void WebRtcTransportImp::onRtcp(const char *buf, size_t len) {
|
||||||
_bytes_usage += len;
|
_bytes_usage += len;
|
||||||
auto rtcps = RtcpHeader::loadFromBytes((char *) buf, len);
|
auto rtcps = RtcpHeader::loadFromBytes((char *) buf, len);
|
||||||
@ -546,6 +548,20 @@ void WebRtcTransportImp::onRtcp(const char *buf, size_t len) {
|
|||||||
//todo 支持pli等更多类型的rtcp
|
//todo 支持pli等更多类型的rtcp
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case RtcpType::RTCP_RTPFB: {
|
||||||
|
//todo 测试打印twcc
|
||||||
|
RtcpPli *rtpfb = (RtcpPli *)rtcp;
|
||||||
|
auto fci = (uint8_t *)rtpfb + sizeof (RtcpPli);
|
||||||
|
if(rtpfb->report_count == 15){
|
||||||
|
//TWCC
|
||||||
|
FCI_TWCC *twcc = (FCI_TWCC *) (fci);
|
||||||
|
auto fci_size = rtpfb->getSize() - 12;
|
||||||
|
InfoL << hexdump(fci, fci_size);
|
||||||
|
twcc->net2Host(fci_size);
|
||||||
|
InfoL << "\n" << twcc->dumpString(fci_size);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
default: break;
|
default: break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user