Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(314)

Side by Side Diff: media/cast/sender/video_encoder_impl_unittest.cc

Issue 877393003: [Cast] Software encoder support for varying video frame sizes. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Created 5 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2014 The Chromium Authors. All rights reserved. 1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include <vector> 5 #include <vector>
6 6
7 #include "base/bind.h" 7 #include "base/bind.h"
8 #include "base/memory/ref_counted.h" 8 #include "base/memory/ref_counted.h"
9 #include "base/memory/scoped_ptr.h" 9 #include "base/memory/scoped_ptr.h"
10 #include "media/base/video_frame.h" 10 #include "media/base/video_frame.h"
11 #include "media/cast/cast_defines.h" 11 #include "media/cast/cast_defines.h"
12 #include "media/cast/cast_environment.h" 12 #include "media/cast/cast_environment.h"
13 #include "media/cast/sender/video_encoder_impl.h" 13 #include "media/cast/sender/video_encoder_impl.h"
14 #include "media/cast/test/fake_single_thread_task_runner.h" 14 #include "media/cast/test/fake_single_thread_task_runner.h"
15 #include "media/cast/test/utility/default_config.h" 15 #include "media/cast/test/utility/default_config.h"
16 #include "media/cast/test/utility/video_utility.h" 16 #include "media/cast/test/utility/video_utility.h"
17 #include "testing/gmock/include/gmock/gmock.h" 17 #include "testing/gtest/include/gtest/gtest.h"
18 18
19 namespace media { 19 namespace media {
20 namespace cast { 20 namespace cast {
21 21
22 using testing::_; 22 class VideoEncoderImplTest : public ::testing::TestWithParam<Codec> {
23
24 namespace {
25 class TestVideoEncoderCallback
26 : public base::RefCountedThreadSafe<TestVideoEncoderCallback> {
27 public:
28 explicit TestVideoEncoderCallback(bool multiple_buffer_mode)
29 : multiple_buffer_mode_(multiple_buffer_mode),
30 count_frames_delivered_(0) {}
31
32 int count_frames_delivered() const {
33 return count_frames_delivered_;
34 }
35
36 void SetExpectedResult(uint32 expected_frame_id,
37 uint32 expected_last_referenced_frame_id,
38 uint32 expected_rtp_timestamp,
39 const base::TimeTicks& expected_reference_time) {
40 expected_frame_id_ = expected_frame_id;
41 expected_last_referenced_frame_id_ = expected_last_referenced_frame_id;
42 expected_rtp_timestamp_ = expected_rtp_timestamp;
43 expected_reference_time_ = expected_reference_time;
44 }
45
46 void DeliverEncodedVideoFrame(
47 scoped_ptr<EncodedFrame> encoded_frame) {
48 if (expected_frame_id_ != expected_last_referenced_frame_id_) {
49 EXPECT_EQ(EncodedFrame::DEPENDENT, encoded_frame->dependency);
50 } else if (!multiple_buffer_mode_) {
51 EXPECT_EQ(EncodedFrame::KEY, encoded_frame->dependency);
52 }
53 EXPECT_EQ(expected_frame_id_, encoded_frame->frame_id);
54 EXPECT_EQ(expected_last_referenced_frame_id_,
55 encoded_frame->referenced_frame_id)
56 << "frame id: " << expected_frame_id_;
57 EXPECT_EQ(expected_rtp_timestamp_, encoded_frame->rtp_timestamp);
58 EXPECT_EQ(expected_reference_time_, encoded_frame->reference_time);
59 EXPECT_FALSE(encoded_frame->data.empty());
60 ++count_frames_delivered_;
61 }
62
63 private:
64 friend class base::RefCountedThreadSafe<TestVideoEncoderCallback>;
65 virtual ~TestVideoEncoderCallback() {}
66
67 const bool multiple_buffer_mode_;
68 int count_frames_delivered_;
69
70 uint32 expected_frame_id_;
71 uint32 expected_last_referenced_frame_id_;
72 uint32 expected_rtp_timestamp_;
73 base::TimeTicks expected_reference_time_;
74
75 DISALLOW_COPY_AND_ASSIGN(TestVideoEncoderCallback);
76 };
77 } // namespace
78
79 class VideoEncoderImplTest : public ::testing::Test {
80 protected: 23 protected:
81 VideoEncoderImplTest() { 24 VideoEncoderImplTest()
82 video_config_ = GetDefaultVideoSenderConfig(); 25 : testing_clock_(new base::SimpleTestTickClock()),
83 video_config_.codec = CODEC_VIDEO_VP8; 26 task_runner_(new test::FakeSingleThreadTaskRunner(testing_clock_)),
84 gfx::Size size(video_config_.width, video_config_.height); 27 cast_environment_(new CastEnvironment(
85 video_frame_ = media::VideoFrame::CreateFrame( 28 scoped_ptr<base::TickClock>(testing_clock_).Pass(),
86 VideoFrame::I420, size, gfx::Rect(size), size, base::TimeDelta()); 29 task_runner_,
87 PopulateVideoFrame(video_frame_.get(), 123); 30 task_runner_,
31 task_runner_)),
32 video_config_(GetDefaultVideoSenderConfig()),
33 count_frames_delivered_(0) {
34 testing_clock_->Advance(base::TimeTicks::Now() - base::TimeTicks());
35 first_frame_time_ = testing_clock_->NowTicks();
88 } 36 }
89 37
90 ~VideoEncoderImplTest() override {} 38 ~VideoEncoderImplTest() override {}
91 39
92 void SetUp() override { 40 void SetUp() override {
93 testing_clock_ = new base::SimpleTestTickClock(); 41 video_config_.codec = GetParam();
94 testing_clock_->Advance(base::TimeTicks::Now() - base::TimeTicks());
95 task_runner_ = new test::FakeSingleThreadTaskRunner(testing_clock_);
96 cast_environment_ =
97 new CastEnvironment(scoped_ptr<base::TickClock>(testing_clock_).Pass(),
98 task_runner_,
99 task_runner_,
100 task_runner_);
101 } 42 }
102 43
103 void TearDown() override { 44 void TearDown() override {
104 video_encoder_.reset(); 45 video_encoder_.reset();
105 task_runner_->RunTasks(); 46 task_runner_->RunTasks();
106 } 47 }
107 48
108 void CreateEncoder() { 49 void CreateEncoder(bool three_buffer_mode) {
109 test_video_encoder_callback_ = new TestVideoEncoderCallback( 50 video_config_.max_number_of_video_buffers_used =
110 video_config_.max_number_of_video_buffers_used != 1); 51 (three_buffer_mode ? 3 : 1);
111 video_encoder_.reset( 52 video_encoder_.reset(new VideoEncoderImpl(
112 new VideoEncoderImpl(cast_environment_, video_config_)); 53 cast_environment_,
113 } 54 video_config_,
114 55 CastInitializationCallback()));
115 void AdvanceClockAndVideoFrameTimestamp() { 56 task_runner_->RunTasks();
57 }
58
59 VideoEncoder* video_encoder() const {
60 return video_encoder_.get();
61 }
62
63 void AdvanceClock() {
116 testing_clock_->Advance(base::TimeDelta::FromMilliseconds(33)); 64 testing_clock_->Advance(base::TimeDelta::FromMilliseconds(33));
117 video_frame_->set_timestamp( 65 }
118 video_frame_->timestamp() + base::TimeDelta::FromMilliseconds(33)); 66
119 } 67 base::TimeTicks Now() const {
120 68 return testing_clock_->NowTicks();
121 base::SimpleTestTickClock* testing_clock_; // Owned by CastEnvironment. 69 }
122 scoped_refptr<TestVideoEncoderCallback> test_video_encoder_callback_; 70
71 void RunTasks() const {
72 return task_runner_->RunTasks();
73 }
74
75 int count_frames_delivered() const {
76 return count_frames_delivered_;
77 }
78
79 VideoEncoder::FrameEncodedCallback CreateFrameDeliverCallback(
80 uint32 expected_frame_id,
81 uint32 expected_last_referenced_frame_id,
82 uint32 expected_rtp_timestamp,
83 const base::TimeTicks& expected_reference_time) {
84 return base::Bind(&VideoEncoderImplTest::DeliverEncodedVideoFrame,
85 base::Unretained(this),
86 expected_frame_id,
87 expected_last_referenced_frame_id,
88 expected_rtp_timestamp,
89 expected_reference_time);
90 }
91
92 scoped_refptr<media::VideoFrame> CreateTestVideoFrame(
93 const gfx::Size& size) const {
94 const scoped_refptr<media::VideoFrame> frame =
95 media::VideoFrame::CreateFrame(
96 VideoFrame::I420, size, gfx::Rect(size), size,
97 testing_clock_->NowTicks() - first_frame_time_);
98 PopulateVideoFrame(frame.get(), 123);
99 return frame;
100 }
101
102 private:
103 void DeliverEncodedVideoFrame(
hubbe 2015/01/31 00:55:51 A few comments wouldn't hurt in this file. Just so
miu 2015/01/31 04:53:04 Done.
104 uint32 expected_frame_id,
105 uint32 expected_last_referenced_frame_id,
106 uint32 expected_rtp_timestamp,
107 const base::TimeTicks& expected_reference_time,
108 scoped_ptr<EncodedFrame> encoded_frame) {
109 if (expected_frame_id != expected_last_referenced_frame_id) {
110 EXPECT_EQ(EncodedFrame::DEPENDENT, encoded_frame->dependency);
111 } else if (video_config_.max_number_of_video_buffers_used == 1) {
112 EXPECT_EQ(EncodedFrame::KEY, encoded_frame->dependency);
113 }
114 EXPECT_EQ(expected_frame_id, encoded_frame->frame_id);
115 EXPECT_EQ(expected_last_referenced_frame_id,
116 encoded_frame->referenced_frame_id)
117 << "frame id: " << expected_frame_id;
118 EXPECT_EQ(expected_rtp_timestamp, encoded_frame->rtp_timestamp);
119 EXPECT_EQ(expected_reference_time, encoded_frame->reference_time);
120 EXPECT_FALSE(encoded_frame->data.empty());
121 ++count_frames_delivered_;
122 }
123
124 base::SimpleTestTickClock* const testing_clock_; // Owned by CastEnvironment.
125 const scoped_refptr<test::FakeSingleThreadTaskRunner> task_runner_;
126 const scoped_refptr<CastEnvironment> cast_environment_;
123 VideoSenderConfig video_config_; 127 VideoSenderConfig video_config_;
124 scoped_refptr<test::FakeSingleThreadTaskRunner> task_runner_; 128 base::TimeTicks first_frame_time_;
125 scoped_ptr<VideoEncoder> video_encoder_; 129 scoped_ptr<VideoEncoder> video_encoder_;
126 scoped_refptr<media::VideoFrame> video_frame_; 130
127 131 int count_frames_delivered_;
128 scoped_refptr<CastEnvironment> cast_environment_;
129 132
130 DISALLOW_COPY_AND_ASSIGN(VideoEncoderImplTest); 133 DISALLOW_COPY_AND_ASSIGN(VideoEncoderImplTest);
131 }; 134 };
132 135
133 TEST_F(VideoEncoderImplTest, GeneratesKeyFrameThenOnlyDeltaFrames) { 136 TEST_P(VideoEncoderImplTest, GeneratesKeyFrameThenOnlyDeltaFrames) {
134 CreateEncoder(); 137 CreateEncoder(false);
135 138
136 VideoEncoder::FrameEncodedCallback frame_encoded_callback = 139 EXPECT_EQ(0, count_frames_delivered());
137 base::Bind(&TestVideoEncoderCallback::DeliverEncodedVideoFrame, 140
138 test_video_encoder_callback_.get()); 141 scoped_refptr<media::VideoFrame> video_frame =
139 142 CreateTestVideoFrame(gfx::Size(1280, 720));
140 EXPECT_EQ(0, test_video_encoder_callback_->count_frames_delivered()); 143 EXPECT_TRUE(video_encoder()->EncodeVideoFrame(
141 144 video_frame,
142 test_video_encoder_callback_->SetExpectedResult( 145 Now(),
143 0, 0, TimeDeltaToRtpDelta(video_frame_->timestamp(), kVideoFrequency), 146 CreateFrameDeliverCallback(
144 testing_clock_->NowTicks()); 147 0, 0, TimeDeltaToRtpDelta(video_frame->timestamp(), kVideoFrequency),
145 EXPECT_TRUE(video_encoder_->EncodeVideoFrame( 148 Now())));
146 video_frame_, testing_clock_->NowTicks(), frame_encoded_callback)); 149 RunTasks();
147 task_runner_->RunTasks();
148 150
149 for (uint32 frame_id = 1; frame_id < 10; ++frame_id) { 151 for (uint32 frame_id = 1; frame_id < 10; ++frame_id) {
150 AdvanceClockAndVideoFrameTimestamp(); 152 AdvanceClock();
151 test_video_encoder_callback_->SetExpectedResult( 153 video_frame = CreateTestVideoFrame(gfx::Size(1280, 720));
152 frame_id, 154 EXPECT_TRUE(video_encoder()->EncodeVideoFrame(
153 frame_id - 1, 155 video_frame,
154 TimeDeltaToRtpDelta(video_frame_->timestamp(), kVideoFrequency), 156 Now(),
155 testing_clock_->NowTicks()); 157 CreateFrameDeliverCallback(
156 EXPECT_TRUE(video_encoder_->EncodeVideoFrame( 158 frame_id, frame_id - 1,
157 video_frame_, testing_clock_->NowTicks(), frame_encoded_callback)); 159 TimeDeltaToRtpDelta(video_frame->timestamp(), kVideoFrequency),
158 task_runner_->RunTasks(); 160 Now())));
159 } 161 RunTasks();
160 162 }
161 EXPECT_EQ(10, test_video_encoder_callback_->count_frames_delivered()); 163
164 EXPECT_EQ(10, count_frames_delivered());
162 } 165 }
163 166
164 TEST_F(VideoEncoderImplTest, 167 TEST_P(VideoEncoderImplTest,
165 FramesDoNotDependOnUnackedFramesInMultiBufferMode) { 168 FramesDoNotDependOnUnackedFramesInMultiBufferMode) {
166 video_config_.max_number_of_video_buffers_used = 3; 169 if (GetParam() != CODEC_VIDEO_VP8)
167 CreateEncoder(); 170 return; // Only test multibuffer mode for the VP8 encoder.
168 171 CreateEncoder(true);
169 VideoEncoder::FrameEncodedCallback frame_encoded_callback = 172
170 base::Bind(&TestVideoEncoderCallback::DeliverEncodedVideoFrame, 173 EXPECT_EQ(0, count_frames_delivered());
171 test_video_encoder_callback_.get()); 174
172 175 scoped_refptr<media::VideoFrame> video_frame =
173 EXPECT_EQ(0, test_video_encoder_callback_->count_frames_delivered()); 176 CreateTestVideoFrame(gfx::Size(1280, 720));
174 177 EXPECT_TRUE(video_encoder()->EncodeVideoFrame(
175 test_video_encoder_callback_->SetExpectedResult( 178 video_frame,
176 0, 0, TimeDeltaToRtpDelta(video_frame_->timestamp(), kVideoFrequency), 179 Now(),
177 testing_clock_->NowTicks()); 180 CreateFrameDeliverCallback(
178 EXPECT_TRUE(video_encoder_->EncodeVideoFrame( 181 0, 0, TimeDeltaToRtpDelta(video_frame->timestamp(), kVideoFrequency),
179 video_frame_, testing_clock_->NowTicks(), frame_encoded_callback)); 182 Now())));
180 task_runner_->RunTasks(); 183 RunTasks();
181 184
182 AdvanceClockAndVideoFrameTimestamp(); 185 AdvanceClock();
183 video_encoder_->LatestFrameIdToReference(0); 186 video_encoder()->LatestFrameIdToReference(0);
184 test_video_encoder_callback_->SetExpectedResult( 187 video_frame = CreateTestVideoFrame(gfx::Size(1280, 720));
185 1, 0, TimeDeltaToRtpDelta(video_frame_->timestamp(), kVideoFrequency), 188 EXPECT_TRUE(video_encoder()->EncodeVideoFrame(
186 testing_clock_->NowTicks()); 189 video_frame,
187 EXPECT_TRUE(video_encoder_->EncodeVideoFrame( 190 Now(),
188 video_frame_, testing_clock_->NowTicks(), frame_encoded_callback)); 191 CreateFrameDeliverCallback(
189 task_runner_->RunTasks(); 192 1, 0, TimeDeltaToRtpDelta(video_frame->timestamp(), kVideoFrequency),
190 193 Now())));
191 AdvanceClockAndVideoFrameTimestamp(); 194 RunTasks();
192 video_encoder_->LatestFrameIdToReference(1); 195
193 test_video_encoder_callback_->SetExpectedResult( 196 AdvanceClock();
194 2, 1, TimeDeltaToRtpDelta(video_frame_->timestamp(), kVideoFrequency), 197 video_encoder()->LatestFrameIdToReference(1);
195 testing_clock_->NowTicks()); 198 video_frame = CreateTestVideoFrame(gfx::Size(1280, 720));
196 EXPECT_TRUE(video_encoder_->EncodeVideoFrame( 199 EXPECT_TRUE(video_encoder()->EncodeVideoFrame(
197 video_frame_, testing_clock_->NowTicks(), frame_encoded_callback)); 200 video_frame,
198 task_runner_->RunTasks(); 201 Now(),
199 202 CreateFrameDeliverCallback(
200 video_encoder_->LatestFrameIdToReference(2); 203 2, 1, TimeDeltaToRtpDelta(video_frame->timestamp(), kVideoFrequency),
204 Now())));
205 RunTasks();
206
207 video_encoder()->LatestFrameIdToReference(2);
201 208
202 for (uint32 frame_id = 3; frame_id < 10; ++frame_id) { 209 for (uint32 frame_id = 3; frame_id < 10; ++frame_id) {
203 AdvanceClockAndVideoFrameTimestamp(); 210 AdvanceClock();
204 test_video_encoder_callback_->SetExpectedResult( 211 video_frame = CreateTestVideoFrame(gfx::Size(1280, 720));
205 frame_id, 2, 212 EXPECT_TRUE(video_encoder()->EncodeVideoFrame(
206 TimeDeltaToRtpDelta(video_frame_->timestamp(), kVideoFrequency), 213 video_frame,
207 testing_clock_->NowTicks()); 214 Now(),
208 EXPECT_TRUE(video_encoder_->EncodeVideoFrame( 215 CreateFrameDeliverCallback(
209 video_frame_, testing_clock_->NowTicks(), frame_encoded_callback)); 216 frame_id, 2,
210 task_runner_->RunTasks(); 217 TimeDeltaToRtpDelta(video_frame->timestamp(), kVideoFrequency),
211 } 218 Now())));
212 219 RunTasks();
213 EXPECT_EQ(10, test_video_encoder_callback_->count_frames_delivered()); 220 }
221
222 EXPECT_EQ(10, count_frames_delivered());
214 } 223 }
215 224
225 // This tests that the encoder continues to output EncodedFrames as the frame
226 // size changes. See media/cast/receiver/video_decoder_unittest.cc for a
227 // complete encode/decode cycle of varied frame sizes that actually checks the
228 // frame content.
229 TEST_P(VideoEncoderImplTest, EncodesVariedFrameSizes) {
230 CreateEncoder(false);
231 ASSERT_TRUE(video_encoder()->CanEncodeVariedFrameSizes());
232
233 EXPECT_EQ(0, count_frames_delivered());
234
235 std::vector<gfx::Size> frame_sizes;
236 frame_sizes.push_back(gfx::Size(1280, 720));
237 frame_sizes.push_back(gfx::Size(640, 360)); // Shrink both dimensions.
238 frame_sizes.push_back(gfx::Size(300, 200)); // Shrink both dimensions again.
239 frame_sizes.push_back(gfx::Size(200, 300)); // Same area.
240 frame_sizes.push_back(gfx::Size(600, 400)); // Grow both dimensions.
241 frame_sizes.push_back(gfx::Size(638, 400)); // Shrink only one dimension.
242 frame_sizes.push_back(gfx::Size(638, 398)); // Shrink the other dimension.
243 frame_sizes.push_back(gfx::Size(320, 180)); // Shrink both dimensions again.
244 frame_sizes.push_back(gfx::Size(322, 180)); // Grow only one dimension.
245 frame_sizes.push_back(gfx::Size(322, 182)); // Grow the other dimension.
246 frame_sizes.push_back(gfx::Size(1920, 1080)); // Grow both dimensions again.
247
248 uint32 frame_id = 0;
249
250 // Encode one frame at each size. Expect nothing but key frames to come out.
251 for (const auto& frame_size : frame_sizes) {
252 AdvanceClock();
253 const scoped_refptr<media::VideoFrame> video_frame =
254 CreateTestVideoFrame(frame_size);
255 EXPECT_TRUE(video_encoder()->EncodeVideoFrame(
256 video_frame,
257 Now(),
258 CreateFrameDeliverCallback(
259 frame_id,
260 frame_id,
261 TimeDeltaToRtpDelta(video_frame->timestamp(), kVideoFrequency),
262 Now())));
263 RunTasks();
264 ++frame_id;
265 }
266
267 // Encode 10 frames at each size. Expect one key frame followed by nine delta
268 // frames for each frame size.
269 for (const auto& frame_size : frame_sizes) {
270 for (int i = 0; i < 10; ++i) {
271 AdvanceClock();
272 const scoped_refptr<media::VideoFrame> video_frame =
273 CreateTestVideoFrame(frame_size);
274 EXPECT_TRUE(video_encoder()->EncodeVideoFrame(
275 video_frame,
276 Now(),
277 CreateFrameDeliverCallback(
278 frame_id,
279 i == 0 ? frame_id : frame_id - 1,
280 TimeDeltaToRtpDelta(video_frame->timestamp(), kVideoFrequency),
281 Now())));
282 RunTasks();
283 ++frame_id;
284 }
285 }
286
287 EXPECT_EQ(static_cast<int>(frame_id), count_frames_delivered());
288 }
289
290 INSTANTIATE_TEST_CASE_P(,
291 VideoEncoderImplTest,
292 ::testing::Values(CODEC_VIDEO_FAKE, CODEC_VIDEO_VP8));
293
216 } // namespace cast 294 } // namespace cast
217 } // namespace media 295 } // namespace media
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698