|
|
|
|
@ -24,6 +24,7 @@
|
|
|
|
|
#include "test/time_controller/simulated_time_controller.h"
|
|
|
|
|
|
|
|
|
|
using ::testing::_;
|
|
|
|
|
using ::testing::AtLeast;
|
|
|
|
|
using ::testing::Return;
|
|
|
|
|
using ::testing::SaveArg;
|
|
|
|
|
|
|
|
|
|
@ -48,17 +49,6 @@ class MockPacketRouter : public PacketRouter {
|
|
|
|
|
|
|
|
|
|
namespace test {
|
|
|
|
|
|
|
|
|
|
class TaskQueuePacedSenderTest : public ::testing::Test {
|
|
|
|
|
public:
|
|
|
|
|
TaskQueuePacedSenderTest()
|
|
|
|
|
: time_controller_(Timestamp::Millis(1234)),
|
|
|
|
|
pacer_(time_controller_.GetClock(),
|
|
|
|
|
&packet_router_,
|
|
|
|
|
/*event_log=*/nullptr,
|
|
|
|
|
/*field_trials=*/nullptr,
|
|
|
|
|
time_controller_.GetTaskQueueFactory()) {}
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
std::unique_ptr<RtpPacketToSend> BuildRtpPacket(RtpPacketMediaType type) {
|
|
|
|
|
auto packet = std::make_unique<RtpPacketToSend>(nullptr);
|
|
|
|
|
packet->set_packet_type(type);
|
|
|
|
|
@ -92,56 +82,65 @@ class TaskQueuePacedSenderTest : public ::testing::Test {
|
|
|
|
|
return packets;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Timestamp CurrentTime() { return time_controller_.GetClock()->CurrentTime(); }
|
|
|
|
|
TEST(TaskQueuePacedSenderTest, PacesPackets) {
|
|
|
|
|
GlobalSimulatedTimeController time_controller(Timestamp::Millis(1234));
|
|
|
|
|
MockPacketRouter packet_router;
|
|
|
|
|
TaskQueuePacedSender pacer(time_controller.GetClock(), &packet_router,
|
|
|
|
|
/*event_log=*/nullptr,
|
|
|
|
|
/*field_trials=*/nullptr,
|
|
|
|
|
time_controller.GetTaskQueueFactory(),
|
|
|
|
|
PacingController::kMinSleepTime);
|
|
|
|
|
|
|
|
|
|
GlobalSimulatedTimeController time_controller_;
|
|
|
|
|
MockPacketRouter packet_router_;
|
|
|
|
|
TaskQueuePacedSender pacer_;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
TEST_F(TaskQueuePacedSenderTest, PacesPackets) {
|
|
|
|
|
// Insert a number of packets, covering one second.
|
|
|
|
|
static constexpr size_t kPacketsToSend = 42;
|
|
|
|
|
pacer_.SetPacingRates(
|
|
|
|
|
pacer.SetPacingRates(
|
|
|
|
|
DataRate::BitsPerSec(kDefaultPacketSize * 8 * kPacketsToSend),
|
|
|
|
|
DataRate::Zero());
|
|
|
|
|
pacer_.EnqueuePackets(
|
|
|
|
|
pacer.EnqueuePackets(
|
|
|
|
|
GeneratePackets(RtpPacketMediaType::kVideo, kPacketsToSend));
|
|
|
|
|
|
|
|
|
|
// Expect all of them to be sent.
|
|
|
|
|
size_t packets_sent = 0;
|
|
|
|
|
Timestamp end_time = Timestamp::PlusInfinity();
|
|
|
|
|
EXPECT_CALL(packet_router_, SendPacket)
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket)
|
|
|
|
|
.WillRepeatedly([&](std::unique_ptr<RtpPacketToSend> packet,
|
|
|
|
|
const PacedPacketInfo& cluster_info) {
|
|
|
|
|
++packets_sent;
|
|
|
|
|
if (packets_sent == kPacketsToSend) {
|
|
|
|
|
end_time = time_controller_.GetClock()->CurrentTime();
|
|
|
|
|
end_time = time_controller.GetClock()->CurrentTime();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const Timestamp start_time = time_controller_.GetClock()->CurrentTime();
|
|
|
|
|
const Timestamp start_time = time_controller.GetClock()->CurrentTime();
|
|
|
|
|
|
|
|
|
|
// Packets should be sent over a period of close to 1s. Expect a little lower
|
|
|
|
|
// than this since initial probing is a bit quicker.
|
|
|
|
|
time_controller_.AdvanceTime(TimeDelta::Seconds(1));
|
|
|
|
|
// Packets should be sent over a period of close to 1s. Expect a little
|
|
|
|
|
// lower than this since initial probing is a bit quicker.
|
|
|
|
|
time_controller.AdvanceTime(TimeDelta::Seconds(1));
|
|
|
|
|
EXPECT_EQ(packets_sent, kPacketsToSend);
|
|
|
|
|
ASSERT_TRUE(end_time.IsFinite());
|
|
|
|
|
EXPECT_NEAR((end_time - start_time).ms<double>(), 1000.0, 50.0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(TaskQueuePacedSenderTest, ReschedulesProcessOnRateChange) {
|
|
|
|
|
GlobalSimulatedTimeController time_controller(Timestamp::Millis(1234));
|
|
|
|
|
MockPacketRouter packet_router;
|
|
|
|
|
TaskQueuePacedSender pacer(time_controller.GetClock(), &packet_router,
|
|
|
|
|
/*event_log=*/nullptr,
|
|
|
|
|
/*field_trials=*/nullptr,
|
|
|
|
|
time_controller.GetTaskQueueFactory(),
|
|
|
|
|
PacingController::kMinSleepTime);
|
|
|
|
|
|
|
|
|
|
TEST_F(TaskQueuePacedSenderTest, ReschedulesProcessOnRateChange) {
|
|
|
|
|
// Insert a number of packets to be sent 200ms apart.
|
|
|
|
|
const size_t kPacketsPerSecond = 5;
|
|
|
|
|
const DataRate kPacingRate =
|
|
|
|
|
DataRate::BitsPerSec(kDefaultPacketSize * 8 * kPacketsPerSecond);
|
|
|
|
|
pacer_.SetPacingRates(kPacingRate, DataRate::Zero());
|
|
|
|
|
pacer.SetPacingRates(kPacingRate, DataRate::Zero());
|
|
|
|
|
|
|
|
|
|
// Send some initial packets to be rid of any probes.
|
|
|
|
|
EXPECT_CALL(packet_router_, SendPacket).Times(kPacketsPerSecond);
|
|
|
|
|
pacer_.EnqueuePackets(
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket).Times(kPacketsPerSecond);
|
|
|
|
|
pacer.EnqueuePackets(
|
|
|
|
|
GeneratePackets(RtpPacketMediaType::kVideo, kPacketsPerSecond));
|
|
|
|
|
time_controller_.AdvanceTime(TimeDelta::Seconds(1));
|
|
|
|
|
time_controller.AdvanceTime(TimeDelta::Seconds(1));
|
|
|
|
|
|
|
|
|
|
// Insert three packets, and record send time of each of them.
|
|
|
|
|
// After the second packet is sent, double the send rate so we can
|
|
|
|
|
@ -150,51 +149,126 @@ TEST_F(TaskQueuePacedSenderTest, ReschedulesProcessOnRateChange) {
|
|
|
|
|
Timestamp second_packet_time = Timestamp::MinusInfinity();
|
|
|
|
|
Timestamp third_packet_time = Timestamp::MinusInfinity();
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(packet_router_, SendPacket)
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket)
|
|
|
|
|
.Times(3)
|
|
|
|
|
.WillRepeatedly([&](std::unique_ptr<RtpPacketToSend> packet,
|
|
|
|
|
const PacedPacketInfo& cluster_info) {
|
|
|
|
|
if (first_packet_time.IsInfinite()) {
|
|
|
|
|
first_packet_time = CurrentTime();
|
|
|
|
|
first_packet_time = time_controller.GetClock()->CurrentTime();
|
|
|
|
|
} else if (second_packet_time.IsInfinite()) {
|
|
|
|
|
second_packet_time = CurrentTime();
|
|
|
|
|
pacer_.SetPacingRates(2 * kPacingRate, DataRate::Zero());
|
|
|
|
|
second_packet_time = time_controller.GetClock()->CurrentTime();
|
|
|
|
|
pacer.SetPacingRates(2 * kPacingRate, DataRate::Zero());
|
|
|
|
|
} else {
|
|
|
|
|
third_packet_time = CurrentTime();
|
|
|
|
|
third_packet_time = time_controller.GetClock()->CurrentTime();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
pacer_.EnqueuePackets(GeneratePackets(RtpPacketMediaType::kVideo, 3));
|
|
|
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(500));
|
|
|
|
|
pacer.EnqueuePackets(GeneratePackets(RtpPacketMediaType::kVideo, 3));
|
|
|
|
|
time_controller.AdvanceTime(TimeDelta::Millis(500));
|
|
|
|
|
ASSERT_TRUE(third_packet_time.IsFinite());
|
|
|
|
|
EXPECT_NEAR((second_packet_time - first_packet_time).ms<double>(), 200.0,
|
|
|
|
|
1.0);
|
|
|
|
|
EXPECT_NEAR((third_packet_time - second_packet_time).ms<double>(), 100.0,
|
|
|
|
|
1.0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(TaskQueuePacedSenderTest, SendsAudioImmediately) {
|
|
|
|
|
GlobalSimulatedTimeController time_controller(Timestamp::Millis(1234));
|
|
|
|
|
MockPacketRouter packet_router;
|
|
|
|
|
TaskQueuePacedSender pacer(time_controller.GetClock(), &packet_router,
|
|
|
|
|
/*event_log=*/nullptr,
|
|
|
|
|
/*field_trials=*/nullptr,
|
|
|
|
|
time_controller.GetTaskQueueFactory(),
|
|
|
|
|
PacingController::kMinSleepTime);
|
|
|
|
|
|
|
|
|
|
TEST_F(TaskQueuePacedSenderTest, SendsAudioImmediately) {
|
|
|
|
|
const DataRate kPacingDataRate = DataRate::KilobitsPerSec(125);
|
|
|
|
|
const DataSize kPacketSize = DataSize::Bytes(kDefaultPacketSize);
|
|
|
|
|
const TimeDelta kPacketPacingTime = kPacketSize / kPacingDataRate;
|
|
|
|
|
|
|
|
|
|
pacer_.SetPacingRates(kPacingDataRate, DataRate::Zero());
|
|
|
|
|
pacer.SetPacingRates(kPacingDataRate, DataRate::Zero());
|
|
|
|
|
|
|
|
|
|
// Add some initial video packets, only one should be sent.
|
|
|
|
|
EXPECT_CALL(packet_router_, SendPacket);
|
|
|
|
|
pacer_.EnqueuePackets(GeneratePackets(RtpPacketMediaType::kVideo, 10));
|
|
|
|
|
time_controller_.AdvanceTime(TimeDelta::Zero());
|
|
|
|
|
::testing::Mock::VerifyAndClearExpectations(&packet_router_);
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket);
|
|
|
|
|
pacer.EnqueuePackets(GeneratePackets(RtpPacketMediaType::kVideo, 10));
|
|
|
|
|
time_controller.AdvanceTime(TimeDelta::Zero());
|
|
|
|
|
::testing::Mock::VerifyAndClearExpectations(&packet_router);
|
|
|
|
|
|
|
|
|
|
// Advance time, but still before next packet should be sent.
|
|
|
|
|
time_controller_.AdvanceTime(kPacketPacingTime / 2);
|
|
|
|
|
time_controller.AdvanceTime(kPacketPacingTime / 2);
|
|
|
|
|
|
|
|
|
|
// Insert an audio packet, it should be sent immediately.
|
|
|
|
|
EXPECT_CALL(packet_router_, SendPacket);
|
|
|
|
|
pacer_.EnqueuePackets(GeneratePackets(RtpPacketMediaType::kAudio, 1));
|
|
|
|
|
time_controller_.AdvanceTime(TimeDelta::Zero());
|
|
|
|
|
::testing::Mock::VerifyAndClearExpectations(&packet_router_);
|
|
|
|
|
}
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket);
|
|
|
|
|
pacer.EnqueuePackets(GeneratePackets(RtpPacketMediaType::kAudio, 1));
|
|
|
|
|
time_controller.AdvanceTime(TimeDelta::Zero());
|
|
|
|
|
::testing::Mock::VerifyAndClearExpectations(&packet_router);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(TaskQueuePacedSenderTest, SleepsDuringCoalscingWindow) {
|
|
|
|
|
const TimeDelta kCoalescingWindow = TimeDelta::Millis(5);
|
|
|
|
|
GlobalSimulatedTimeController time_controller(Timestamp::Millis(1234));
|
|
|
|
|
MockPacketRouter packet_router;
|
|
|
|
|
TaskQueuePacedSender pacer(time_controller.GetClock(), &packet_router,
|
|
|
|
|
/*event_log=*/nullptr,
|
|
|
|
|
/*field_trials=*/nullptr,
|
|
|
|
|
time_controller.GetTaskQueueFactory(),
|
|
|
|
|
kCoalescingWindow);
|
|
|
|
|
|
|
|
|
|
// Set rates so one packet adds one ms of buffer level.
|
|
|
|
|
const DataSize kPacketSize = DataSize::Bytes(kDefaultPacketSize);
|
|
|
|
|
const TimeDelta kPacketPacingTime = TimeDelta::Millis(1);
|
|
|
|
|
const DataRate kPacingDataRate = kPacketSize / kPacketPacingTime;
|
|
|
|
|
|
|
|
|
|
pacer.SetPacingRates(kPacingDataRate, DataRate::Zero());
|
|
|
|
|
|
|
|
|
|
// Add 10 packets. The first should be sent immediately since the buffers
|
|
|
|
|
// are clear.
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket);
|
|
|
|
|
pacer.EnqueuePackets(GeneratePackets(RtpPacketMediaType::kVideo, 10));
|
|
|
|
|
time_controller.AdvanceTime(TimeDelta::Zero());
|
|
|
|
|
::testing::Mock::VerifyAndClearExpectations(&packet_router);
|
|
|
|
|
|
|
|
|
|
// Advance time to 1ms before the coalescing window ends. No packets should
|
|
|
|
|
// be sent.
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket).Times(0);
|
|
|
|
|
time_controller.AdvanceTime(kCoalescingWindow - TimeDelta::Millis(1));
|
|
|
|
|
|
|
|
|
|
// Advance time to where coalescing window ends. All packets that should
|
|
|
|
|
// have been sent up til now will be sent.
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket).Times(5);
|
|
|
|
|
time_controller.AdvanceTime(TimeDelta::Millis(1));
|
|
|
|
|
::testing::Mock::VerifyAndClearExpectations(&packet_router);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(TaskQueuePacedSenderTest, ProbingOverridesCoalescingWindow) {
|
|
|
|
|
const TimeDelta kCoalescingWindow = TimeDelta::Millis(5);
|
|
|
|
|
GlobalSimulatedTimeController time_controller(Timestamp::Millis(1234));
|
|
|
|
|
MockPacketRouter packet_router;
|
|
|
|
|
TaskQueuePacedSender pacer(time_controller.GetClock(), &packet_router,
|
|
|
|
|
/*event_log=*/nullptr,
|
|
|
|
|
/*field_trials=*/nullptr,
|
|
|
|
|
time_controller.GetTaskQueueFactory(),
|
|
|
|
|
kCoalescingWindow);
|
|
|
|
|
|
|
|
|
|
// Set rates so one packet adds one ms of buffer level.
|
|
|
|
|
const DataSize kPacketSize = DataSize::Bytes(kDefaultPacketSize);
|
|
|
|
|
const TimeDelta kPacketPacingTime = TimeDelta::Millis(1);
|
|
|
|
|
const DataRate kPacingDataRate = kPacketSize / kPacketPacingTime;
|
|
|
|
|
|
|
|
|
|
pacer.SetPacingRates(kPacingDataRate, DataRate::Zero());
|
|
|
|
|
|
|
|
|
|
// Add 10 packets. The first should be sent immediately since the buffers
|
|
|
|
|
// are clear. This will also trigger the probe to start.
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket).Times(AtLeast(1));
|
|
|
|
|
pacer.CreateProbeCluster(kPacingDataRate * 2, 17);
|
|
|
|
|
pacer.EnqueuePackets(GeneratePackets(RtpPacketMediaType::kVideo, 10));
|
|
|
|
|
time_controller.AdvanceTime(TimeDelta::Zero());
|
|
|
|
|
::testing::Mock::VerifyAndClearExpectations(&packet_router);
|
|
|
|
|
|
|
|
|
|
// Advance time to 1ms before the coalescing window ends. Packets should be
|
|
|
|
|
// flying.
|
|
|
|
|
EXPECT_CALL(packet_router, SendPacket).Times(AtLeast(1));
|
|
|
|
|
time_controller.AdvanceTime(kCoalescingWindow - TimeDelta::Millis(1));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace test
|
|
|
|
|
} // namespace webrtc
|
|
|
|
|
|