blob: 09300fb6359502500fa3351ae07aed9e407ec020 [file] [log] [blame]
philipelbe7a9e52016-05-19 12:19:35 +02001/*
2 * Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "modules/video_coding/frame_buffer2.h"
philipelbe7a9e52016-05-19 12:19:35 +020012
13#include <algorithm>
14#include <cstring>
15#include <limits>
Mirko Bonadei317a1f02019-09-17 17:06:18 +020016#include <memory>
philipelbe7a9e52016-05-19 12:19:35 +020017#include <vector>
18
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020019#include "modules/video_coding/frame_object.h"
20#include "modules/video_coding/jitter_estimator.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020021#include "modules/video_coding/timing.h"
Bjorn Tereliusa194e582017-10-25 13:07:09 +020022#include "rtc_base/numerics/sequence_number_util.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020023#include "rtc_base/platform_thread.h"
24#include "rtc_base/random.h"
25#include "system_wrappers/include/clock.h"
Niels Möller7cca0422019-04-29 16:12:19 +020026#include "test/field_trial.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020027#include "test/gmock.h"
28#include "test/gtest.h"
philipelbe7a9e52016-05-19 12:19:35 +020029
Mirko Bonadei6a489f22019-04-09 15:11:12 +020030using ::testing::_;
31using ::testing::Return;
philipela45102f2017-02-22 05:30:39 -080032
philipelbe7a9e52016-05-19 12:19:35 +020033namespace webrtc {
34namespace video_coding {
35
36class VCMTimingFake : public VCMTiming {
37 public:
38 explicit VCMTimingFake(Clock* clock) : VCMTiming(clock) {}
39
40 int64_t RenderTimeMs(uint32_t frame_timestamp,
41 int64_t now_ms) const override {
42 if (last_ms_ == -1) {
43 last_ms_ = now_ms + kDelayMs;
44 last_timestamp_ = frame_timestamp;
45 }
46
47 uint32_t diff = MinDiff(frame_timestamp, last_timestamp_);
48 if (AheadOf(frame_timestamp, last_timestamp_))
49 last_ms_ += diff / 90;
50 else
51 last_ms_ -= diff / 90;
52
53 last_timestamp_ = frame_timestamp;
54 return last_ms_;
55 }
56
Ilya Nikolaevskiy8c4fe162018-02-27 15:49:47 +010057 int64_t MaxWaitingTime(int64_t render_time_ms,
58 int64_t now_ms) const override {
59 return render_time_ms - now_ms - kDecodeTime;
philipelbe7a9e52016-05-19 12:19:35 +020060 }
61
Johannes Kronbfd343b2019-07-01 10:07:50 +020062 bool GetTimings(int* max_decode_ms,
philipela45102f2017-02-22 05:30:39 -080063 int* current_delay_ms,
64 int* target_delay_ms,
65 int* jitter_buffer_ms,
66 int* min_playout_delay_ms,
67 int* render_delay_ms) const override {
68 return true;
69 }
70
Niels Möller7cca0422019-04-29 16:12:19 +020071 int GetCurrentJitter() {
Niels Möller7cca0422019-04-29 16:12:19 +020072 int max_decode_ms;
73 int current_delay_ms;
74 int target_delay_ms;
75 int jitter_buffer_ms;
76 int min_playout_delay_ms;
77 int render_delay_ms;
Johannes Kronbfd343b2019-07-01 10:07:50 +020078 VCMTiming::GetTimings(&max_decode_ms, &current_delay_ms, &target_delay_ms,
79 &jitter_buffer_ms, &min_playout_delay_ms,
80 &render_delay_ms);
Niels Möller7cca0422019-04-29 16:12:19 +020081 return jitter_buffer_ms;
82 }
83
philipelbe7a9e52016-05-19 12:19:35 +020084 private:
85 static constexpr int kDelayMs = 50;
86 static constexpr int kDecodeTime = kDelayMs / 2;
87 mutable uint32_t last_timestamp_ = 0;
88 mutable int64_t last_ms_ = -1;
89};
90
philipele7c891f2018-02-22 14:35:06 +010091class FrameObjectFake : public EncodedFrame {
philipelbe7a9e52016-05-19 12:19:35 +020092 public:
philipelb4d31082016-07-11 08:46:29 -070093 int64_t ReceivedTime() const override { return 0; }
94
95 int64_t RenderTime() const override { return _renderTimeMs; }
Niels Möller7cca0422019-04-29 16:12:19 +020096
97 bool delayed_by_retransmission() const override {
98 return delayed_by_retransmission_;
99 }
100 void set_delayed_by_retransmission(bool delayed) {
101 delayed_by_retransmission_ = delayed;
102 }
103
104 private:
105 bool delayed_by_retransmission_ = false;
philipela45102f2017-02-22 05:30:39 -0800106};
107
108class VCMReceiveStatisticsCallbackMock : public VCMReceiveStatisticsCallback {
109 public:
ilnik6d5b4d62017-08-30 03:32:14 -0700110 MOCK_METHOD3(OnCompleteFrame,
111 void(bool is_keyframe,
112 size_t size_bytes,
113 VideoContentType content_type));
Johannes Kron0c141c52019-08-26 15:04:43 +0200114 MOCK_METHOD1(OnDroppedFrames, void(uint32_t frames_dropped));
philipela45102f2017-02-22 05:30:39 -0800115 MOCK_METHOD1(OnDiscardedPacketsUpdated, void(int discarded_packets));
116 MOCK_METHOD1(OnFrameCountsUpdated, void(const FrameCounts& frame_counts));
Johannes Kronbfd343b2019-07-01 10:07:50 +0200117 MOCK_METHOD6(OnFrameBufferTimingsUpdated,
118 void(int max_decode_ms,
philipela45102f2017-02-22 05:30:39 -0800119 int current_delay_ms,
120 int target_delay_ms,
121 int jitter_buffer_ms,
122 int min_playout_delay_ms,
123 int render_delay_ms));
ilnik2edc6842017-07-06 03:06:50 -0700124 MOCK_METHOD1(OnTimingFrameInfoUpdated, void(const TimingFrameInfo& info));
philipelbe7a9e52016-05-19 12:19:35 +0200125};
126
127class TestFrameBuffer2 : public ::testing::Test {
128 protected:
129 static constexpr int kMaxReferences = 5;
130 static constexpr int kFps1 = 1000;
131 static constexpr int kFps10 = kFps1 / 10;
132 static constexpr int kFps20 = kFps1 / 20;
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100133 static constexpr size_t kFrameSize = 10;
philipelbe7a9e52016-05-19 12:19:35 +0200134
135 TestFrameBuffer2()
Niels Möller7cca0422019-04-29 16:12:19 +0200136 : trial_("WebRTC-AddRttToPlayoutDelay/Enabled/"),
137 clock_(0),
philipelbe7a9e52016-05-19 12:19:35 +0200138 timing_(&clock_),
Jonas Olssona4d87372019-07-05 19:08:33 +0200139 buffer_(new FrameBuffer(&clock_, &timing_, &stats_callback_)),
philipelbe7a9e52016-05-19 12:19:35 +0200140 rand_(0x34678213),
141 tear_down_(false),
Niels Möllerc572ff32018-11-07 08:43:50 +0100142 extract_thread_(&ExtractLoop, this, "Extract Thread") {}
philipelbe7a9e52016-05-19 12:19:35 +0200143
144 void SetUp() override { extract_thread_.Start(); }
145
146 void TearDown() override {
147 tear_down_ = true;
148 trigger_extract_event_.Set();
149 extract_thread_.Stop();
150 }
151
152 template <typename... T>
Niels Möller7cca0422019-04-29 16:12:19 +0200153 std::unique_ptr<FrameObjectFake> CreateFrame(uint16_t picture_id,
154 uint8_t spatial_layer,
155 int64_t ts_ms,
156 bool inter_layer_predicted,
157 bool last_spatial_layer,
Sergey Silkin2799e632019-05-17 09:51:39 +0200158 size_t frame_size_bytes,
Niels Möller7cca0422019-04-29 16:12:19 +0200159 T... refs) {
philipelbe7a9e52016-05-19 12:19:35 +0200160 static_assert(sizeof...(refs) <= kMaxReferences,
philipele7c891f2018-02-22 14:35:06 +0100161 "To many references specified for EncodedFrame.");
kwiberg5b9746e2017-08-16 04:52:35 -0700162 std::array<uint16_t, sizeof...(refs)> references = {
163 {rtc::checked_cast<uint16_t>(refs)...}};
philipelbe7a9e52016-05-19 12:19:35 +0200164
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200165 auto frame = std::make_unique<FrameObjectFake>();
philipel0fa82a62018-03-19 15:34:53 +0100166 frame->id.picture_id = picture_id;
167 frame->id.spatial_layer = spatial_layer;
Sergey Silkin61832dd2018-12-20 14:32:14 +0100168 frame->SetSpatialIndex(spatial_layer);
Niels Möller23775882018-08-16 10:24:12 +0200169 frame->SetTimestamp(ts_ms * 90);
philipelbe7a9e52016-05-19 12:19:35 +0200170 frame->num_references = references.size();
171 frame->inter_layer_predicted = inter_layer_predicted;
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100172 frame->is_last_spatial_layer = last_spatial_layer;
173 // Add some data to buffer.
Niels Möllerb9bfe652019-10-03 08:43:53 +0200174 frame->SetEncodedData(EncodedImageBuffer::Create(frame_size_bytes));
philipelbe7a9e52016-05-19 12:19:35 +0200175 for (size_t r = 0; r < references.size(); ++r)
176 frame->references[r] = references[r];
Niels Möller7cca0422019-04-29 16:12:19 +0200177 return frame;
178 }
philipelbe7a9e52016-05-19 12:19:35 +0200179
Niels Möller7cca0422019-04-29 16:12:19 +0200180 template <typename... T>
181 int InsertFrame(uint16_t picture_id,
182 uint8_t spatial_layer,
183 int64_t ts_ms,
184 bool inter_layer_predicted,
185 bool last_spatial_layer,
Sergey Silkin2799e632019-05-17 09:51:39 +0200186 size_t frame_size_bytes,
Niels Möller7cca0422019-04-29 16:12:19 +0200187 T... refs) {
Sergey Silkin2799e632019-05-17 09:51:39 +0200188 return buffer_->InsertFrame(
189 CreateFrame(picture_id, spatial_layer, ts_ms, inter_layer_predicted,
190 last_spatial_layer, frame_size_bytes, refs...));
Niels Möller7cca0422019-04-29 16:12:19 +0200191 }
192
193 int InsertNackedFrame(uint16_t picture_id, int64_t ts_ms) {
194 std::unique_ptr<FrameObjectFake> frame =
Sergey Silkin2799e632019-05-17 09:51:39 +0200195 CreateFrame(picture_id, 0, ts_ms, false, true, kFrameSize);
Niels Möller7cca0422019-04-29 16:12:19 +0200196 frame->set_delayed_by_retransmission(true);
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200197 return buffer_->InsertFrame(std::move(frame));
philipelbe7a9e52016-05-19 12:19:35 +0200198 }
199
philipel3042c2d2017-08-18 04:55:02 -0700200 void ExtractFrame(int64_t max_wait_time = 0, bool keyframe_required = false) {
philipelbe7a9e52016-05-19 12:19:35 +0200201 crit_.Enter();
202 if (max_wait_time == 0) {
philipele7c891f2018-02-22 14:35:06 +0100203 std::unique_ptr<EncodedFrame> frame;
philipel3042c2d2017-08-18 04:55:02 -0700204 FrameBuffer::ReturnReason res =
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200205 buffer_->NextFrame(0, &frame, keyframe_required);
philipel75562822016-09-05 10:57:41 +0200206 if (res != FrameBuffer::ReturnReason::kStopped)
207 frames_.emplace_back(std::move(frame));
philipelbe7a9e52016-05-19 12:19:35 +0200208 crit_.Leave();
209 } else {
210 max_wait_time_ = max_wait_time;
211 trigger_extract_event_.Set();
212 crit_.Leave();
213 // Make sure |crit_| is aquired by |extract_thread_| before returning.
214 crit_acquired_event_.Wait(rtc::Event::kForever);
215 }
216 }
217
218 void CheckFrame(size_t index, int picture_id, int spatial_layer) {
219 rtc::CritScope lock(&crit_);
220 ASSERT_LT(index, frames_.size());
221 ASSERT_TRUE(frames_[index]);
philipel0fa82a62018-03-19 15:34:53 +0100222 ASSERT_EQ(picture_id, frames_[index]->id.picture_id);
223 ASSERT_EQ(spatial_layer, frames_[index]->id.spatial_layer);
philipelbe7a9e52016-05-19 12:19:35 +0200224 }
225
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100226 void CheckFrameSize(size_t index, size_t size) {
227 rtc::CritScope lock(&crit_);
228 ASSERT_LT(index, frames_.size());
229 ASSERT_TRUE(frames_[index]);
230 ASSERT_EQ(frames_[index]->size(), size);
231 }
232
philipelbe7a9e52016-05-19 12:19:35 +0200233 void CheckNoFrame(size_t index) {
234 rtc::CritScope lock(&crit_);
235 ASSERT_LT(index, frames_.size());
236 ASSERT_FALSE(frames_[index]);
237 }
238
tommi0f8b4032017-02-22 11:22:05 -0800239 static void ExtractLoop(void* obj) {
philipelbe7a9e52016-05-19 12:19:35 +0200240 TestFrameBuffer2* tfb = static_cast<TestFrameBuffer2*>(obj);
241 while (true) {
242 tfb->trigger_extract_event_.Wait(rtc::Event::kForever);
243 {
244 rtc::CritScope lock(&tfb->crit_);
245 tfb->crit_acquired_event_.Set();
246 if (tfb->tear_down_)
tommi0f8b4032017-02-22 11:22:05 -0800247 return;
philipelbe7a9e52016-05-19 12:19:35 +0200248
philipele7c891f2018-02-22 14:35:06 +0100249 std::unique_ptr<EncodedFrame> frame;
philipel75562822016-09-05 10:57:41 +0200250 FrameBuffer::ReturnReason res =
philipelcd936fd2019-05-02 13:53:10 +0200251 tfb->buffer_->NextFrame(tfb->max_wait_time_, &frame, false);
philipel75562822016-09-05 10:57:41 +0200252 if (res != FrameBuffer::ReturnReason::kStopped)
253 tfb->frames_.emplace_back(std::move(frame));
philipelbe7a9e52016-05-19 12:19:35 +0200254 }
255 }
256 }
257
258 uint32_t Rand() { return rand_.Rand<uint32_t>(); }
259
Niels Möller7cca0422019-04-29 16:12:19 +0200260 // The ProtectionMode tests depends on rtt-multiplier experiment.
261 test::ScopedFieldTrials trial_;
philipelbe7a9e52016-05-19 12:19:35 +0200262 SimulatedClock clock_;
263 VCMTimingFake timing_;
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200264 std::unique_ptr<FrameBuffer> buffer_;
philipele7c891f2018-02-22 14:35:06 +0100265 std::vector<std::unique_ptr<EncodedFrame>> frames_;
philipelbe7a9e52016-05-19 12:19:35 +0200266 Random rand_;
philipela45102f2017-02-22 05:30:39 -0800267 ::testing::NiceMock<VCMReceiveStatisticsCallbackMock> stats_callback_;
philipelbe7a9e52016-05-19 12:19:35 +0200268
269 int64_t max_wait_time_;
270 bool tear_down_;
271 rtc::PlatformThread extract_thread_;
272 rtc::Event trigger_extract_event_;
273 rtc::Event crit_acquired_event_;
274 rtc::CriticalSection crit_;
275};
276
Sergey Silkin2799e632019-05-17 09:51:39 +0200277// From https://en.cppreference.com/w/cpp/language/static: "If ... a constexpr
278// static data member (since C++11) is odr-used, a definition at namespace scope
279// is still required... This definition is deprecated for constexpr data members
280// since C++17."
281// kFrameSize is odr-used since it is passed by reference to EXPECT_EQ().
282#if __cplusplus < 201703L
283constexpr size_t TestFrameBuffer2::kFrameSize;
284#endif
285
philipel6e8224f2016-05-19 17:07:44 +0200286// Following tests are timing dependent. Either the timeouts have to
287// be increased by a large margin, which would slow down all trybots,
288// or we disable them for the very slow ones, like we do here.
289#if !defined(ADDRESS_SANITIZER) && !defined(MEMORY_SANITIZER)
philipelbe7a9e52016-05-19 12:19:35 +0200290TEST_F(TestFrameBuffer2, WaitForFrame) {
291 uint16_t pid = Rand();
292 uint32_t ts = Rand();
293
philipel6e8224f2016-05-19 17:07:44 +0200294 ExtractFrame(50);
Sergey Silkin2799e632019-05-17 09:51:39 +0200295 InsertFrame(pid, 0, ts, false, true, kFrameSize);
philipelbe7a9e52016-05-19 12:19:35 +0200296 CheckFrame(0, pid, 0);
297}
298
299TEST_F(TestFrameBuffer2, OneSuperFrame) {
300 uint16_t pid = Rand();
301 uint32_t ts = Rand();
302
Sergey Silkin2799e632019-05-17 09:51:39 +0200303 InsertFrame(pid, 0, ts, false, false, kFrameSize);
304 InsertFrame(pid, 1, ts, true, true, kFrameSize);
philipele0b2f152016-09-28 10:23:49 +0200305 ExtractFrame();
306
Sergey Silkin61832dd2018-12-20 14:32:14 +0100307 CheckFrame(0, pid, 1);
philipele0b2f152016-09-28 10:23:49 +0200308}
309
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200310TEST_F(TestFrameBuffer2, ZeroPlayoutDelay) {
311 VCMTiming timing(&clock_);
Niels Möllerd9c2d942019-04-30 09:16:36 +0200312 buffer_.reset(new FrameBuffer(&clock_, &timing, &stats_callback_));
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200313 const PlayoutDelay kPlayoutDelayMs = {0, 0};
314 std::unique_ptr<FrameObjectFake> test_frame(new FrameObjectFake());
315 test_frame->id.picture_id = 0;
316 test_frame->SetPlayoutDelay(kPlayoutDelayMs);
317 buffer_->InsertFrame(std::move(test_frame));
318 ExtractFrame(0, false);
319 CheckFrame(0, 0, 0);
320 EXPECT_EQ(0, frames_[0]->RenderTimeMs());
321}
322
aleloi3e005282017-01-26 05:38:00 -0800323// Flaky test, see bugs.webrtc.org/7068.
324TEST_F(TestFrameBuffer2, DISABLED_OneUnorderedSuperFrame) {
philipele0b2f152016-09-28 10:23:49 +0200325 uint16_t pid = Rand();
326 uint32_t ts = Rand();
327
philipel6e8224f2016-05-19 17:07:44 +0200328 ExtractFrame(50);
Sergey Silkin2799e632019-05-17 09:51:39 +0200329 InsertFrame(pid, 1, ts, true, true, kFrameSize);
330 InsertFrame(pid, 0, ts, false, false, kFrameSize);
philipelbe7a9e52016-05-19 12:19:35 +0200331 ExtractFrame();
332
333 CheckFrame(0, pid, 0);
334 CheckFrame(1, pid, 1);
335}
336
philipel94616b32016-05-20 10:43:01 +0200337TEST_F(TestFrameBuffer2, DISABLED_OneLayerStreamReordered) {
philipel6e8224f2016-05-19 17:07:44 +0200338 uint16_t pid = Rand();
339 uint32_t ts = Rand();
340
Sergey Silkin2799e632019-05-17 09:51:39 +0200341 InsertFrame(pid, 0, ts, false, true, kFrameSize);
philipel6e8224f2016-05-19 17:07:44 +0200342 ExtractFrame();
343 CheckFrame(0, pid, 0);
344 for (int i = 1; i < 10; i += 2) {
345 ExtractFrame(50);
Sergey Silkin2799e632019-05-17 09:51:39 +0200346 InsertFrame(pid + i + 1, 0, ts + (i + 1) * kFps10, false, true, kFrameSize,
347 pid + i);
philipel6e8224f2016-05-19 17:07:44 +0200348 clock_.AdvanceTimeMilliseconds(kFps10);
Sergey Silkin2799e632019-05-17 09:51:39 +0200349 InsertFrame(pid + i, 0, ts + i * kFps10, false, true, kFrameSize,
350 pid + i - 1);
philipel6e8224f2016-05-19 17:07:44 +0200351 clock_.AdvanceTimeMilliseconds(kFps10);
352 ExtractFrame();
353 CheckFrame(i, pid + i, 0);
354 CheckFrame(i + 1, pid + i + 1, 0);
355 }
356}
357#endif // Timing dependent tests.
358
359TEST_F(TestFrameBuffer2, ExtractFromEmptyBuffer) {
360 ExtractFrame();
361 CheckNoFrame(0);
362}
363
philipel93e451b2016-10-06 12:25:13 +0200364TEST_F(TestFrameBuffer2, MissingFrame) {
365 uint16_t pid = Rand();
366 uint32_t ts = Rand();
367
Sergey Silkin2799e632019-05-17 09:51:39 +0200368 InsertFrame(pid, 0, ts, false, true, kFrameSize);
369 InsertFrame(pid + 2, 0, ts, false, true, kFrameSize, pid);
370 InsertFrame(pid + 3, 0, ts, false, true, kFrameSize, pid + 1, pid + 2);
philipel93e451b2016-10-06 12:25:13 +0200371 ExtractFrame();
372 ExtractFrame();
373 ExtractFrame();
374
375 CheckFrame(0, pid, 0);
376 CheckFrame(1, pid + 2, 0);
377 CheckNoFrame(2);
378}
379
philipelbe7a9e52016-05-19 12:19:35 +0200380TEST_F(TestFrameBuffer2, OneLayerStream) {
381 uint16_t pid = Rand();
382 uint32_t ts = Rand();
383
Sergey Silkin2799e632019-05-17 09:51:39 +0200384 InsertFrame(pid, 0, ts, false, true, kFrameSize);
philipelbe7a9e52016-05-19 12:19:35 +0200385 ExtractFrame();
386 CheckFrame(0, pid, 0);
387 for (int i = 1; i < 10; ++i) {
Sergey Silkin2799e632019-05-17 09:51:39 +0200388 InsertFrame(pid + i, 0, ts + i * kFps10, false, true, kFrameSize,
389 pid + i - 1);
philipelbe7a9e52016-05-19 12:19:35 +0200390 ExtractFrame();
391 clock_.AdvanceTimeMilliseconds(kFps10);
392 CheckFrame(i, pid + i, 0);
393 }
394}
395
philipelbe7a9e52016-05-19 12:19:35 +0200396TEST_F(TestFrameBuffer2, DropTemporalLayerSlowDecoder) {
397 uint16_t pid = Rand();
398 uint32_t ts = Rand();
399
Sergey Silkin2799e632019-05-17 09:51:39 +0200400 InsertFrame(pid, 0, ts, false, true, kFrameSize);
401 InsertFrame(pid + 1, 0, ts + kFps20, false, true, kFrameSize, pid);
philipelbe7a9e52016-05-19 12:19:35 +0200402 for (int i = 2; i < 10; i += 2) {
403 uint32_t ts_tl0 = ts + i / 2 * kFps10;
Sergey Silkin2799e632019-05-17 09:51:39 +0200404 InsertFrame(pid + i, 0, ts_tl0, false, true, kFrameSize, pid + i - 2);
405 InsertFrame(pid + i + 1, 0, ts_tl0 + kFps20, false, true, kFrameSize,
406 pid + i, pid + i - 1);
philipelbe7a9e52016-05-19 12:19:35 +0200407 }
408
Johannes Kron0c141c52019-08-26 15:04:43 +0200409 EXPECT_CALL(stats_callback_, OnDroppedFrames(1)).Times(3);
410
philipelbe7a9e52016-05-19 12:19:35 +0200411 for (int i = 0; i < 10; ++i) {
412 ExtractFrame();
Ilya Nikolaevskiy8c4fe162018-02-27 15:49:47 +0100413 clock_.AdvanceTimeMilliseconds(70);
philipelbe7a9e52016-05-19 12:19:35 +0200414 }
415
416 CheckFrame(0, pid, 0);
417 CheckFrame(1, pid + 1, 0);
418 CheckFrame(2, pid + 2, 0);
419 CheckFrame(3, pid + 4, 0);
420 CheckFrame(4, pid + 6, 0);
421 CheckFrame(5, pid + 8, 0);
422 CheckNoFrame(6);
423 CheckNoFrame(7);
424 CheckNoFrame(8);
425 CheckNoFrame(9);
426}
427
Johannes Kron0c141c52019-08-26 15:04:43 +0200428TEST_F(TestFrameBuffer2, DropFramesIfSystemIsStalled) {
429 uint16_t pid = Rand();
430 uint32_t ts = Rand();
431
432 InsertFrame(pid, 0, ts, false, true, kFrameSize);
433 InsertFrame(pid + 1, 0, ts + 1 * kFps10, false, true, kFrameSize, pid);
434 InsertFrame(pid + 2, 0, ts + 2 * kFps10, false, true, kFrameSize, pid + 1);
435 InsertFrame(pid + 3, 0, ts + 3 * kFps10, false, true, kFrameSize);
436
437 ExtractFrame();
438 // Jump forward in time, simulating the system being stalled for some reason.
439 clock_.AdvanceTimeMilliseconds(3 * kFps10);
440 // Extract one more frame, expect second and third frame to be dropped.
441 EXPECT_CALL(stats_callback_, OnDroppedFrames(2)).Times(1);
442 ExtractFrame();
443
444 CheckFrame(0, pid + 0, 0);
445 CheckFrame(1, pid + 3, 0);
446}
447
448TEST_F(TestFrameBuffer2, DroppedFramesCountedOnClear) {
449 uint16_t pid = Rand();
450 uint32_t ts = Rand();
451
452 InsertFrame(pid, 0, ts, false, true, kFrameSize);
453 for (int i = 1; i < 5; ++i) {
454 InsertFrame(pid + i, 0, ts + i * kFps10, false, true, kFrameSize,
455 pid + i - 1);
456 }
457
458 // All frames should be dropped when Clear is called.
459 EXPECT_CALL(stats_callback_, OnDroppedFrames(5)).Times(1);
460 buffer_->Clear();
461}
462
philipelbe7a9e52016-05-19 12:19:35 +0200463TEST_F(TestFrameBuffer2, InsertLateFrame) {
464 uint16_t pid = Rand();
465 uint32_t ts = Rand();
466
Sergey Silkin2799e632019-05-17 09:51:39 +0200467 InsertFrame(pid, 0, ts, false, true, kFrameSize);
philipelbe7a9e52016-05-19 12:19:35 +0200468 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200469 InsertFrame(pid + 2, 0, ts, false, true, kFrameSize);
philipelbe7a9e52016-05-19 12:19:35 +0200470 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200471 InsertFrame(pid + 1, 0, ts, false, true, kFrameSize, pid);
philipelbe7a9e52016-05-19 12:19:35 +0200472 ExtractFrame();
473
474 CheckFrame(0, pid, 0);
475 CheckFrame(1, pid + 2, 0);
476 CheckNoFrame(2);
477}
478
Niels Möller7cca0422019-04-29 16:12:19 +0200479TEST_F(TestFrameBuffer2, ProtectionModeNackFEC) {
philipel4f6cd6a2016-08-03 10:59:32 +0200480 uint16_t pid = Rand();
481 uint32_t ts = Rand();
Niels Möller7cca0422019-04-29 16:12:19 +0200482 constexpr int64_t kRttMs = 200;
483 buffer_->UpdateRtt(kRttMs);
philipel4f6cd6a2016-08-03 10:59:32 +0200484
Niels Möller7cca0422019-04-29 16:12:19 +0200485 // Jitter estimate unaffected by RTT in this protection mode.
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200486 buffer_->SetProtectionMode(kProtectionNackFEC);
Niels Möller7cca0422019-04-29 16:12:19 +0200487 InsertNackedFrame(pid, ts);
488 InsertNackedFrame(pid + 1, ts + 100);
489 InsertNackedFrame(pid + 2, ts + 200);
Sergey Silkin2799e632019-05-17 09:51:39 +0200490 InsertFrame(pid + 3, 0, ts + 300, false, true, kFrameSize);
philipel4f6cd6a2016-08-03 10:59:32 +0200491 ExtractFrame();
Niels Möller7cca0422019-04-29 16:12:19 +0200492 ExtractFrame();
493 ExtractFrame();
494 ExtractFrame();
495 ASSERT_EQ(4u, frames_.size());
496 EXPECT_LT(timing_.GetCurrentJitter(), kRttMs);
497}
498
499TEST_F(TestFrameBuffer2, ProtectionModeNack) {
500 uint16_t pid = Rand();
501 uint32_t ts = Rand();
502 constexpr int64_t kRttMs = 200;
503
504 buffer_->UpdateRtt(kRttMs);
505
506 // Jitter estimate includes RTT (after 3 retransmitted packets)
507 buffer_->SetProtectionMode(kProtectionNack);
508 InsertNackedFrame(pid, ts);
509 InsertNackedFrame(pid + 1, ts + 100);
510 InsertNackedFrame(pid + 2, ts + 200);
Sergey Silkin2799e632019-05-17 09:51:39 +0200511 InsertFrame(pid + 3, 0, ts + 300, false, true, kFrameSize);
Niels Möller7cca0422019-04-29 16:12:19 +0200512 ExtractFrame();
513 ExtractFrame();
514 ExtractFrame();
515 ExtractFrame();
516 ASSERT_EQ(4u, frames_.size());
517
518 EXPECT_GT(timing_.GetCurrentJitter(), kRttMs);
philipel4f6cd6a2016-08-03 10:59:32 +0200519}
520
philipele0b2f152016-09-28 10:23:49 +0200521TEST_F(TestFrameBuffer2, NoContinuousFrame) {
522 uint16_t pid = Rand();
523 uint32_t ts = Rand();
524
Sergey Silkin2799e632019-05-17 09:51:39 +0200525 EXPECT_EQ(-1, InsertFrame(pid + 1, 0, ts, false, true, kFrameSize, pid));
philipele0b2f152016-09-28 10:23:49 +0200526}
527
528TEST_F(TestFrameBuffer2, LastContinuousFrameSingleLayer) {
529 uint16_t pid = Rand();
530 uint32_t ts = Rand();
531
Sergey Silkin2799e632019-05-17 09:51:39 +0200532 EXPECT_EQ(pid, InsertFrame(pid, 0, ts, false, true, kFrameSize));
533 EXPECT_EQ(pid, InsertFrame(pid + 2, 0, ts, false, true, kFrameSize, pid + 1));
534 EXPECT_EQ(pid + 2, InsertFrame(pid + 1, 0, ts, false, true, kFrameSize, pid));
535 EXPECT_EQ(pid + 2,
536 InsertFrame(pid + 4, 0, ts, false, true, kFrameSize, pid + 3));
537 EXPECT_EQ(pid + 5, InsertFrame(pid + 5, 0, ts, false, true, kFrameSize));
philipele0b2f152016-09-28 10:23:49 +0200538}
539
540TEST_F(TestFrameBuffer2, LastContinuousFrameTwoLayers) {
541 uint16_t pid = Rand();
542 uint32_t ts = Rand();
543
Sergey Silkin2799e632019-05-17 09:51:39 +0200544 EXPECT_EQ(pid, InsertFrame(pid, 0, ts, false, false, kFrameSize));
545 EXPECT_EQ(pid, InsertFrame(pid, 1, ts, true, true, kFrameSize));
546 EXPECT_EQ(pid, InsertFrame(pid + 1, 1, ts, true, true, kFrameSize, pid));
547 EXPECT_EQ(pid,
548 InsertFrame(pid + 2, 0, ts, false, false, kFrameSize, pid + 1));
549 EXPECT_EQ(pid, InsertFrame(pid + 2, 1, ts, true, true, kFrameSize, pid + 1));
550 EXPECT_EQ(pid,
551 InsertFrame(pid + 3, 0, ts, false, false, kFrameSize, pid + 2));
552 EXPECT_EQ(pid + 3,
553 InsertFrame(pid + 1, 0, ts, false, false, kFrameSize, pid));
554 EXPECT_EQ(pid + 3,
555 InsertFrame(pid + 3, 1, ts, true, true, kFrameSize, pid + 2));
philipele0b2f152016-09-28 10:23:49 +0200556}
557
philipelfcc60062017-01-18 05:35:20 -0800558TEST_F(TestFrameBuffer2, PictureIdJumpBack) {
559 uint16_t pid = Rand();
560 uint32_t ts = Rand();
561
Sergey Silkin2799e632019-05-17 09:51:39 +0200562 EXPECT_EQ(pid, InsertFrame(pid, 0, ts, false, true, kFrameSize));
563 EXPECT_EQ(pid + 1,
564 InsertFrame(pid + 1, 0, ts + 1, false, true, kFrameSize, pid));
philipelfcc60062017-01-18 05:35:20 -0800565 ExtractFrame();
566 CheckFrame(0, pid, 0);
567
568 // Jump back in pid but increase ts.
Sergey Silkin2799e632019-05-17 09:51:39 +0200569 EXPECT_EQ(pid - 1, InsertFrame(pid - 1, 0, ts + 2, false, true, kFrameSize));
philipelfcc60062017-01-18 05:35:20 -0800570 ExtractFrame();
571 ExtractFrame();
572 CheckFrame(1, pid - 1, 0);
573 CheckNoFrame(2);
574}
575
philipela45102f2017-02-22 05:30:39 -0800576TEST_F(TestFrameBuffer2, StatsCallback) {
577 uint16_t pid = Rand();
578 uint32_t ts = Rand();
579 const int kFrameSize = 5000;
580
ilnik6d5b4d62017-08-30 03:32:14 -0700581 EXPECT_CALL(stats_callback_,
582 OnCompleteFrame(true, kFrameSize, VideoContentType::UNSPECIFIED));
Johannes Kronbfd343b2019-07-01 10:07:50 +0200583 EXPECT_CALL(stats_callback_, OnFrameBufferTimingsUpdated(_, _, _, _, _, _));
philipela45102f2017-02-22 05:30:39 -0800584
585 {
586 std::unique_ptr<FrameObjectFake> frame(new FrameObjectFake());
Niels Möllerb9bfe652019-10-03 08:43:53 +0200587 frame->SetEncodedData(EncodedImageBuffer::Create(kFrameSize));
philipel0fa82a62018-03-19 15:34:53 +0100588 frame->id.picture_id = pid;
589 frame->id.spatial_layer = 0;
Niels Möller23775882018-08-16 10:24:12 +0200590 frame->SetTimestamp(ts);
philipela45102f2017-02-22 05:30:39 -0800591 frame->num_references = 0;
592 frame->inter_layer_predicted = false;
593
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200594 EXPECT_EQ(buffer_->InsertFrame(std::move(frame)), pid);
philipela45102f2017-02-22 05:30:39 -0800595 }
596
597 ExtractFrame();
598 CheckFrame(0, pid, 0);
599}
600
philipel146a48b2017-04-20 04:04:38 -0700601TEST_F(TestFrameBuffer2, ForwardJumps) {
Sergey Silkin2799e632019-05-17 09:51:39 +0200602 EXPECT_EQ(5453, InsertFrame(5453, 0, 1, false, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700603 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200604 EXPECT_EQ(5454, InsertFrame(5454, 0, 1, false, true, kFrameSize, 5453));
philipel146a48b2017-04-20 04:04:38 -0700605 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200606 EXPECT_EQ(15670, InsertFrame(15670, 0, 1, false, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700607 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200608 EXPECT_EQ(29804, InsertFrame(29804, 0, 1, false, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700609 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200610 EXPECT_EQ(29805, InsertFrame(29805, 0, 1, false, true, kFrameSize, 29804));
philipel146a48b2017-04-20 04:04:38 -0700611 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200612 EXPECT_EQ(29806, InsertFrame(29806, 0, 1, false, true, kFrameSize, 29805));
philipel146a48b2017-04-20 04:04:38 -0700613 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200614 EXPECT_EQ(33819, InsertFrame(33819, 0, 1, false, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700615 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200616 EXPECT_EQ(41248, InsertFrame(41248, 0, 1, false, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700617 ExtractFrame();
618}
619
philipelf6842692017-04-28 03:29:15 -0700620TEST_F(TestFrameBuffer2, DuplicateFrames) {
Sergey Silkin2799e632019-05-17 09:51:39 +0200621 EXPECT_EQ(22256, InsertFrame(22256, 0, 1, false, true, kFrameSize));
philipelf6842692017-04-28 03:29:15 -0700622 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200623 EXPECT_EQ(22256, InsertFrame(22256, 0, 1, false, true, kFrameSize));
philipelf6842692017-04-28 03:29:15 -0700624}
625
philipel112adf92017-06-15 09:06:21 -0700626// TODO(philipel): implement more unittests related to invalid references.
627TEST_F(TestFrameBuffer2, InvalidReferences) {
Sergey Silkin2799e632019-05-17 09:51:39 +0200628 EXPECT_EQ(-1, InsertFrame(0, 0, 1000, false, true, kFrameSize, 2));
629 EXPECT_EQ(1, InsertFrame(1, 0, 2000, false, true, kFrameSize));
philipel112adf92017-06-15 09:06:21 -0700630 ExtractFrame();
Sergey Silkin2799e632019-05-17 09:51:39 +0200631 EXPECT_EQ(2, InsertFrame(2, 0, 3000, false, true, kFrameSize, 1));
philipel112adf92017-06-15 09:06:21 -0700632}
633
philipel3042c2d2017-08-18 04:55:02 -0700634TEST_F(TestFrameBuffer2, KeyframeRequired) {
Sergey Silkin2799e632019-05-17 09:51:39 +0200635 EXPECT_EQ(1, InsertFrame(1, 0, 1000, false, true, kFrameSize));
636 EXPECT_EQ(2, InsertFrame(2, 0, 2000, false, true, kFrameSize, 1));
637 EXPECT_EQ(3, InsertFrame(3, 0, 3000, false, true, kFrameSize));
philipel3042c2d2017-08-18 04:55:02 -0700638 ExtractFrame();
639 ExtractFrame(0, true);
640 ExtractFrame();
641
642 CheckFrame(0, 1, 0);
643 CheckFrame(1, 3, 0);
644 CheckNoFrame(2);
645}
646
philipel9771c502018-03-02 11:06:27 +0100647TEST_F(TestFrameBuffer2, KeyframeClearsFullBuffer) {
648 const int kMaxBufferSize = 600;
649
650 for (int i = 1; i <= kMaxBufferSize; ++i)
Sergey Silkin2799e632019-05-17 09:51:39 +0200651 EXPECT_EQ(-1, InsertFrame(i, 0, i * 1000, false, true, kFrameSize, i - 1));
philipel9771c502018-03-02 11:06:27 +0100652 ExtractFrame();
653 CheckNoFrame(0);
654
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100655 EXPECT_EQ(kMaxBufferSize + 1,
656 InsertFrame(kMaxBufferSize + 1, 0, (kMaxBufferSize + 1) * 1000,
Sergey Silkin2799e632019-05-17 09:51:39 +0200657 false, true, kFrameSize));
philipel9771c502018-03-02 11:06:27 +0100658 ExtractFrame();
659 CheckFrame(1, kMaxBufferSize + 1, 0);
660}
661
philipel798b2822018-06-11 13:10:14 +0200662TEST_F(TestFrameBuffer2, DontUpdateOnUndecodableFrame) {
Sergey Silkin2799e632019-05-17 09:51:39 +0200663 InsertFrame(1, 0, 0, false, true, kFrameSize);
philipel798b2822018-06-11 13:10:14 +0200664 ExtractFrame(0, true);
Sergey Silkin2799e632019-05-17 09:51:39 +0200665 InsertFrame(3, 0, 0, false, true, kFrameSize, 2, 0);
666 InsertFrame(3, 0, 0, false, true, kFrameSize, 0);
667 InsertFrame(2, 0, 0, false, true, kFrameSize);
philipel798b2822018-06-11 13:10:14 +0200668 ExtractFrame(0, true);
669 ExtractFrame(0, true);
670}
671
philipel6d216502018-10-22 14:36:45 +0200672TEST_F(TestFrameBuffer2, DontDecodeOlderTimestamp) {
Sergey Silkin2799e632019-05-17 09:51:39 +0200673 InsertFrame(2, 0, 1, false, true, kFrameSize);
674 InsertFrame(1, 0, 2, false, true,
675 kFrameSize); // Older picture id but newer timestamp.
philipel6d216502018-10-22 14:36:45 +0200676 ExtractFrame(0);
677 ExtractFrame(0);
678 CheckFrame(0, 1, 0);
679 CheckNoFrame(1);
680
Sergey Silkin2799e632019-05-17 09:51:39 +0200681 InsertFrame(3, 0, 4, false, true, kFrameSize);
682 InsertFrame(4, 0, 3, false, true,
683 kFrameSize); // Newer picture id but older timestamp.
philipel6d216502018-10-22 14:36:45 +0200684 ExtractFrame(0);
685 ExtractFrame(0);
686 CheckFrame(2, 3, 0);
687 CheckNoFrame(3);
688}
689
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100690TEST_F(TestFrameBuffer2, CombineFramesToSuperframe) {
691 uint16_t pid = Rand();
692 uint32_t ts = Rand();
693
Sergey Silkin2799e632019-05-17 09:51:39 +0200694 InsertFrame(pid, 0, ts, false, false, kFrameSize);
695 InsertFrame(pid, 1, ts, true, true, 2 * kFrameSize);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100696 ExtractFrame(0);
697 ExtractFrame(0);
Sergey Silkin61832dd2018-12-20 14:32:14 +0100698 CheckFrame(0, pid, 1);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100699 CheckNoFrame(1);
700 // Two frames should be combined and returned together.
Sergey Silkin2799e632019-05-17 09:51:39 +0200701 CheckFrameSize(0, 3 * kFrameSize);
702
703 EXPECT_EQ(frames_[0]->SpatialIndex(), 1);
704 EXPECT_EQ(frames_[0]->SpatialLayerFrameSize(0), kFrameSize);
705 EXPECT_EQ(frames_[0]->SpatialLayerFrameSize(1), 2 * kFrameSize);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100706}
707
708TEST_F(TestFrameBuffer2, HigherSpatialLayerNonDecodable) {
709 uint16_t pid = Rand();
710 uint32_t ts = Rand();
711
Sergey Silkin2799e632019-05-17 09:51:39 +0200712 InsertFrame(pid, 0, ts, false, false, kFrameSize);
713 InsertFrame(pid, 1, ts, true, true, kFrameSize);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100714
715 ExtractFrame(0);
Sergey Silkin61832dd2018-12-20 14:32:14 +0100716 CheckFrame(0, pid, 1);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100717
Sergey Silkin2799e632019-05-17 09:51:39 +0200718 InsertFrame(pid + 1, 1, ts + kFps20, false, true, kFrameSize, pid);
719 InsertFrame(pid + 2, 0, ts + kFps10, false, false, kFrameSize, pid);
720 InsertFrame(pid + 2, 1, ts + kFps10, true, true, kFrameSize, pid + 1);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100721
722 clock_.AdvanceTimeMilliseconds(1000);
723 // Frame pid+1 is decodable but too late.
724 // In superframe pid+2 frame sid=0 is decodable, but frame sid=1 is not.
725 // Incorrect implementation might skip pid+1 frame and output undecodable
726 // pid+2 instead.
727 ExtractFrame();
728 ExtractFrame();
729 CheckFrame(1, pid + 1, 1);
Sergey Silkin61832dd2018-12-20 14:32:14 +0100730 CheckFrame(2, pid + 2, 1);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100731}
732
philipelbe7a9e52016-05-19 12:19:35 +0200733} // namespace video_coding
734} // namespace webrtc