blob: 129c25f647a694a3488fc177b856f0822e652936 [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"
philipel19053972020-01-29 17:36:11 +010029#include "test/time_controller/simulated_time_controller.h"
philipelbe7a9e52016-05-19 12:19:35 +020030
Mirko Bonadei6a489f22019-04-09 15:11:12 +020031using ::testing::_;
32using ::testing::Return;
philipela45102f2017-02-22 05:30:39 -080033
philipelbe7a9e52016-05-19 12:19:35 +020034namespace webrtc {
35namespace video_coding {
36
37class VCMTimingFake : public VCMTiming {
38 public:
39 explicit VCMTimingFake(Clock* clock) : VCMTiming(clock) {}
40
41 int64_t RenderTimeMs(uint32_t frame_timestamp,
42 int64_t now_ms) const override {
43 if (last_ms_ == -1) {
44 last_ms_ = now_ms + kDelayMs;
45 last_timestamp_ = frame_timestamp;
46 }
47
48 uint32_t diff = MinDiff(frame_timestamp, last_timestamp_);
49 if (AheadOf(frame_timestamp, last_timestamp_))
50 last_ms_ += diff / 90;
51 else
52 last_ms_ -= diff / 90;
53
54 last_timestamp_ = frame_timestamp;
55 return last_ms_;
56 }
57
Rezaul Barbhuiya82c22482021-08-05 17:54:11 -070058 int64_t MaxWaitingTime(int64_t render_time_ms,
Johannes Kron2ddc39e2021-08-10 16:56:12 +020059 int64_t now_ms,
60 bool too_many_frames_queued) const override {
Ilya Nikolaevskiy8c4fe162018-02-27 15:49:47 +010061 return render_time_ms - now_ms - kDecodeTime;
philipelbe7a9e52016-05-19 12:19:35 +020062 }
63
Johannes Kronbfd343b2019-07-01 10:07:50 +020064 bool GetTimings(int* max_decode_ms,
philipela45102f2017-02-22 05:30:39 -080065 int* current_delay_ms,
66 int* target_delay_ms,
67 int* jitter_buffer_ms,
68 int* min_playout_delay_ms,
69 int* render_delay_ms) const override {
70 return true;
71 }
72
Niels Möller7cca0422019-04-29 16:12:19 +020073 int GetCurrentJitter() {
Niels Möller7cca0422019-04-29 16:12:19 +020074 int max_decode_ms;
75 int current_delay_ms;
76 int target_delay_ms;
77 int jitter_buffer_ms;
78 int min_playout_delay_ms;
79 int render_delay_ms;
Johannes Kronbfd343b2019-07-01 10:07:50 +020080 VCMTiming::GetTimings(&max_decode_ms, &current_delay_ms, &target_delay_ms,
81 &jitter_buffer_ms, &min_playout_delay_ms,
82 &render_delay_ms);
Niels Möller7cca0422019-04-29 16:12:19 +020083 return jitter_buffer_ms;
84 }
85
philipelbe7a9e52016-05-19 12:19:35 +020086 private:
87 static constexpr int kDelayMs = 50;
88 static constexpr int kDecodeTime = kDelayMs / 2;
89 mutable uint32_t last_timestamp_ = 0;
90 mutable int64_t last_ms_ = -1;
91};
92
philipele7c891f2018-02-22 14:35:06 +010093class FrameObjectFake : public EncodedFrame {
philipelbe7a9e52016-05-19 12:19:35 +020094 public:
philipelb4d31082016-07-11 08:46:29 -070095 int64_t ReceivedTime() const override { return 0; }
96
97 int64_t RenderTime() const override { return _renderTimeMs; }
Niels Möller7cca0422019-04-29 16:12:19 +020098
99 bool delayed_by_retransmission() const override {
100 return delayed_by_retransmission_;
101 }
102 void set_delayed_by_retransmission(bool delayed) {
103 delayed_by_retransmission_ = delayed;
104 }
105
106 private:
107 bool delayed_by_retransmission_ = false;
philipela45102f2017-02-22 05:30:39 -0800108};
109
110class VCMReceiveStatisticsCallbackMock : public VCMReceiveStatisticsCallback {
111 public:
Danil Chapovalovf2c0f152020-05-19 13:20:27 +0200112 MOCK_METHOD(void,
113 OnCompleteFrame,
114 (bool is_keyframe,
115 size_t size_bytes,
116 VideoContentType content_type),
117 (override));
118 MOCK_METHOD(void, OnDroppedFrames, (uint32_t frames_dropped), (override));
119 MOCK_METHOD(void,
120 OnFrameBufferTimingsUpdated,
121 (int max_decode_ms,
122 int current_delay_ms,
123 int target_delay_ms,
124 int jitter_buffer_ms,
125 int min_playout_delay_ms,
126 int render_delay_ms),
127 (override));
128 MOCK_METHOD(void,
129 OnTimingFrameInfoUpdated,
130 (const TimingFrameInfo& info),
131 (override));
philipelbe7a9e52016-05-19 12:19:35 +0200132};
133
134class TestFrameBuffer2 : public ::testing::Test {
135 protected:
136 static constexpr int kMaxReferences = 5;
137 static constexpr int kFps1 = 1000;
138 static constexpr int kFps10 = kFps1 / 10;
139 static constexpr int kFps20 = kFps1 / 20;
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100140 static constexpr size_t kFrameSize = 10;
philipelbe7a9e52016-05-19 12:19:35 +0200141
142 TestFrameBuffer2()
Evan Shrubsole9146d762021-10-13 11:19:48 +0200143 : time_controller_(Timestamp::Seconds(0)),
philipel19053972020-01-29 17:36:11 +0100144 time_task_queue_(
145 time_controller_.GetTaskQueueFactory()->CreateTaskQueue(
146 "extract queue",
147 TaskQueueFactory::Priority::NORMAL)),
148 timing_(time_controller_.GetClock()),
149 buffer_(new FrameBuffer(time_controller_.GetClock(),
150 &timing_,
151 &stats_callback_)),
152 rand_(0x34678213) {}
philipelbe7a9e52016-05-19 12:19:35 +0200153
154 template <typename... T>
Niels Möller7cca0422019-04-29 16:12:19 +0200155 std::unique_ptr<FrameObjectFake> CreateFrame(uint16_t picture_id,
156 uint8_t spatial_layer,
157 int64_t ts_ms,
Niels Möller7cca0422019-04-29 16:12:19 +0200158 bool last_spatial_layer,
Sergey Silkin2799e632019-05-17 09:51:39 +0200159 size_t frame_size_bytes,
Niels Möller7cca0422019-04-29 16:12:19 +0200160 T... refs) {
philipelbe7a9e52016-05-19 12:19:35 +0200161 static_assert(sizeof...(refs) <= kMaxReferences,
philipele7c891f2018-02-22 14:35:06 +0100162 "To many references specified for EncodedFrame.");
kwiberg5b9746e2017-08-16 04:52:35 -0700163 std::array<uint16_t, sizeof...(refs)> references = {
164 {rtc::checked_cast<uint16_t>(refs)...}};
philipelbe7a9e52016-05-19 12:19:35 +0200165
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200166 auto frame = std::make_unique<FrameObjectFake>();
philipel9aa9b8d2021-02-15 13:31:29 +0100167 frame->SetId(picture_id);
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();
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100171 frame->is_last_spatial_layer = last_spatial_layer;
172 // Add some data to buffer.
Niels Möllerb9bfe652019-10-03 08:43:53 +0200173 frame->SetEncodedData(EncodedImageBuffer::Create(frame_size_bytes));
philipelbe7a9e52016-05-19 12:19:35 +0200174 for (size_t r = 0; r < references.size(); ++r)
175 frame->references[r] = references[r];
Niels Möller7cca0422019-04-29 16:12:19 +0200176 return frame;
177 }
philipelbe7a9e52016-05-19 12:19:35 +0200178
Niels Möller7cca0422019-04-29 16:12:19 +0200179 template <typename... T>
180 int InsertFrame(uint16_t picture_id,
181 uint8_t spatial_layer,
182 int64_t ts_ms,
Niels Möller7cca0422019-04-29 16:12:19 +0200183 bool last_spatial_layer,
Sergey Silkin2799e632019-05-17 09:51:39 +0200184 size_t frame_size_bytes,
Niels Möller7cca0422019-04-29 16:12:19 +0200185 T... refs) {
philipelcb327d92020-12-10 10:49:20 +0100186 return buffer_->InsertFrame(CreateFrame(picture_id, spatial_layer, ts_ms,
187 last_spatial_layer,
188 frame_size_bytes, refs...));
Niels Möller7cca0422019-04-29 16:12:19 +0200189 }
190
191 int InsertNackedFrame(uint16_t picture_id, int64_t ts_ms) {
192 std::unique_ptr<FrameObjectFake> frame =
philipelcb327d92020-12-10 10:49:20 +0100193 CreateFrame(picture_id, 0, ts_ms, true, kFrameSize);
Niels Möller7cca0422019-04-29 16:12:19 +0200194 frame->set_delayed_by_retransmission(true);
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200195 return buffer_->InsertFrame(std::move(frame));
philipelbe7a9e52016-05-19 12:19:35 +0200196 }
197
philipel3042c2d2017-08-18 04:55:02 -0700198 void ExtractFrame(int64_t max_wait_time = 0, bool keyframe_required = false) {
philipel19053972020-01-29 17:36:11 +0100199 time_task_queue_.PostTask([this, max_wait_time, keyframe_required]() {
200 buffer_->NextFrame(
201 max_wait_time, keyframe_required, &time_task_queue_,
philipelca188092021-03-23 12:00:49 +0100202 [this](std::unique_ptr<EncodedFrame> frame,
philipel19053972020-01-29 17:36:11 +0100203 video_coding::FrameBuffer::ReturnReason reason) {
204 if (reason != FrameBuffer::ReturnReason::kStopped) {
205 frames_.emplace_back(std::move(frame));
206 }
207 });
208 });
philipelbe7a9e52016-05-19 12:19:35 +0200209 if (max_wait_time == 0) {
Danil Chapovalov55284022020-02-07 14:53:52 +0100210 time_controller_.AdvanceTime(TimeDelta::Millis(0));
philipelbe7a9e52016-05-19 12:19:35 +0200211 }
212 }
213
214 void CheckFrame(size_t index, int picture_id, int spatial_layer) {
philipelbe7a9e52016-05-19 12:19:35 +0200215 ASSERT_LT(index, frames_.size());
216 ASSERT_TRUE(frames_[index]);
philipel9aa9b8d2021-02-15 13:31:29 +0100217 ASSERT_EQ(picture_id, frames_[index]->Id());
philipelcb327d92020-12-10 10:49:20 +0100218 ASSERT_EQ(spatial_layer, frames_[index]->SpatialIndex().value_or(0));
philipelbe7a9e52016-05-19 12:19:35 +0200219 }
220
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100221 void CheckFrameSize(size_t index, size_t size) {
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100222 ASSERT_LT(index, frames_.size());
223 ASSERT_TRUE(frames_[index]);
224 ASSERT_EQ(frames_[index]->size(), size);
225 }
226
philipelbe7a9e52016-05-19 12:19:35 +0200227 void CheckNoFrame(size_t index) {
philipelbe7a9e52016-05-19 12:19:35 +0200228 ASSERT_LT(index, frames_.size());
229 ASSERT_FALSE(frames_[index]);
230 }
231
philipelbe7a9e52016-05-19 12:19:35 +0200232 uint32_t Rand() { return rand_.Rand<uint32_t>(); }
233
philipel19053972020-01-29 17:36:11 +0100234 webrtc::GlobalSimulatedTimeController time_controller_;
235 rtc::TaskQueue time_task_queue_;
philipelbe7a9e52016-05-19 12:19:35 +0200236 VCMTimingFake timing_;
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200237 std::unique_ptr<FrameBuffer> buffer_;
philipele7c891f2018-02-22 14:35:06 +0100238 std::vector<std::unique_ptr<EncodedFrame>> frames_;
philipelbe7a9e52016-05-19 12:19:35 +0200239 Random rand_;
philipela45102f2017-02-22 05:30:39 -0800240 ::testing::NiceMock<VCMReceiveStatisticsCallbackMock> stats_callback_;
philipelbe7a9e52016-05-19 12:19:35 +0200241};
242
Sergey Silkin2799e632019-05-17 09:51:39 +0200243// From https://en.cppreference.com/w/cpp/language/static: "If ... a constexpr
244// static data member (since C++11) is odr-used, a definition at namespace scope
245// is still required... This definition is deprecated for constexpr data members
246// since C++17."
247// kFrameSize is odr-used since it is passed by reference to EXPECT_EQ().
248#if __cplusplus < 201703L
249constexpr size_t TestFrameBuffer2::kFrameSize;
250#endif
251
philipelbe7a9e52016-05-19 12:19:35 +0200252TEST_F(TestFrameBuffer2, WaitForFrame) {
253 uint16_t pid = Rand();
254 uint32_t ts = Rand();
255
philipel6e8224f2016-05-19 17:07:44 +0200256 ExtractFrame(50);
philipelcb327d92020-12-10 10:49:20 +0100257 InsertFrame(pid, 0, ts, true, kFrameSize);
Danil Chapovalov55284022020-02-07 14:53:52 +0100258 time_controller_.AdvanceTime(TimeDelta::Millis(50));
philipelbe7a9e52016-05-19 12:19:35 +0200259 CheckFrame(0, pid, 0);
260}
261
262TEST_F(TestFrameBuffer2, OneSuperFrame) {
263 uint16_t pid = Rand();
264 uint32_t ts = Rand();
265
philipelcb327d92020-12-10 10:49:20 +0100266 InsertFrame(pid, 0, ts, false, kFrameSize);
267 InsertFrame(pid + 1, 1, ts, true, kFrameSize);
philipele0b2f152016-09-28 10:23:49 +0200268 ExtractFrame();
269
Sergey Silkin61832dd2018-12-20 14:32:14 +0100270 CheckFrame(0, pid, 1);
philipele0b2f152016-09-28 10:23:49 +0200271}
272
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200273TEST_F(TestFrameBuffer2, ZeroPlayoutDelay) {
philipel19053972020-01-29 17:36:11 +0100274 VCMTiming timing(time_controller_.GetClock());
275 buffer_.reset(
276 new FrameBuffer(time_controller_.GetClock(), &timing, &stats_callback_));
Niels Möllerd381eed2020-09-02 15:34:40 +0200277 const VideoPlayoutDelay kPlayoutDelayMs = {0, 0};
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200278 std::unique_ptr<FrameObjectFake> test_frame(new FrameObjectFake());
philipel9aa9b8d2021-02-15 13:31:29 +0100279 test_frame->SetId(0);
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200280 test_frame->SetPlayoutDelay(kPlayoutDelayMs);
281 buffer_->InsertFrame(std::move(test_frame));
282 ExtractFrame(0, false);
283 CheckFrame(0, 0, 0);
284 EXPECT_EQ(0, frames_[0]->RenderTimeMs());
285}
286
aleloi3e005282017-01-26 05:38:00 -0800287// Flaky test, see bugs.webrtc.org/7068.
288TEST_F(TestFrameBuffer2, DISABLED_OneUnorderedSuperFrame) {
philipele0b2f152016-09-28 10:23:49 +0200289 uint16_t pid = Rand();
290 uint32_t ts = Rand();
291
philipel6e8224f2016-05-19 17:07:44 +0200292 ExtractFrame(50);
philipelcb327d92020-12-10 10:49:20 +0100293 InsertFrame(pid, 1, ts, true, kFrameSize);
294 InsertFrame(pid, 0, ts, false, kFrameSize);
Danil Chapovalov55284022020-02-07 14:53:52 +0100295 time_controller_.AdvanceTime(TimeDelta::Millis(0));
philipelbe7a9e52016-05-19 12:19:35 +0200296
297 CheckFrame(0, pid, 0);
298 CheckFrame(1, pid, 1);
299}
300
philipel94616b32016-05-20 10:43:01 +0200301TEST_F(TestFrameBuffer2, DISABLED_OneLayerStreamReordered) {
philipel6e8224f2016-05-19 17:07:44 +0200302 uint16_t pid = Rand();
303 uint32_t ts = Rand();
304
Sergey Silkin2799e632019-05-17 09:51:39 +0200305 InsertFrame(pid, 0, ts, false, true, kFrameSize);
philipel6e8224f2016-05-19 17:07:44 +0200306 ExtractFrame();
307 CheckFrame(0, pid, 0);
308 for (int i = 1; i < 10; i += 2) {
309 ExtractFrame(50);
philipelcb327d92020-12-10 10:49:20 +0100310 InsertFrame(pid + i + 1, 0, ts + (i + 1) * kFps10, true, kFrameSize,
Sergey Silkin2799e632019-05-17 09:51:39 +0200311 pid + i);
Danil Chapovalov55284022020-02-07 14:53:52 +0100312 time_controller_.AdvanceTime(TimeDelta::Millis(kFps10));
philipelcb327d92020-12-10 10:49:20 +0100313 InsertFrame(pid + i, 0, ts + i * kFps10, true, kFrameSize, pid + i - 1);
Danil Chapovalov55284022020-02-07 14:53:52 +0100314 time_controller_.AdvanceTime(TimeDelta::Millis(kFps10));
philipel6e8224f2016-05-19 17:07:44 +0200315 ExtractFrame();
316 CheckFrame(i, pid + i, 0);
317 CheckFrame(i + 1, pid + i + 1, 0);
318 }
319}
philipel6e8224f2016-05-19 17:07:44 +0200320
321TEST_F(TestFrameBuffer2, ExtractFromEmptyBuffer) {
322 ExtractFrame();
323 CheckNoFrame(0);
324}
325
philipel93e451b2016-10-06 12:25:13 +0200326TEST_F(TestFrameBuffer2, MissingFrame) {
327 uint16_t pid = Rand();
328 uint32_t ts = Rand();
329
philipelcb327d92020-12-10 10:49:20 +0100330 InsertFrame(pid, 0, ts, true, kFrameSize);
331 InsertFrame(pid + 2, 0, ts, true, kFrameSize, pid);
332 InsertFrame(pid + 3, 0, ts, true, kFrameSize, pid + 1, pid + 2);
philipel93e451b2016-10-06 12:25:13 +0200333 ExtractFrame();
334 ExtractFrame();
335 ExtractFrame();
336
337 CheckFrame(0, pid, 0);
338 CheckFrame(1, pid + 2, 0);
339 CheckNoFrame(2);
340}
341
philipelbe7a9e52016-05-19 12:19:35 +0200342TEST_F(TestFrameBuffer2, OneLayerStream) {
343 uint16_t pid = Rand();
344 uint32_t ts = Rand();
345
philipelcb327d92020-12-10 10:49:20 +0100346 InsertFrame(pid, 0, ts, true, kFrameSize);
philipelbe7a9e52016-05-19 12:19:35 +0200347 ExtractFrame();
348 CheckFrame(0, pid, 0);
349 for (int i = 1; i < 10; ++i) {
philipelcb327d92020-12-10 10:49:20 +0100350 InsertFrame(pid + i, 0, ts + i * kFps10, true, kFrameSize, pid + i - 1);
philipelbe7a9e52016-05-19 12:19:35 +0200351 ExtractFrame();
Danil Chapovalov55284022020-02-07 14:53:52 +0100352 time_controller_.AdvanceTime(TimeDelta::Millis(kFps10));
philipelbe7a9e52016-05-19 12:19:35 +0200353 CheckFrame(i, pid + i, 0);
354 }
355}
356
philipelbe7a9e52016-05-19 12:19:35 +0200357TEST_F(TestFrameBuffer2, DropTemporalLayerSlowDecoder) {
358 uint16_t pid = Rand();
359 uint32_t ts = Rand();
360
philipelcb327d92020-12-10 10:49:20 +0100361 InsertFrame(pid, 0, ts, true, kFrameSize);
362 InsertFrame(pid + 1, 0, ts + kFps20, true, kFrameSize, pid);
philipelbe7a9e52016-05-19 12:19:35 +0200363 for (int i = 2; i < 10; i += 2) {
364 uint32_t ts_tl0 = ts + i / 2 * kFps10;
philipelcb327d92020-12-10 10:49:20 +0100365 InsertFrame(pid + i, 0, ts_tl0, true, kFrameSize, pid + i - 2);
366 InsertFrame(pid + i + 1, 0, ts_tl0 + kFps20, true, kFrameSize, pid + i,
367 pid + i - 1);
philipelbe7a9e52016-05-19 12:19:35 +0200368 }
369
Johannes Kron0c141c52019-08-26 15:04:43 +0200370 EXPECT_CALL(stats_callback_, OnDroppedFrames(1)).Times(3);
371
philipelbe7a9e52016-05-19 12:19:35 +0200372 for (int i = 0; i < 10; ++i) {
373 ExtractFrame();
Danil Chapovalov55284022020-02-07 14:53:52 +0100374 time_controller_.AdvanceTime(TimeDelta::Millis(70));
philipelbe7a9e52016-05-19 12:19:35 +0200375 }
376
377 CheckFrame(0, pid, 0);
378 CheckFrame(1, pid + 1, 0);
379 CheckFrame(2, pid + 2, 0);
380 CheckFrame(3, pid + 4, 0);
381 CheckFrame(4, pid + 6, 0);
382 CheckFrame(5, pid + 8, 0);
383 CheckNoFrame(6);
384 CheckNoFrame(7);
385 CheckNoFrame(8);
386 CheckNoFrame(9);
387}
388
Johannes Kron0c141c52019-08-26 15:04:43 +0200389TEST_F(TestFrameBuffer2, DropFramesIfSystemIsStalled) {
390 uint16_t pid = Rand();
391 uint32_t ts = Rand();
392
philipelcb327d92020-12-10 10:49:20 +0100393 InsertFrame(pid, 0, ts, true, kFrameSize);
394 InsertFrame(pid + 1, 0, ts + 1 * kFps10, true, kFrameSize, pid);
395 InsertFrame(pid + 2, 0, ts + 2 * kFps10, true, kFrameSize, pid + 1);
396 InsertFrame(pid + 3, 0, ts + 3 * kFps10, true, kFrameSize);
Johannes Kron0c141c52019-08-26 15:04:43 +0200397
398 ExtractFrame();
399 // Jump forward in time, simulating the system being stalled for some reason.
Danil Chapovalov55284022020-02-07 14:53:52 +0100400 time_controller_.AdvanceTime(TimeDelta::Millis(3) * kFps10);
Johannes Kron0c141c52019-08-26 15:04:43 +0200401 // Extract one more frame, expect second and third frame to be dropped.
402 EXPECT_CALL(stats_callback_, OnDroppedFrames(2)).Times(1);
403 ExtractFrame();
404
405 CheckFrame(0, pid + 0, 0);
406 CheckFrame(1, pid + 3, 0);
407}
408
409TEST_F(TestFrameBuffer2, DroppedFramesCountedOnClear) {
410 uint16_t pid = Rand();
411 uint32_t ts = Rand();
412
philipelcb327d92020-12-10 10:49:20 +0100413 InsertFrame(pid, 0, ts, true, kFrameSize);
Johannes Kron0c141c52019-08-26 15:04:43 +0200414 for (int i = 1; i < 5; ++i) {
philipelcb327d92020-12-10 10:49:20 +0100415 InsertFrame(pid + i, 0, ts + i * kFps10, true, kFrameSize, pid + i - 1);
Johannes Kron0c141c52019-08-26 15:04:43 +0200416 }
417
418 // All frames should be dropped when Clear is called.
419 EXPECT_CALL(stats_callback_, OnDroppedFrames(5)).Times(1);
420 buffer_->Clear();
421}
422
philipelbe7a9e52016-05-19 12:19:35 +0200423TEST_F(TestFrameBuffer2, InsertLateFrame) {
424 uint16_t pid = Rand();
425 uint32_t ts = Rand();
426
philipelcb327d92020-12-10 10:49:20 +0100427 InsertFrame(pid, 0, ts, true, kFrameSize);
philipelbe7a9e52016-05-19 12:19:35 +0200428 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100429 InsertFrame(pid + 2, 0, ts, true, kFrameSize);
philipelbe7a9e52016-05-19 12:19:35 +0200430 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100431 InsertFrame(pid + 1, 0, ts, true, kFrameSize, pid);
philipelbe7a9e52016-05-19 12:19:35 +0200432 ExtractFrame();
433
434 CheckFrame(0, pid, 0);
435 CheckFrame(1, pid + 2, 0);
436 CheckNoFrame(2);
437}
438
Niels Möller7cca0422019-04-29 16:12:19 +0200439TEST_F(TestFrameBuffer2, ProtectionModeNackFEC) {
philipel4f6cd6a2016-08-03 10:59:32 +0200440 uint16_t pid = Rand();
441 uint32_t ts = Rand();
Niels Möller7cca0422019-04-29 16:12:19 +0200442 constexpr int64_t kRttMs = 200;
443 buffer_->UpdateRtt(kRttMs);
philipel4f6cd6a2016-08-03 10:59:32 +0200444
Niels Möller7cca0422019-04-29 16:12:19 +0200445 // Jitter estimate unaffected by RTT in this protection mode.
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200446 buffer_->SetProtectionMode(kProtectionNackFEC);
Niels Möller7cca0422019-04-29 16:12:19 +0200447 InsertNackedFrame(pid, ts);
448 InsertNackedFrame(pid + 1, ts + 100);
449 InsertNackedFrame(pid + 2, ts + 200);
philipelcb327d92020-12-10 10:49:20 +0100450 InsertFrame(pid + 3, 0, ts + 300, true, kFrameSize);
philipel4f6cd6a2016-08-03 10:59:32 +0200451 ExtractFrame();
Niels Möller7cca0422019-04-29 16:12:19 +0200452 ExtractFrame();
453 ExtractFrame();
454 ExtractFrame();
455 ASSERT_EQ(4u, frames_.size());
456 EXPECT_LT(timing_.GetCurrentJitter(), kRttMs);
457}
458
philipele0b2f152016-09-28 10:23:49 +0200459TEST_F(TestFrameBuffer2, NoContinuousFrame) {
460 uint16_t pid = Rand();
461 uint32_t ts = Rand();
462
philipelcb327d92020-12-10 10:49:20 +0100463 EXPECT_EQ(-1, InsertFrame(pid + 1, 0, ts, true, kFrameSize, pid));
philipele0b2f152016-09-28 10:23:49 +0200464}
465
466TEST_F(TestFrameBuffer2, LastContinuousFrameSingleLayer) {
467 uint16_t pid = Rand();
468 uint32_t ts = Rand();
469
philipelcb327d92020-12-10 10:49:20 +0100470 EXPECT_EQ(pid, InsertFrame(pid, 0, ts, true, kFrameSize));
471 EXPECT_EQ(pid, InsertFrame(pid + 2, 0, ts, true, kFrameSize, pid + 1));
472 EXPECT_EQ(pid + 2, InsertFrame(pid + 1, 0, ts, true, kFrameSize, pid));
473 EXPECT_EQ(pid + 2, InsertFrame(pid + 4, 0, ts, true, kFrameSize, pid + 3));
474 EXPECT_EQ(pid + 5, InsertFrame(pid + 5, 0, ts, true, kFrameSize));
philipele0b2f152016-09-28 10:23:49 +0200475}
476
477TEST_F(TestFrameBuffer2, LastContinuousFrameTwoLayers) {
478 uint16_t pid = Rand();
479 uint32_t ts = Rand();
480
philipelcb327d92020-12-10 10:49:20 +0100481 EXPECT_EQ(pid, InsertFrame(pid, 0, ts, false, kFrameSize));
482 EXPECT_EQ(pid + 1, InsertFrame(pid + 1, 1, ts, true, kFrameSize));
483 EXPECT_EQ(pid + 1,
484 InsertFrame(pid + 3, 1, ts, true, kFrameSize, pid + 1, pid + 2));
485 EXPECT_EQ(pid + 1, InsertFrame(pid + 4, 0, ts, false, kFrameSize, pid + 2));
486 EXPECT_EQ(pid + 1,
487 InsertFrame(pid + 5, 1, ts, true, kFrameSize, pid + 3, pid + 4));
488 EXPECT_EQ(pid + 1, InsertFrame(pid + 6, 0, ts, false, kFrameSize, pid + 4));
489 EXPECT_EQ(pid + 6, InsertFrame(pid + 2, 0, ts, false, kFrameSize, pid));
490 EXPECT_EQ(pid + 7,
491 InsertFrame(pid + 7, 1, ts, true, kFrameSize, pid + 5, pid + 6));
philipele0b2f152016-09-28 10:23:49 +0200492}
493
philipelfcc60062017-01-18 05:35:20 -0800494TEST_F(TestFrameBuffer2, PictureIdJumpBack) {
495 uint16_t pid = Rand();
496 uint32_t ts = Rand();
497
philipelcb327d92020-12-10 10:49:20 +0100498 EXPECT_EQ(pid, InsertFrame(pid, 0, ts, true, kFrameSize));
499 EXPECT_EQ(pid + 1, InsertFrame(pid + 1, 0, ts + 1, true, kFrameSize, pid));
philipelfcc60062017-01-18 05:35:20 -0800500 ExtractFrame();
501 CheckFrame(0, pid, 0);
502
503 // Jump back in pid but increase ts.
philipelcb327d92020-12-10 10:49:20 +0100504 EXPECT_EQ(pid - 1, InsertFrame(pid - 1, 0, ts + 2, true, kFrameSize));
philipelfcc60062017-01-18 05:35:20 -0800505 ExtractFrame();
506 ExtractFrame();
507 CheckFrame(1, pid - 1, 0);
508 CheckNoFrame(2);
509}
510
philipela45102f2017-02-22 05:30:39 -0800511TEST_F(TestFrameBuffer2, StatsCallback) {
512 uint16_t pid = Rand();
513 uint32_t ts = Rand();
514 const int kFrameSize = 5000;
515
ilnik6d5b4d62017-08-30 03:32:14 -0700516 EXPECT_CALL(stats_callback_,
517 OnCompleteFrame(true, kFrameSize, VideoContentType::UNSPECIFIED));
Johannes Kronbfd343b2019-07-01 10:07:50 +0200518 EXPECT_CALL(stats_callback_, OnFrameBufferTimingsUpdated(_, _, _, _, _, _));
philipela45102f2017-02-22 05:30:39 -0800519
520 {
521 std::unique_ptr<FrameObjectFake> frame(new FrameObjectFake());
Niels Möllerb9bfe652019-10-03 08:43:53 +0200522 frame->SetEncodedData(EncodedImageBuffer::Create(kFrameSize));
philipel9aa9b8d2021-02-15 13:31:29 +0100523 frame->SetId(pid);
Niels Möller23775882018-08-16 10:24:12 +0200524 frame->SetTimestamp(ts);
philipela45102f2017-02-22 05:30:39 -0800525 frame->num_references = 0;
philipela45102f2017-02-22 05:30:39 -0800526
Stefan Holmer812ceaf2018-05-15 13:00:10 +0200527 EXPECT_EQ(buffer_->InsertFrame(std::move(frame)), pid);
philipela45102f2017-02-22 05:30:39 -0800528 }
529
530 ExtractFrame();
531 CheckFrame(0, pid, 0);
532}
533
philipel146a48b2017-04-20 04:04:38 -0700534TEST_F(TestFrameBuffer2, ForwardJumps) {
philipelcb327d92020-12-10 10:49:20 +0100535 EXPECT_EQ(5453, InsertFrame(5453, 0, 1, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700536 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100537 EXPECT_EQ(5454, InsertFrame(5454, 0, 1, true, kFrameSize, 5453));
philipel146a48b2017-04-20 04:04:38 -0700538 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100539 EXPECT_EQ(15670, InsertFrame(15670, 0, 1, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700540 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100541 EXPECT_EQ(29804, InsertFrame(29804, 0, 1, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700542 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100543 EXPECT_EQ(29805, InsertFrame(29805, 0, 1, true, kFrameSize, 29804));
philipel146a48b2017-04-20 04:04:38 -0700544 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100545 EXPECT_EQ(29806, InsertFrame(29806, 0, 1, true, kFrameSize, 29805));
philipel146a48b2017-04-20 04:04:38 -0700546 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100547 EXPECT_EQ(33819, InsertFrame(33819, 0, 1, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700548 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100549 EXPECT_EQ(41248, InsertFrame(41248, 0, 1, true, kFrameSize));
philipel146a48b2017-04-20 04:04:38 -0700550 ExtractFrame();
551}
552
philipelf6842692017-04-28 03:29:15 -0700553TEST_F(TestFrameBuffer2, DuplicateFrames) {
philipelcb327d92020-12-10 10:49:20 +0100554 EXPECT_EQ(22256, InsertFrame(22256, 0, 1, true, kFrameSize));
philipelf6842692017-04-28 03:29:15 -0700555 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100556 EXPECT_EQ(22256, InsertFrame(22256, 0, 1, true, kFrameSize));
philipelf6842692017-04-28 03:29:15 -0700557}
558
philipel112adf92017-06-15 09:06:21 -0700559// TODO(philipel): implement more unittests related to invalid references.
560TEST_F(TestFrameBuffer2, InvalidReferences) {
philipelcb327d92020-12-10 10:49:20 +0100561 EXPECT_EQ(-1, InsertFrame(0, 0, 1000, true, kFrameSize, 2));
562 EXPECT_EQ(1, InsertFrame(1, 0, 2000, true, kFrameSize));
philipel112adf92017-06-15 09:06:21 -0700563 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100564 EXPECT_EQ(2, InsertFrame(2, 0, 3000, true, kFrameSize, 1));
philipel112adf92017-06-15 09:06:21 -0700565}
566
philipel3042c2d2017-08-18 04:55:02 -0700567TEST_F(TestFrameBuffer2, KeyframeRequired) {
philipelcb327d92020-12-10 10:49:20 +0100568 EXPECT_EQ(1, InsertFrame(1, 0, 1000, true, kFrameSize));
569 EXPECT_EQ(2, InsertFrame(2, 0, 2000, true, kFrameSize, 1));
570 EXPECT_EQ(3, InsertFrame(3, 0, 3000, true, kFrameSize));
philipel3042c2d2017-08-18 04:55:02 -0700571 ExtractFrame();
572 ExtractFrame(0, true);
573 ExtractFrame();
574
575 CheckFrame(0, 1, 0);
576 CheckFrame(1, 3, 0);
577 CheckNoFrame(2);
578}
579
philipel9771c502018-03-02 11:06:27 +0100580TEST_F(TestFrameBuffer2, KeyframeClearsFullBuffer) {
581 const int kMaxBufferSize = 600;
582
583 for (int i = 1; i <= kMaxBufferSize; ++i)
philipelcb327d92020-12-10 10:49:20 +0100584 EXPECT_EQ(-1, InsertFrame(i, 0, i * 1000, true, kFrameSize, i - 1));
philipel9771c502018-03-02 11:06:27 +0100585 ExtractFrame();
586 CheckNoFrame(0);
587
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100588 EXPECT_EQ(kMaxBufferSize + 1,
589 InsertFrame(kMaxBufferSize + 1, 0, (kMaxBufferSize + 1) * 1000,
philipelcb327d92020-12-10 10:49:20 +0100590 true, kFrameSize));
philipel9771c502018-03-02 11:06:27 +0100591 ExtractFrame();
592 CheckFrame(1, kMaxBufferSize + 1, 0);
593}
594
philipel798b2822018-06-11 13:10:14 +0200595TEST_F(TestFrameBuffer2, DontUpdateOnUndecodableFrame) {
philipelcb327d92020-12-10 10:49:20 +0100596 InsertFrame(1, 0, 0, true, kFrameSize);
philipel798b2822018-06-11 13:10:14 +0200597 ExtractFrame(0, true);
philipelcb327d92020-12-10 10:49:20 +0100598 InsertFrame(3, 0, 0, true, kFrameSize, 2, 0);
599 InsertFrame(3, 0, 0, true, kFrameSize, 0);
600 InsertFrame(2, 0, 0, true, kFrameSize);
philipel798b2822018-06-11 13:10:14 +0200601 ExtractFrame(0, true);
602 ExtractFrame(0, true);
603}
604
philipel6d216502018-10-22 14:36:45 +0200605TEST_F(TestFrameBuffer2, DontDecodeOlderTimestamp) {
philipelcb327d92020-12-10 10:49:20 +0100606 InsertFrame(2, 0, 1, true, kFrameSize);
607 InsertFrame(1, 0, 2, true,
Sergey Silkin2799e632019-05-17 09:51:39 +0200608 kFrameSize); // Older picture id but newer timestamp.
philipel6d216502018-10-22 14:36:45 +0200609 ExtractFrame(0);
610 ExtractFrame(0);
611 CheckFrame(0, 1, 0);
612 CheckNoFrame(1);
613
philipelcb327d92020-12-10 10:49:20 +0100614 InsertFrame(3, 0, 4, true, kFrameSize);
615 InsertFrame(4, 0, 3, true,
Sergey Silkin2799e632019-05-17 09:51:39 +0200616 kFrameSize); // Newer picture id but older timestamp.
philipel6d216502018-10-22 14:36:45 +0200617 ExtractFrame(0);
618 ExtractFrame(0);
619 CheckFrame(2, 3, 0);
620 CheckNoFrame(3);
621}
622
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100623TEST_F(TestFrameBuffer2, CombineFramesToSuperframe) {
624 uint16_t pid = Rand();
625 uint32_t ts = Rand();
626
philipelcb327d92020-12-10 10:49:20 +0100627 InsertFrame(pid, 0, ts, false, kFrameSize);
628 InsertFrame(pid + 1, 1, ts, true, 2 * kFrameSize, pid);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100629 ExtractFrame(0);
630 ExtractFrame(0);
Sergey Silkin61832dd2018-12-20 14:32:14 +0100631 CheckFrame(0, pid, 1);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100632 CheckNoFrame(1);
633 // Two frames should be combined and returned together.
Sergey Silkin2799e632019-05-17 09:51:39 +0200634 CheckFrameSize(0, 3 * kFrameSize);
635
636 EXPECT_EQ(frames_[0]->SpatialIndex(), 1);
637 EXPECT_EQ(frames_[0]->SpatialLayerFrameSize(0), kFrameSize);
638 EXPECT_EQ(frames_[0]->SpatialLayerFrameSize(1), 2 * kFrameSize);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100639}
640
641TEST_F(TestFrameBuffer2, HigherSpatialLayerNonDecodable) {
642 uint16_t pid = Rand();
643 uint32_t ts = Rand();
644
philipelcb327d92020-12-10 10:49:20 +0100645 InsertFrame(pid, 0, ts, false, kFrameSize);
646 InsertFrame(pid + 1, 1, ts, true, kFrameSize, pid);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100647
648 ExtractFrame(0);
Sergey Silkin61832dd2018-12-20 14:32:14 +0100649 CheckFrame(0, pid, 1);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100650
philipelcb327d92020-12-10 10:49:20 +0100651 InsertFrame(pid + 3, 1, ts + kFps20, true, kFrameSize, pid);
652 InsertFrame(pid + 4, 0, ts + kFps10, false, kFrameSize, pid);
653 InsertFrame(pid + 5, 1, ts + kFps10, true, kFrameSize, pid + 3, pid + 4);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100654
Danil Chapovalov55284022020-02-07 14:53:52 +0100655 time_controller_.AdvanceTime(TimeDelta::Millis(1000));
philipelcb327d92020-12-10 10:49:20 +0100656 // Frame pid+3 is decodable but too late.
657 // In superframe pid+4 is decodable, but frame pid+5 is not.
658 // Incorrect implementation might skip pid+2 frame and output undecodable
659 // pid+5 instead.
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100660 ExtractFrame();
661 ExtractFrame();
philipelcb327d92020-12-10 10:49:20 +0100662 CheckFrame(1, pid + 3, 1);
663 CheckFrame(2, pid + 4, 1);
Ilya Nikolaevskiy5546aef2018-12-04 15:54:52 +0100664}
665
philipelbe7a9e52016-05-19 12:19:35 +0200666} // namespace video_coding
667} // namespace webrtc