mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-13 05:40:42 +01:00

Bug: webrtc:9598 Change-Id: I8a54af88708e5d96e46ba67ab0ef2e0e59fe0b86 Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/300941 Reviewed-by: Björn Terelius <terelius@webrtc.org> Reviewed-by: Åsa Persson <asapersson@webrtc.org> Commit-Queue: Danil Chapovalov <danilchap@webrtc.org> Cr-Commit-Position: refs/heads/main@{#39887}
844 lines
35 KiB
C++
844 lines
35 KiB
C++
/*
|
|
* Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#include "modules/rtp_rtcp/source/rtcp_sender.h"
|
|
|
|
#include <memory>
|
|
#include <utility>
|
|
|
|
#include "absl/base/macros.h"
|
|
#include "api/units/time_delta.h"
|
|
#include "modules/rtp_rtcp/include/rtp_rtcp_defines.h"
|
|
#include "modules/rtp_rtcp/source/rtcp_packet/bye.h"
|
|
#include "modules/rtp_rtcp/source/rtcp_packet/common_header.h"
|
|
#include "modules/rtp_rtcp/source/rtp_packet_received.h"
|
|
#include "modules/rtp_rtcp/source/rtp_rtcp_impl2.h"
|
|
#include "rtc_base/rate_limiter.h"
|
|
#include "rtc_base/thread.h"
|
|
#include "test/gmock.h"
|
|
#include "test/gtest.h"
|
|
#include "test/mock_transport.h"
|
|
#include "test/rtcp_packet_parser.h"
|
|
|
|
using ::testing::_;
|
|
using ::testing::ElementsAre;
|
|
using ::testing::Eq;
|
|
using ::testing::Invoke;
|
|
using ::testing::Property;
|
|
using ::testing::SizeIs;
|
|
|
|
namespace webrtc {
|
|
|
|
class RtcpPacketTypeCounterObserverImpl : public RtcpPacketTypeCounterObserver {
|
|
public:
|
|
RtcpPacketTypeCounterObserverImpl() : ssrc_(0) {}
|
|
~RtcpPacketTypeCounterObserverImpl() override = default;
|
|
void RtcpPacketTypesCounterUpdated(
|
|
uint32_t ssrc,
|
|
const RtcpPacketTypeCounter& packet_counter) override {
|
|
ssrc_ = ssrc;
|
|
counter_ = packet_counter;
|
|
}
|
|
uint32_t ssrc_;
|
|
RtcpPacketTypeCounter counter_;
|
|
};
|
|
|
|
class TestTransport : public Transport {
|
|
public:
|
|
TestTransport() {}
|
|
|
|
bool SendRtp(const uint8_t* /*data*/,
|
|
size_t /*len*/,
|
|
const PacketOptions& options) override {
|
|
return false;
|
|
}
|
|
bool SendRtcp(const uint8_t* data, size_t len) override {
|
|
parser_.Parse(data, len);
|
|
return true;
|
|
}
|
|
test::RtcpPacketParser parser_;
|
|
};
|
|
|
|
namespace {
|
|
static const uint32_t kSenderSsrc = 0x11111111;
|
|
static const uint32_t kRemoteSsrc = 0x22222222;
|
|
static const uint32_t kStartRtpTimestamp = 0x34567;
|
|
static const uint32_t kRtpTimestamp = 0x45678;
|
|
|
|
std::unique_ptr<RTCPSender> CreateRtcpSender(
|
|
const RTCPSender::Configuration& config,
|
|
bool init_timestamps = true) {
|
|
auto rtcp_sender = std::make_unique<RTCPSender>(config);
|
|
rtcp_sender->SetRemoteSSRC(kRemoteSsrc);
|
|
if (init_timestamps) {
|
|
rtcp_sender->SetTimestampOffset(kStartRtpTimestamp);
|
|
rtcp_sender->SetLastRtpTime(kRtpTimestamp, config.clock->CurrentTime(),
|
|
/*payload_type=*/0);
|
|
}
|
|
return rtcp_sender;
|
|
}
|
|
} // namespace
|
|
|
|
class RtcpSenderTest : public ::testing::Test {
|
|
protected:
|
|
RtcpSenderTest()
|
|
: clock_(1335900000),
|
|
receive_statistics_(ReceiveStatistics::Create(&clock_)) {
|
|
rtp_rtcp_impl_.reset(new ModuleRtpRtcpImpl2(GetDefaultRtpRtcpConfig()));
|
|
}
|
|
|
|
RTCPSender::Configuration GetDefaultConfig() {
|
|
RTCPSender::Configuration configuration;
|
|
configuration.audio = false;
|
|
configuration.clock = &clock_;
|
|
configuration.outgoing_transport = &test_transport_;
|
|
configuration.rtcp_report_interval = TimeDelta::Millis(1000);
|
|
configuration.receive_statistics = receive_statistics_.get();
|
|
configuration.local_media_ssrc = kSenderSsrc;
|
|
return configuration;
|
|
}
|
|
|
|
RtpRtcpInterface::Configuration GetDefaultRtpRtcpConfig() {
|
|
RTCPSender::Configuration config = GetDefaultConfig();
|
|
RtpRtcpInterface::Configuration result;
|
|
result.audio = config.audio;
|
|
result.clock = config.clock;
|
|
result.outgoing_transport = config.outgoing_transport;
|
|
result.rtcp_report_interval_ms = config.rtcp_report_interval->ms();
|
|
result.receive_statistics = config.receive_statistics;
|
|
result.local_media_ssrc = config.local_media_ssrc;
|
|
return result;
|
|
}
|
|
|
|
void InsertIncomingPacket(uint32_t remote_ssrc, uint16_t seq_num) {
|
|
RtpPacketReceived packet;
|
|
packet.SetSsrc(remote_ssrc);
|
|
packet.SetSequenceNumber(seq_num);
|
|
packet.SetTimestamp(12345);
|
|
packet.SetPayloadSize(100 - 12);
|
|
receive_statistics_->OnRtpPacket(packet);
|
|
}
|
|
|
|
test::RtcpPacketParser* parser() { return &test_transport_.parser_; }
|
|
|
|
RTCPSender::FeedbackState feedback_state() {
|
|
return rtp_rtcp_impl_->GetFeedbackState();
|
|
}
|
|
|
|
rtc::AutoThread main_thread_;
|
|
SimulatedClock clock_;
|
|
TestTransport test_transport_;
|
|
std::unique_ptr<ReceiveStatistics> receive_statistics_;
|
|
std::unique_ptr<ModuleRtpRtcpImpl2> rtp_rtcp_impl_;
|
|
};
|
|
|
|
TEST_F(RtcpSenderTest, SetRtcpStatus) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
EXPECT_EQ(RtcpMode::kOff, rtcp_sender->Status());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
EXPECT_EQ(RtcpMode::kReducedSize, rtcp_sender->Status());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SetSendingStatus) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
EXPECT_FALSE(rtcp_sender->Sending());
|
|
rtcp_sender->SetSendingStatus(feedback_state(), true);
|
|
EXPECT_TRUE(rtcp_sender->Sending());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, NoPacketSentIfOff) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kOff);
|
|
EXPECT_EQ(-1, rtcp_sender->SendRTCP(feedback_state(), kRtcpSr));
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendSr) {
|
|
const uint32_t kPacketCount = 0x12345;
|
|
const uint32_t kOctetCount = 0x23456;
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
RTCPSender::FeedbackState feedback_state = rtp_rtcp_impl_->GetFeedbackState();
|
|
rtcp_sender->SetSendingStatus(feedback_state, true);
|
|
feedback_state.packets_sent = kPacketCount;
|
|
feedback_state.media_bytes_sent = kOctetCount;
|
|
NtpTime ntp = clock_.CurrentNtpTime();
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state, kRtcpSr));
|
|
EXPECT_EQ(1, parser()->sender_report()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->sender_report()->sender_ssrc());
|
|
EXPECT_EQ(ntp, parser()->sender_report()->ntp());
|
|
EXPECT_EQ(kPacketCount, parser()->sender_report()->sender_packet_count());
|
|
EXPECT_EQ(kOctetCount, parser()->sender_report()->sender_octet_count());
|
|
EXPECT_EQ(kStartRtpTimestamp + kRtpTimestamp,
|
|
parser()->sender_report()->rtp_timestamp());
|
|
EXPECT_EQ(0U, parser()->sender_report()->report_blocks().size());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendConsecutiveSrWithExactSlope) {
|
|
const uint32_t kPacketCount = 0x12345;
|
|
const uint32_t kOctetCount = 0x23456;
|
|
const int kTimeBetweenSRsUs = 10043; // Not exact value in milliseconds.
|
|
const int kExtraPackets = 30;
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
// Make sure clock is not exactly at some milliseconds point.
|
|
clock_.AdvanceTimeMicroseconds(kTimeBetweenSRsUs);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
RTCPSender::FeedbackState feedback_state = rtp_rtcp_impl_->GetFeedbackState();
|
|
rtcp_sender->SetSendingStatus(feedback_state, true);
|
|
feedback_state.packets_sent = kPacketCount;
|
|
feedback_state.media_bytes_sent = kOctetCount;
|
|
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state, kRtcpSr));
|
|
EXPECT_EQ(1, parser()->sender_report()->num_packets());
|
|
NtpTime ntp1 = parser()->sender_report()->ntp();
|
|
uint32_t rtp1 = parser()->sender_report()->rtp_timestamp();
|
|
|
|
// Send more SRs to ensure slope is always exact for different offsets
|
|
for (int packets = 1; packets <= kExtraPackets; ++packets) {
|
|
clock_.AdvanceTimeMicroseconds(kTimeBetweenSRsUs);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state, kRtcpSr));
|
|
EXPECT_EQ(packets + 1, parser()->sender_report()->num_packets());
|
|
|
|
NtpTime ntp2 = parser()->sender_report()->ntp();
|
|
uint32_t rtp2 = parser()->sender_report()->rtp_timestamp();
|
|
|
|
uint32_t ntp_diff_in_rtp_units =
|
|
(ntp2.ToMs() - ntp1.ToMs()) * (kVideoPayloadTypeFrequency / 1000);
|
|
EXPECT_EQ(rtp2 - rtp1, ntp_diff_in_rtp_units);
|
|
}
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, DoNotSendSrBeforeRtp) {
|
|
RTCPSender::Configuration config;
|
|
config.clock = &clock_;
|
|
config.receive_statistics = receive_statistics_.get();
|
|
config.outgoing_transport = &test_transport_;
|
|
config.rtcp_report_interval = TimeDelta::Millis(1000);
|
|
config.local_media_ssrc = kSenderSsrc;
|
|
auto rtcp_sender = CreateRtcpSender(config, /*init_timestamps=*/false);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), true);
|
|
|
|
// Sender Report shouldn't be send as an SR nor as a Report.
|
|
rtcp_sender->SendRTCP(feedback_state(), kRtcpSr);
|
|
EXPECT_EQ(0, parser()->sender_report()->num_packets());
|
|
rtcp_sender->SendRTCP(feedback_state(), kRtcpReport);
|
|
EXPECT_EQ(0, parser()->sender_report()->num_packets());
|
|
// Other packets (e.g. Pli) are allowed, even if useless.
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpPli));
|
|
EXPECT_EQ(1, parser()->pli()->num_packets());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, DoNotSendCompundBeforeRtp) {
|
|
RTCPSender::Configuration config;
|
|
config.clock = &clock_;
|
|
config.receive_statistics = receive_statistics_.get();
|
|
config.outgoing_transport = &test_transport_;
|
|
config.rtcp_report_interval = TimeDelta::Millis(1000);
|
|
config.local_media_ssrc = kSenderSsrc;
|
|
auto rtcp_sender = CreateRtcpSender(config, /*init_timestamps=*/false);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), true);
|
|
|
|
// In compound mode no packets are allowed (e.g. Pli) because compound mode
|
|
// should start with Sender Report.
|
|
EXPECT_EQ(-1, rtcp_sender->SendRTCP(feedback_state(), kRtcpPli));
|
|
EXPECT_EQ(0, parser()->pli()->num_packets());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendRr) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpRr));
|
|
EXPECT_EQ(1, parser()->receiver_report()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->receiver_report()->sender_ssrc());
|
|
EXPECT_EQ(0U, parser()->receiver_report()->report_blocks().size());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, DoesntSendEmptyRrInReducedSizeMode) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
rtcp_sender->SendRTCP(feedback_state(), kRtcpRr);
|
|
EXPECT_EQ(parser()->receiver_report()->num_packets(), 0);
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendRrWithOneReportBlock) {
|
|
const uint16_t kSeqNum = 11111;
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
InsertIncomingPacket(kRemoteSsrc, kSeqNum);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpRr));
|
|
EXPECT_EQ(1, parser()->receiver_report()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->receiver_report()->sender_ssrc());
|
|
ASSERT_EQ(1U, parser()->receiver_report()->report_blocks().size());
|
|
const rtcp::ReportBlock& rb = parser()->receiver_report()->report_blocks()[0];
|
|
EXPECT_EQ(kRemoteSsrc, rb.source_ssrc());
|
|
EXPECT_EQ(0U, rb.fraction_lost());
|
|
EXPECT_EQ(0, rb.cumulative_lost());
|
|
EXPECT_EQ(kSeqNum, rb.extended_high_seq_num());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendRrWithTwoReportBlocks) {
|
|
const uint16_t kSeqNum = 11111;
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
InsertIncomingPacket(kRemoteSsrc, kSeqNum);
|
|
InsertIncomingPacket(kRemoteSsrc + 1, kSeqNum + 1);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpRr));
|
|
EXPECT_EQ(1, parser()->receiver_report()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->receiver_report()->sender_ssrc());
|
|
EXPECT_THAT(
|
|
parser()->receiver_report()->report_blocks(),
|
|
UnorderedElementsAre(
|
|
Property(&rtcp::ReportBlock::source_ssrc, Eq(kRemoteSsrc)),
|
|
Property(&rtcp::ReportBlock::source_ssrc, Eq(kRemoteSsrc + 1))));
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendSdes) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
EXPECT_EQ(0, rtcp_sender->SetCNAME("alice@host"));
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpSdes));
|
|
EXPECT_EQ(1, parser()->sdes()->num_packets());
|
|
EXPECT_EQ(1U, parser()->sdes()->chunks().size());
|
|
EXPECT_EQ(kSenderSsrc, parser()->sdes()->chunks()[0].ssrc);
|
|
EXPECT_EQ("alice@host", parser()->sdes()->chunks()[0].cname);
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SdesIncludedInCompoundPacket) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
EXPECT_EQ(0, rtcp_sender->SetCNAME("alice@host"));
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
EXPECT_EQ(1, parser()->receiver_report()->num_packets());
|
|
EXPECT_EQ(1, parser()->sdes()->num_packets());
|
|
EXPECT_EQ(1U, parser()->sdes()->chunks().size());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendBye) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpBye));
|
|
EXPECT_EQ(1, parser()->bye()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->bye()->sender_ssrc());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, StopSendingTriggersBye) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), true);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), false);
|
|
EXPECT_EQ(1, parser()->bye()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->bye()->sender_ssrc());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendFir) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpFir));
|
|
EXPECT_EQ(1, parser()->fir()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->fir()->sender_ssrc());
|
|
EXPECT_EQ(1U, parser()->fir()->requests().size());
|
|
EXPECT_EQ(kRemoteSsrc, parser()->fir()->requests()[0].ssrc);
|
|
uint8_t seq = parser()->fir()->requests()[0].seq_nr;
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpFir));
|
|
EXPECT_EQ(2, parser()->fir()->num_packets());
|
|
EXPECT_EQ(seq + 1, parser()->fir()->requests()[0].seq_nr);
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendPli) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpPli));
|
|
EXPECT_EQ(1, parser()->pli()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->pli()->sender_ssrc());
|
|
EXPECT_EQ(kRemoteSsrc, parser()->pli()->media_ssrc());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendNack) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
const uint16_t kList[] = {0, 1, 16};
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpNack,
|
|
ABSL_ARRAYSIZE(kList), kList));
|
|
EXPECT_EQ(1, parser()->nack()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->nack()->sender_ssrc());
|
|
EXPECT_EQ(kRemoteSsrc, parser()->nack()->media_ssrc());
|
|
EXPECT_THAT(parser()->nack()->packet_ids(), ElementsAre(0, 1, 16));
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendLossNotificationBufferingNotAllowed) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
constexpr uint16_t kLastDecoded = 0x1234;
|
|
constexpr uint16_t kLastReceived = 0x4321;
|
|
constexpr bool kDecodabilityFlag = true;
|
|
constexpr bool kBufferingAllowed = false;
|
|
EXPECT_EQ(rtcp_sender->SendLossNotification(feedback_state(), kLastDecoded,
|
|
kLastReceived, kDecodabilityFlag,
|
|
kBufferingAllowed),
|
|
0);
|
|
EXPECT_EQ(parser()->processed_rtcp_packets(), 1u);
|
|
EXPECT_EQ(parser()->loss_notification()->num_packets(), 1);
|
|
EXPECT_EQ(kSenderSsrc, parser()->loss_notification()->sender_ssrc());
|
|
EXPECT_EQ(kRemoteSsrc, parser()->loss_notification()->media_ssrc());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendLossNotificationBufferingAllowed) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
constexpr uint16_t kLastDecoded = 0x1234;
|
|
constexpr uint16_t kLastReceived = 0x4321;
|
|
constexpr bool kDecodabilityFlag = true;
|
|
constexpr bool kBufferingAllowed = true;
|
|
EXPECT_EQ(rtcp_sender->SendLossNotification(feedback_state(), kLastDecoded,
|
|
kLastReceived, kDecodabilityFlag,
|
|
kBufferingAllowed),
|
|
0);
|
|
|
|
// No RTCP messages sent yet.
|
|
ASSERT_EQ(parser()->processed_rtcp_packets(), 0u);
|
|
|
|
// Sending another messages triggers sending the LNTF messages as well.
|
|
const uint16_t kList[] = {0, 1, 16};
|
|
EXPECT_EQ(rtcp_sender->SendRTCP(feedback_state(), kRtcpNack,
|
|
ABSL_ARRAYSIZE(kList), kList),
|
|
0);
|
|
|
|
// Exactly one packet was produced, and it contained both the buffered LNTF
|
|
// as well as the message that had triggered the packet.
|
|
EXPECT_EQ(parser()->processed_rtcp_packets(), 1u);
|
|
EXPECT_EQ(parser()->loss_notification()->num_packets(), 1);
|
|
EXPECT_EQ(parser()->loss_notification()->sender_ssrc(), kSenderSsrc);
|
|
EXPECT_EQ(parser()->loss_notification()->media_ssrc(), kRemoteSsrc);
|
|
EXPECT_EQ(parser()->nack()->num_packets(), 1);
|
|
EXPECT_EQ(parser()->nack()->sender_ssrc(), kSenderSsrc);
|
|
EXPECT_EQ(parser()->nack()->media_ssrc(), kRemoteSsrc);
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, RembNotIncludedBeforeSet) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
|
|
rtcp_sender->SendRTCP(feedback_state(), kRtcpRr);
|
|
|
|
ASSERT_EQ(1, parser()->receiver_report()->num_packets());
|
|
EXPECT_EQ(0, parser()->remb()->num_packets());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, RembNotIncludedAfterUnset) {
|
|
const int64_t kBitrate = 261011;
|
|
const std::vector<uint32_t> kSsrcs = {kRemoteSsrc, kRemoteSsrc + 1};
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetRemb(kBitrate, kSsrcs);
|
|
rtcp_sender->SendRTCP(feedback_state(), kRtcpRr);
|
|
ASSERT_EQ(1, parser()->receiver_report()->num_packets());
|
|
EXPECT_EQ(1, parser()->remb()->num_packets());
|
|
|
|
// Turn off REMB. rtcp_sender no longer should send it.
|
|
rtcp_sender->UnsetRemb();
|
|
rtcp_sender->SendRTCP(feedback_state(), kRtcpRr);
|
|
ASSERT_EQ(2, parser()->receiver_report()->num_packets());
|
|
EXPECT_EQ(1, parser()->remb()->num_packets());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendRemb) {
|
|
const int64_t kBitrate = 261011;
|
|
const std::vector<uint32_t> kSsrcs = {kRemoteSsrc, kRemoteSsrc + 1};
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
rtcp_sender->SetRemb(kBitrate, kSsrcs);
|
|
|
|
rtcp_sender->SendRTCP(feedback_state(), kRtcpRemb);
|
|
|
|
EXPECT_EQ(1, parser()->remb()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->remb()->sender_ssrc());
|
|
EXPECT_EQ(kBitrate, parser()->remb()->bitrate_bps());
|
|
EXPECT_THAT(parser()->remb()->ssrcs(),
|
|
ElementsAre(kRemoteSsrc, kRemoteSsrc + 1));
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, RembIncludedInEachCompoundPacketAfterSet) {
|
|
const int kBitrate = 261011;
|
|
const std::vector<uint32_t> kSsrcs = {kRemoteSsrc, kRemoteSsrc + 1};
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetRemb(kBitrate, kSsrcs);
|
|
|
|
rtcp_sender->SendRTCP(feedback_state(), kRtcpReport);
|
|
EXPECT_EQ(1, parser()->remb()->num_packets());
|
|
// REMB should be included in each compound packet.
|
|
rtcp_sender->SendRTCP(feedback_state(), kRtcpReport);
|
|
EXPECT_EQ(2, parser()->remb()->num_packets());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendXrWithDlrr) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
RTCPSender::FeedbackState feedback_state = rtp_rtcp_impl_->GetFeedbackState();
|
|
rtcp::ReceiveTimeInfo last_xr_rr;
|
|
last_xr_rr.ssrc = 0x11111111;
|
|
last_xr_rr.last_rr = 0x22222222;
|
|
last_xr_rr.delay_since_last_rr = 0x33333333;
|
|
feedback_state.last_xr_rtis.push_back(last_xr_rr);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state, kRtcpReport));
|
|
EXPECT_EQ(1, parser()->xr()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->xr()->sender_ssrc());
|
|
ASSERT_THAT(parser()->xr()->dlrr().sub_blocks(), SizeIs(1));
|
|
EXPECT_EQ(last_xr_rr.ssrc, parser()->xr()->dlrr().sub_blocks()[0].ssrc);
|
|
EXPECT_EQ(last_xr_rr.last_rr, parser()->xr()->dlrr().sub_blocks()[0].last_rr);
|
|
EXPECT_EQ(last_xr_rr.delay_since_last_rr,
|
|
parser()->xr()->dlrr().sub_blocks()[0].delay_since_last_rr);
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendXrWithMultipleDlrrSubBlocks) {
|
|
const size_t kNumReceivers = 2;
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
RTCPSender::FeedbackState feedback_state = rtp_rtcp_impl_->GetFeedbackState();
|
|
for (size_t i = 0; i < kNumReceivers; ++i) {
|
|
rtcp::ReceiveTimeInfo last_xr_rr;
|
|
last_xr_rr.ssrc = i;
|
|
last_xr_rr.last_rr = (i + 1) * 100;
|
|
last_xr_rr.delay_since_last_rr = (i + 2) * 200;
|
|
feedback_state.last_xr_rtis.push_back(last_xr_rr);
|
|
}
|
|
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state, kRtcpReport));
|
|
EXPECT_EQ(1, parser()->xr()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->xr()->sender_ssrc());
|
|
ASSERT_THAT(parser()->xr()->dlrr().sub_blocks(), SizeIs(kNumReceivers));
|
|
for (size_t i = 0; i < kNumReceivers; ++i) {
|
|
EXPECT_EQ(feedback_state.last_xr_rtis[i].ssrc,
|
|
parser()->xr()->dlrr().sub_blocks()[i].ssrc);
|
|
EXPECT_EQ(feedback_state.last_xr_rtis[i].last_rr,
|
|
parser()->xr()->dlrr().sub_blocks()[i].last_rr);
|
|
EXPECT_EQ(feedback_state.last_xr_rtis[i].delay_since_last_rr,
|
|
parser()->xr()->dlrr().sub_blocks()[i].delay_since_last_rr);
|
|
}
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendXrWithRrtr) {
|
|
RTCPSender::Configuration config = GetDefaultConfig();
|
|
config.non_sender_rtt_measurement = true;
|
|
auto rtcp_sender = CreateRtcpSender(config);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), false);
|
|
NtpTime ntp = clock_.CurrentNtpTime();
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
EXPECT_EQ(1, parser()->xr()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->xr()->sender_ssrc());
|
|
EXPECT_FALSE(parser()->xr()->dlrr());
|
|
ASSERT_TRUE(parser()->xr()->rrtr());
|
|
EXPECT_EQ(ntp, parser()->xr()->rrtr()->ntp());
|
|
}
|
|
|
|
// Same test as above, but enable Rrtr with the setter.
|
|
TEST_F(RtcpSenderTest, SendXrWithRrtrUsingSetter) {
|
|
RTCPSender::Configuration config = GetDefaultConfig();
|
|
config.non_sender_rtt_measurement = false;
|
|
auto rtcp_sender = CreateRtcpSender(config);
|
|
rtcp_sender->SetNonSenderRttMeasurement(true);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), false);
|
|
NtpTime ntp = clock_.CurrentNtpTime();
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
EXPECT_EQ(1, parser()->xr()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->xr()->sender_ssrc());
|
|
EXPECT_FALSE(parser()->xr()->dlrr());
|
|
ASSERT_TRUE(parser()->xr()->rrtr());
|
|
EXPECT_EQ(ntp, parser()->xr()->rrtr()->ntp());
|
|
}
|
|
|
|
// Same test as above, but disable Rrtr with the setter.
|
|
TEST_F(RtcpSenderTest, SendsNoRrtrUsingSetter) {
|
|
RTCPSender::Configuration config = GetDefaultConfig();
|
|
config.non_sender_rtt_measurement = true;
|
|
auto rtcp_sender = CreateRtcpSender(config);
|
|
rtcp_sender->SetNonSenderRttMeasurement(false);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), false);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
EXPECT_EQ(0, parser()->xr()->num_packets());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, TestNoXrRrtrSentIfSending) {
|
|
RTCPSender::Configuration config = GetDefaultConfig();
|
|
config.non_sender_rtt_measurement = true;
|
|
auto rtcp_sender = CreateRtcpSender(config);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), true);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
EXPECT_EQ(0, parser()->xr()->num_packets());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, TestNoXrRrtrSentIfNotEnabled) {
|
|
RTCPSender::Configuration config = GetDefaultConfig();
|
|
config.non_sender_rtt_measurement = false;
|
|
auto rtcp_sender = CreateRtcpSender(config);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), false);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
EXPECT_EQ(0, parser()->xr()->num_packets());
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, TestRegisterRtcpPacketTypeObserver) {
|
|
RtcpPacketTypeCounterObserverImpl observer;
|
|
RTCPSender::Configuration config;
|
|
config.clock = &clock_;
|
|
config.receive_statistics = receive_statistics_.get();
|
|
config.outgoing_transport = &test_transport_;
|
|
config.rtcp_packet_type_counter_observer = &observer;
|
|
config.rtcp_report_interval = TimeDelta::Millis(1000);
|
|
auto rtcp_sender = CreateRtcpSender(config);
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpPli));
|
|
EXPECT_EQ(1, parser()->pli()->num_packets());
|
|
EXPECT_EQ(kRemoteSsrc, observer.ssrc_);
|
|
EXPECT_EQ(1U, observer.counter_.pli_packets);
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendTmmbr) {
|
|
const unsigned int kBitrateBps = 312000;
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
rtcp_sender->SetTargetBitrate(kBitrateBps);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpTmmbr));
|
|
EXPECT_EQ(1, parser()->tmmbr()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->tmmbr()->sender_ssrc());
|
|
EXPECT_EQ(1U, parser()->tmmbr()->requests().size());
|
|
EXPECT_EQ(kBitrateBps, parser()->tmmbr()->requests()[0].bitrate_bps());
|
|
// TODO(asapersson): tmmbr_item()->Overhead() looks broken, always zero.
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendTmmbn) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), true);
|
|
std::vector<rtcp::TmmbItem> bounding_set;
|
|
const uint32_t kBitrateBps = 32768000;
|
|
const uint32_t kPacketOh = 40;
|
|
const uint32_t kSourceSsrc = 12345;
|
|
const rtcp::TmmbItem tmmbn(kSourceSsrc, kBitrateBps, kPacketOh);
|
|
bounding_set.push_back(tmmbn);
|
|
rtcp_sender->SetTmmbn(bounding_set);
|
|
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpSr));
|
|
EXPECT_EQ(1, parser()->sender_report()->num_packets());
|
|
EXPECT_EQ(1, parser()->tmmbn()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->tmmbn()->sender_ssrc());
|
|
EXPECT_EQ(1U, parser()->tmmbn()->items().size());
|
|
EXPECT_EQ(kBitrateBps, parser()->tmmbn()->items()[0].bitrate_bps());
|
|
EXPECT_EQ(kPacketOh, parser()->tmmbn()->items()[0].packet_overhead());
|
|
EXPECT_EQ(kSourceSsrc, parser()->tmmbn()->items()[0].ssrc());
|
|
}
|
|
|
|
// This test is written to verify actual behaviour. It does not seem
|
|
// to make much sense to send an empty TMMBN, since there is no place
|
|
// to put an actual limit here. It's just information that no limit
|
|
// is set, which is kind of the starting assumption.
|
|
// See http://code.google.com/p/webrtc/issues/detail?id=468 for one
|
|
// situation where this caused confusion.
|
|
TEST_F(RtcpSenderTest, SendsTmmbnIfSetAndEmpty) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetSendingStatus(feedback_state(), true);
|
|
std::vector<rtcp::TmmbItem> bounding_set;
|
|
rtcp_sender->SetTmmbn(bounding_set);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpSr));
|
|
EXPECT_EQ(1, parser()->sender_report()->num_packets());
|
|
EXPECT_EQ(1, parser()->tmmbn()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->tmmbn()->sender_ssrc());
|
|
EXPECT_EQ(0U, parser()->tmmbn()->items().size());
|
|
}
|
|
|
|
// This test is written to verify that BYE is always the last packet
|
|
// type in a RTCP compoud packet. The rtcp_sender is recreated with
|
|
// mock_transport, which is used to check for whether BYE at the end
|
|
// of a RTCP compound packet.
|
|
TEST_F(RtcpSenderTest, ByeMustBeLast) {
|
|
MockTransport mock_transport;
|
|
EXPECT_CALL(mock_transport, SendRtcp(_, _))
|
|
.WillOnce(Invoke([](const uint8_t* data, size_t len) {
|
|
const uint8_t* next_packet = data;
|
|
const uint8_t* const packet_end = data + len;
|
|
rtcp::CommonHeader packet;
|
|
while (next_packet < packet_end) {
|
|
EXPECT_TRUE(packet.Parse(next_packet, packet_end - next_packet));
|
|
next_packet = packet.NextPacket();
|
|
if (packet.type() ==
|
|
rtcp::Bye::kPacketType) // Main test expectation.
|
|
EXPECT_EQ(0, packet_end - next_packet)
|
|
<< "Bye packet should be last in a compound RTCP packet.";
|
|
if (next_packet == packet_end) // Validate test was set correctly.
|
|
EXPECT_EQ(packet.type(), rtcp::Bye::kPacketType)
|
|
<< "Last packet in this test expected to be Bye.";
|
|
}
|
|
|
|
return true;
|
|
}));
|
|
|
|
// Re-configure rtcp_sender with mock_transport_
|
|
RTCPSender::Configuration config;
|
|
config.clock = &clock_;
|
|
config.receive_statistics = receive_statistics_.get();
|
|
config.outgoing_transport = &mock_transport;
|
|
config.rtcp_report_interval = TimeDelta::Millis(1000);
|
|
config.local_media_ssrc = kSenderSsrc;
|
|
auto rtcp_sender = CreateRtcpSender(config);
|
|
|
|
rtcp_sender->SetTimestampOffset(kStartRtpTimestamp);
|
|
rtcp_sender->SetLastRtpTime(kRtpTimestamp, clock_.CurrentTime(),
|
|
/*payload_type=*/0);
|
|
|
|
// Set up REMB info to be included with BYE.
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
rtcp_sender->SetRemb(1234, {});
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpBye));
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendXrWithTargetBitrate) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
const size_t kNumSpatialLayers = 2;
|
|
const size_t kNumTemporalLayers = 2;
|
|
VideoBitrateAllocation allocation;
|
|
for (size_t sl = 0; sl < kNumSpatialLayers; ++sl) {
|
|
uint32_t start_bitrate_bps = (sl + 1) * 100000;
|
|
for (size_t tl = 0; tl < kNumTemporalLayers; ++tl)
|
|
allocation.SetBitrate(sl, tl, start_bitrate_bps + (tl * 20000));
|
|
}
|
|
rtcp_sender->SetVideoBitrateAllocation(allocation);
|
|
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
EXPECT_EQ(1, parser()->xr()->num_packets());
|
|
EXPECT_EQ(kSenderSsrc, parser()->xr()->sender_ssrc());
|
|
const absl::optional<rtcp::TargetBitrate>& target_bitrate =
|
|
parser()->xr()->target_bitrate();
|
|
ASSERT_TRUE(target_bitrate);
|
|
const std::vector<rtcp::TargetBitrate::BitrateItem>& bitrates =
|
|
target_bitrate->GetTargetBitrates();
|
|
EXPECT_EQ(kNumSpatialLayers * kNumTemporalLayers, bitrates.size());
|
|
|
|
for (size_t sl = 0; sl < kNumSpatialLayers; ++sl) {
|
|
uint32_t start_bitrate_bps = (sl + 1) * 100000;
|
|
for (size_t tl = 0; tl < kNumTemporalLayers; ++tl) {
|
|
size_t index = (sl * kNumSpatialLayers) + tl;
|
|
const rtcp::TargetBitrate::BitrateItem& item = bitrates[index];
|
|
EXPECT_EQ(sl, item.spatial_layer);
|
|
EXPECT_EQ(tl, item.temporal_layer);
|
|
EXPECT_EQ(start_bitrate_bps + (tl * 20000),
|
|
item.target_bitrate_kbps * 1000);
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendImmediateXrWithTargetBitrate) {
|
|
// Initialize. Send a first report right away.
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
clock_.AdvanceTimeMilliseconds(5);
|
|
|
|
// Video bitrate allocation generated, save until next time we send a report.
|
|
VideoBitrateAllocation allocation;
|
|
allocation.SetBitrate(0, 0, 100000);
|
|
rtcp_sender->SetVideoBitrateAllocation(allocation);
|
|
// First seen instance will be sent immediately.
|
|
EXPECT_TRUE(rtcp_sender->TimeToSendRTCPReport(false));
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
clock_.AdvanceTimeMilliseconds(5);
|
|
|
|
// Update bitrate of existing layer, does not quality for immediate sending.
|
|
allocation.SetBitrate(0, 0, 150000);
|
|
rtcp_sender->SetVideoBitrateAllocation(allocation);
|
|
EXPECT_FALSE(rtcp_sender->TimeToSendRTCPReport(false));
|
|
|
|
// A new spatial layer enabled, signal this as soon as possible.
|
|
allocation.SetBitrate(1, 0, 200000);
|
|
rtcp_sender->SetVideoBitrateAllocation(allocation);
|
|
EXPECT_TRUE(rtcp_sender->TimeToSendRTCPReport(false));
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
clock_.AdvanceTimeMilliseconds(5);
|
|
|
|
// Explicitly disable top layer. The same set of layers now has a bitrate
|
|
// defined, but the explicit 0 indicates shutdown. Signal immediately.
|
|
allocation.SetBitrate(1, 0, 0);
|
|
EXPECT_FALSE(rtcp_sender->TimeToSendRTCPReport(false));
|
|
rtcp_sender->SetVideoBitrateAllocation(allocation);
|
|
EXPECT_TRUE(rtcp_sender->TimeToSendRTCPReport(false));
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendTargetBitrateExplicitZeroOnStreamRemoval) {
|
|
// Set up and send a bitrate allocation with two layers.
|
|
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kCompound);
|
|
VideoBitrateAllocation allocation;
|
|
allocation.SetBitrate(0, 0, 100000);
|
|
allocation.SetBitrate(1, 0, 200000);
|
|
rtcp_sender->SetVideoBitrateAllocation(allocation);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
absl::optional<rtcp::TargetBitrate> target_bitrate =
|
|
parser()->xr()->target_bitrate();
|
|
ASSERT_TRUE(target_bitrate);
|
|
std::vector<rtcp::TargetBitrate::BitrateItem> bitrates =
|
|
target_bitrate->GetTargetBitrates();
|
|
ASSERT_EQ(2u, bitrates.size());
|
|
EXPECT_EQ(bitrates[0].target_bitrate_kbps,
|
|
allocation.GetBitrate(0, 0) / 1000);
|
|
EXPECT_EQ(bitrates[1].target_bitrate_kbps,
|
|
allocation.GetBitrate(1, 0) / 1000);
|
|
|
|
// Create a new allocation, where the second stream is no longer available.
|
|
VideoBitrateAllocation new_allocation;
|
|
new_allocation.SetBitrate(0, 0, 150000);
|
|
rtcp_sender->SetVideoBitrateAllocation(new_allocation);
|
|
EXPECT_EQ(0, rtcp_sender->SendRTCP(feedback_state(), kRtcpReport));
|
|
target_bitrate = parser()->xr()->target_bitrate();
|
|
ASSERT_TRUE(target_bitrate);
|
|
bitrates = target_bitrate->GetTargetBitrates();
|
|
|
|
// Two bitrates should still be set, with an explicit entry indicating the
|
|
// removed stream is gone.
|
|
ASSERT_EQ(2u, bitrates.size());
|
|
EXPECT_EQ(bitrates[0].target_bitrate_kbps,
|
|
new_allocation.GetBitrate(0, 0) / 1000);
|
|
EXPECT_EQ(bitrates[1].target_bitrate_kbps, 0u);
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, DoesntSchedulesInitialReportWhenSsrcSetOnConstruction) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
rtcp_sender->SetRemoteSSRC(kRemoteSsrc);
|
|
// New report should not have been scheduled yet.
|
|
clock_.AdvanceTimeMilliseconds(100);
|
|
EXPECT_FALSE(rtcp_sender->TimeToSendRTCPReport(false));
|
|
}
|
|
|
|
TEST_F(RtcpSenderTest, SendsCombinedRtcpPacket) {
|
|
auto rtcp_sender = CreateRtcpSender(GetDefaultConfig());
|
|
rtcp_sender->SetRTCPStatus(RtcpMode::kReducedSize);
|
|
|
|
std::vector<std::unique_ptr<rtcp::RtcpPacket>> packets;
|
|
auto transport_feedback = std::make_unique<rtcp::TransportFeedback>();
|
|
transport_feedback->AddReceivedPacket(321, Timestamp::Millis(10));
|
|
packets.push_back(std::move(transport_feedback));
|
|
auto remote_estimate = std::make_unique<rtcp::RemoteEstimate>();
|
|
packets.push_back(std::move(remote_estimate));
|
|
rtcp_sender->SendCombinedRtcpPacket(std::move(packets));
|
|
|
|
EXPECT_EQ(parser()->transport_feedback()->num_packets(), 1);
|
|
EXPECT_EQ(parser()->transport_feedback()->sender_ssrc(), kSenderSsrc);
|
|
EXPECT_EQ(parser()->app()->num_packets(), 1);
|
|
EXPECT_EQ(parser()->app()->sender_ssrc(), kSenderSsrc);
|
|
}
|
|
|
|
} // namespace webrtc
|