blob: e307d9f70348cedca6fd02d2f3a53041ae57d359 [file] [log] [blame]
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:13 +00001/*
kjellander1afca732016-02-07 20:46:45 -08002 * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved.
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:13 +00003 *
kjellander1afca732016-02-07 20:46:45 -08004 * 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.
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:13 +00009 */
henrike@webrtc.org28e20752013-07-10 00:45:36 +000010
kjellandera96e2d72016-02-04 23:52:28 -080011#ifndef WEBRTC_MEDIA_BASE_VIDEOENGINE_UNITTEST_H_ // NOLINT
12#define WEBRTC_MEDIA_BASE_VIDEOENGINE_UNITTEST_H_
henrike@webrtc.org28e20752013-07-10 00:45:36 +000013
kwiberg686a8ef2016-02-26 03:00:35 -080014#include <memory>
henrike@webrtc.org28e20752013-07-10 00:45:36 +000015#include <string>
16#include <vector>
17
buildbot@webrtc.orga09a9992014-08-13 17:26:08 +000018#include "webrtc/base/bytebuffer.h"
19#include "webrtc/base/gunit.h"
20#include "webrtc/base/timeutils.h"
Fredrik Solenberg709ed672015-09-15 12:26:33 +020021#include "webrtc/call.h"
skvlad11a9cbf2016-10-07 11:53:05 -070022#include "webrtc/logging/rtc_event_log/rtc_event_log.h"
kjellandera96e2d72016-02-04 23:52:28 -080023#include "webrtc/media/base/fakenetworkinterface.h"
24#include "webrtc/media/base/fakevideocapturer.h"
25#include "webrtc/media/base/fakevideorenderer.h"
26#include "webrtc/media/base/mediachannel.h"
27#include "webrtc/media/base/streamparams.h"
kjellander@webrtc.org5ad12972016-02-12 06:39:40 +010028#include "webrtc/media/engine/fakewebrtccall.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000029
henrike@webrtc.org28e20752013-07-10 00:45:36 +000030#define EXPECT_FRAME_WAIT(c, w, h, t) \
31 EXPECT_EQ_WAIT((c), renderer_.num_rendered_frames(), (t)); \
32 EXPECT_EQ((w), renderer_.width()); \
33 EXPECT_EQ((h), renderer_.height()); \
34 EXPECT_EQ(0, renderer_.errors()); \
35
36#define EXPECT_FRAME_ON_RENDERER_WAIT(r, c, w, h, t) \
37 EXPECT_EQ_WAIT((c), (r).num_rendered_frames(), (t)); \
38 EXPECT_EQ((w), (r).width()); \
39 EXPECT_EQ((h), (r).height()); \
40 EXPECT_EQ(0, (r).errors()); \
41
wu@webrtc.org9caf2762013-12-11 18:25:07 +000042#define EXPECT_GT_FRAME_ON_RENDERER_WAIT(r, c, w, h, t) \
43 EXPECT_TRUE_WAIT((r).num_rendered_frames() >= (c) && \
44 (w) == (r).width() && \
45 (h) == (r).height(), (t)); \
Peter Boström0c4e06b2015-10-07 12:23:21 +020046 EXPECT_EQ(0, (r).errors());
wu@webrtc.org9caf2762013-12-11 18:25:07 +000047
Peter Boström0c4e06b2015-10-07 12:23:21 +020048static const uint32_t kTimeout = 5000U;
49static const uint32_t kDefaultReceiveSsrc = 0;
50static const uint32_t kSsrc = 1234u;
51static const uint32_t kRtxSsrc = 4321u;
52static const uint32_t kSsrcs4[] = {1, 2, 3, 4};
perkj26752742016-10-24 01:21:16 -070053static const int kVideoWidth = 640;
54static const int kVideoHeight = 360;
55static const int kFramerate = 30;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000056
57inline bool IsEqualCodec(const cricket::VideoCodec& a,
58 const cricket::VideoCodec& b) {
perkj26752742016-10-24 01:21:16 -070059 return a.id == b.id && a.name == b.name;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000060}
61
sergeyu@chromium.org5bc25c42013-12-05 00:24:06 +000062namespace std {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000063inline std::ostream& operator<<(std::ostream& s, const cricket::VideoCodec& c) {
perkj26752742016-10-24 01:21:16 -070064 s << "{" << c.name << "(" << c.id << ")"
65 << "}";
henrike@webrtc.org28e20752013-07-10 00:45:36 +000066 return s;
67}
sergeyu@chromium.org5bc25c42013-12-05 00:24:06 +000068} // namespace std
henrike@webrtc.org28e20752013-07-10 00:45:36 +000069
henrike@webrtc.org28e20752013-07-10 00:45:36 +000070template<class E, class C>
71class VideoMediaChannelTest : public testing::Test,
72 public sigslot::has_slots<> {
73 protected:
Fredrik Solenberg709ed672015-09-15 12:26:33 +020074 VideoMediaChannelTest<E, C>()
skvlad11a9cbf2016-10-07 11:53:05 -070075 : call_(webrtc::Call::Create(webrtc::Call::Config(&event_log_))) {}
Fredrik Solenberg709ed672015-09-15 12:26:33 +020076
henrike@webrtc.org28e20752013-07-10 00:45:36 +000077 virtual cricket::VideoCodec DefaultCodec() = 0;
78
79 virtual cricket::StreamParams DefaultSendStreamParams() {
80 return cricket::StreamParams::CreateLegacy(kSsrc);
81 }
82
83 virtual void SetUp() {
Fredrik Solenberg9a416bd2015-05-22 09:04:09 +020084 engine_.Init();
nisse51542be2016-02-12 02:27:06 -080085 channel_.reset(engine_.CreateChannel(call_.get(), cricket::MediaConfig(),
86 cricket::VideoOptions()));
Sergey Ulanove2b15012016-11-22 16:08:30 -080087 channel_->OnReadyToSend(true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000088 EXPECT_TRUE(channel_.get() != NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000089 network_interface_.SetDestination(channel_.get());
90 channel_->SetInterface(&network_interface_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000091 media_error_ = cricket::VideoMediaChannel::ERROR_NONE;
Fredrik Solenbergb071a192015-09-17 16:42:56 +020092 cricket::VideoRecvParameters parameters;
93 parameters.codecs = engine_.codecs();
94 channel_->SetRecvParameters(parameters);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000095 EXPECT_TRUE(channel_->AddSendStream(DefaultSendStreamParams()));
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +000096 video_capturer_.reset(CreateFakeVideoCapturer());
henrike@webrtc.org28654cb2013-07-22 21:07:49 +000097 cricket::VideoFormat format(640, 480,
perkj26752742016-10-24 01:21:16 -070098 cricket::VideoFormat::FpsToInterval(kFramerate),
henrike@webrtc.org28654cb2013-07-22 21:07:49 +000099 cricket::FOURCC_I420);
100 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_->Start(format));
deadbeef5a4a75a2016-06-02 16:23:38 -0700101 EXPECT_TRUE(
102 channel_->SetVideoSend(kSsrc, true, nullptr, video_capturer_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000103 }
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000104
105 virtual cricket::FakeVideoCapturer* CreateFakeVideoCapturer() {
106 return new cricket::FakeVideoCapturer();
107 }
108
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000109 // Utility method to setup an additional stream to send and receive video.
110 // Used to test send and recv between two streams.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000111 void SetUpSecondStream() {
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000112 SetUpSecondStreamWithNoRecv();
113 // Setup recv for second stream.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000114 EXPECT_TRUE(channel_->AddRecvStream(
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000115 cricket::StreamParams::CreateLegacy(kSsrc + 2)));
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000116 // Make the second renderer available for use by a new stream.
nisse08582ff2016-02-04 01:24:52 -0800117 EXPECT_TRUE(channel_->SetSink(kSsrc + 2, &renderer2_));
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000118 }
119 // Setup an additional stream just to send video. Defer add recv stream.
120 // This is required if you want to test unsignalled recv of video rtp packets.
121 void SetUpSecondStreamWithNoRecv() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000122 // SetUp() already added kSsrc make sure duplicate SSRCs cant be added.
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000123 EXPECT_TRUE(channel_->AddRecvStream(
124 cricket::StreamParams::CreateLegacy(kSsrc)));
nisse08582ff2016-02-04 01:24:52 -0800125 EXPECT_TRUE(channel_->SetSink(kSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000126 EXPECT_FALSE(channel_->AddSendStream(
127 cricket::StreamParams::CreateLegacy(kSsrc)));
128 EXPECT_TRUE(channel_->AddSendStream(
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000129 cricket::StreamParams::CreateLegacy(kSsrc + 2)));
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000130 // We dont add recv for the second stream.
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000131
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000132 // Setup the receive and renderer for second stream after send.
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000133 video_capturer_2_.reset(CreateFakeVideoCapturer());
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000134 cricket::VideoFormat format(640, 480,
perkj26752742016-10-24 01:21:16 -0700135 cricket::VideoFormat::FpsToInterval(kFramerate),
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000136 cricket::FOURCC_I420);
137 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_2_->Start(format));
138
deadbeef5a4a75a2016-06-02 16:23:38 -0700139 EXPECT_TRUE(channel_->SetVideoSend(kSsrc + 2, true, nullptr,
140 video_capturer_2_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000141 }
142 virtual void TearDown() {
143 channel_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000144 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000145 bool SetDefaultCodec() {
146 return SetOneCodec(DefaultCodec());
147 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000148
perkj26752742016-10-24 01:21:16 -0700149 bool SetOneCodec(int pt, const char* name) {
150 return SetOneCodec(cricket::VideoCodec(pt, name));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000151 }
152 bool SetOneCodec(const cricket::VideoCodec& codec) {
perkj26752742016-10-24 01:21:16 -0700153 cricket::VideoFormat capture_format(
154 kVideoWidth, kVideoHeight,
155 cricket::VideoFormat::FpsToInterval(kFramerate), cricket::FOURCC_I420);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000156
157 if (video_capturer_) {
158 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_->Start(capture_format));
159 }
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000160 if (video_capturer_2_) {
161 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_2_->Start(capture_format));
162 }
163
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000164 bool sending = channel_->sending();
165 bool success = SetSend(false);
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200166 if (success) {
167 cricket::VideoSendParameters parameters;
168 parameters.codecs.push_back(codec);
169 success = channel_->SetSendParameters(parameters);
170 }
171 if (success) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000172 success = SetSend(sending);
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200173 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000174 return success;
175 }
176 bool SetSend(bool send) {
177 return channel_->SetSend(send);
178 }
179 int DrainOutgoingPackets() {
180 int packets = 0;
181 do {
182 packets = NumRtpPackets();
183 // 100 ms should be long enough.
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000184 rtc::Thread::Current()->ProcessMessages(100);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000185 } while (NumRtpPackets() > packets);
186 return NumRtpPackets();
187 }
188 bool SendFrame() {
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000189 if (video_capturer_2_) {
190 video_capturer_2_->CaptureFrame();
191 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000192 return video_capturer_.get() &&
193 video_capturer_->CaptureFrame();
194 }
195 bool WaitAndSendFrame(int wait_ms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000196 bool ret = rtc::Thread::Current()->ProcessMessages(wait_ms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000197 ret &= SendFrame();
198 return ret;
199 }
200 // Sends frames and waits for the decoder to be fully initialized.
201 // Returns the number of frames that were sent.
202 int WaitForDecoder() {
203#if defined(HAVE_OPENMAX)
204 // Send enough frames for the OpenMAX decoder to continue processing, and
205 // return the number of frames sent.
206 // Send frames for a full kTimeout's worth of 15fps video.
207 int frame_count = 0;
208 while (frame_count < static_cast<int>(kTimeout) / 66) {
209 EXPECT_TRUE(WaitAndSendFrame(66));
210 ++frame_count;
211 }
212 return frame_count;
213#else
214 return 0;
215#endif
216 }
217 bool SendCustomVideoFrame(int w, int h) {
218 if (!video_capturer_.get()) return false;
219 return video_capturer_->CaptureCustomFrame(w, h, cricket::FOURCC_I420);
220 }
221 int NumRtpBytes() {
222 return network_interface_.NumRtpBytes();
223 }
Peter Boström0c4e06b2015-10-07 12:23:21 +0200224 int NumRtpBytes(uint32_t ssrc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000225 return network_interface_.NumRtpBytes(ssrc);
226 }
227 int NumRtpPackets() {
228 return network_interface_.NumRtpPackets();
229 }
Peter Boström0c4e06b2015-10-07 12:23:21 +0200230 int NumRtpPackets(uint32_t ssrc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000231 return network_interface_.NumRtpPackets(ssrc);
232 }
233 int NumSentSsrcs() {
234 return network_interface_.NumSentSsrcs();
235 }
jbaucheec21bd2016-03-20 06:15:43 -0700236 const rtc::CopyOnWriteBuffer* GetRtpPacket(int index) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000237 return network_interface_.GetRtpPacket(index);
238 }
239 int NumRtcpPackets() {
240 return network_interface_.NumRtcpPackets();
241 }
jbaucheec21bd2016-03-20 06:15:43 -0700242 const rtc::CopyOnWriteBuffer* GetRtcpPacket(int index) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000243 return network_interface_.GetRtcpPacket(index);
244 }
jbaucheec21bd2016-03-20 06:15:43 -0700245 static int GetPayloadType(const rtc::CopyOnWriteBuffer* p) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000246 int pt = -1;
247 ParseRtpPacket(p, NULL, &pt, NULL, NULL, NULL, NULL);
248 return pt;
249 }
jbaucheec21bd2016-03-20 06:15:43 -0700250 static bool ParseRtpPacket(const rtc::CopyOnWriteBuffer* p,
Peter Boström0c4e06b2015-10-07 12:23:21 +0200251 bool* x,
252 int* pt,
253 int* seqnum,
254 uint32_t* tstamp,
255 uint32_t* ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000256 std::string* payload) {
jbauchf1f87202016-03-30 06:43:37 -0700257 rtc::ByteBufferReader buf(p->data<char>(), p->size());
Peter Boström0c4e06b2015-10-07 12:23:21 +0200258 uint8_t u08 = 0;
259 uint16_t u16 = 0;
260 uint32_t u32 = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000261
262 // Read X and CC fields.
263 if (!buf.ReadUInt8(&u08)) return false;
264 bool extension = ((u08 & 0x10) != 0);
Peter Boström0c4e06b2015-10-07 12:23:21 +0200265 uint8_t cc = (u08 & 0x0F);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000266 if (x) *x = extension;
267
268 // Read PT field.
269 if (!buf.ReadUInt8(&u08)) return false;
270 if (pt) *pt = (u08 & 0x7F);
271
272 // Read Sequence Number field.
273 if (!buf.ReadUInt16(&u16)) return false;
274 if (seqnum) *seqnum = u16;
275
276 // Read Timestamp field.
277 if (!buf.ReadUInt32(&u32)) return false;
278 if (tstamp) *tstamp = u32;
279
280 // Read SSRC field.
281 if (!buf.ReadUInt32(&u32)) return false;
282 if (ssrc) *ssrc = u32;
283
284 // Skip CSRCs.
Peter Boström0c4e06b2015-10-07 12:23:21 +0200285 for (uint8_t i = 0; i < cc; ++i) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000286 if (!buf.ReadUInt32(&u32)) return false;
287 }
288
289 // Skip extension header.
290 if (extension) {
291 // Read Profile-specific extension header ID
292 if (!buf.ReadUInt16(&u16)) return false;
293
294 // Read Extension header length
295 if (!buf.ReadUInt16(&u16)) return false;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200296 uint16_t ext_header_len = u16;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000297
298 // Read Extension header
Peter Boström0c4e06b2015-10-07 12:23:21 +0200299 for (uint16_t i = 0; i < ext_header_len; ++i) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000300 if (!buf.ReadUInt32(&u32)) return false;
301 }
302 }
303
304 if (payload) {
305 return buf.ReadString(payload, buf.Length());
306 }
307 return true;
308 }
309
310 // Parse all RTCP packet, from start_index to stop_index, and count how many
311 // FIR (PT=206 and FMT=4 according to RFC 5104). If successful, set the count
312 // and return true.
313 bool CountRtcpFir(int start_index, int stop_index, int* fir_count) {
314 int count = 0;
315 for (int i = start_index; i < stop_index; ++i) {
jbaucheec21bd2016-03-20 06:15:43 -0700316 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtcpPacket(i));
jbauchf1f87202016-03-30 06:43:37 -0700317 rtc::ByteBufferReader buf(p->data<char>(), p->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000318 size_t total_len = 0;
319 // The packet may be a compound RTCP packet.
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000320 while (total_len < p->size()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000321 // Read FMT, type and length.
Peter Boström0c4e06b2015-10-07 12:23:21 +0200322 uint8_t fmt = 0;
323 uint8_t type = 0;
324 uint16_t length = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000325 if (!buf.ReadUInt8(&fmt)) return false;
326 fmt &= 0x1F;
327 if (!buf.ReadUInt8(&type)) return false;
328 if (!buf.ReadUInt16(&length)) return false;
329 buf.Consume(length * 4); // Skip RTCP data.
330 total_len += (length + 1) * 4;
331 if ((192 == type) || ((206 == type) && (4 == fmt))) {
332 ++count;
333 }
334 }
335 }
336
337 if (fir_count) {
338 *fir_count = count;
339 }
340 return true;
341 }
342
Peter Boström0c4e06b2015-10-07 12:23:21 +0200343 void OnVideoChannelError(uint32_t ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000344 cricket::VideoMediaChannel::Error error) {
345 media_error_ = error;
346 }
347
348 // Test that SetSend works.
349 void SetSend() {
350 EXPECT_FALSE(channel_->sending());
deadbeef5a4a75a2016-06-02 16:23:38 -0700351 EXPECT_TRUE(
352 channel_->SetVideoSend(kSsrc, true, nullptr, video_capturer_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000353 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
354 EXPECT_FALSE(channel_->sending());
355 EXPECT_TRUE(SetSend(true));
356 EXPECT_TRUE(channel_->sending());
357 EXPECT_TRUE(SendFrame());
358 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
359 EXPECT_TRUE(SetSend(false));
360 EXPECT_FALSE(channel_->sending());
361 }
362 // Test that SetSend fails without codecs being set.
363 void SetSendWithoutCodecs() {
364 EXPECT_FALSE(channel_->sending());
365 EXPECT_FALSE(SetSend(true));
366 EXPECT_FALSE(channel_->sending());
367 }
368 // Test that we properly set the send and recv buffer sizes by the time
369 // SetSend is called.
370 void SetSendSetsTransportBufferSizes() {
371 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
372 EXPECT_TRUE(SetSend(true));
buildbot@webrtc.orgae694ef2014-10-28 17:37:17 +0000373 EXPECT_EQ(64 * 1024, network_interface_.sendbuf_size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000374 EXPECT_EQ(64 * 1024, network_interface_.recvbuf_size());
375 }
376 // Tests that we can send frames and the right payload type is used.
377 void Send(const cricket::VideoCodec& codec) {
378 EXPECT_TRUE(SetOneCodec(codec));
379 EXPECT_TRUE(SetSend(true));
380 EXPECT_TRUE(SendFrame());
381 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
jbaucheec21bd2016-03-20 06:15:43 -0700382 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000383 EXPECT_EQ(codec.id, GetPayloadType(p.get()));
384 }
385 // Tests that we can send and receive frames.
386 void SendAndReceive(const cricket::VideoCodec& codec) {
387 EXPECT_TRUE(SetOneCodec(codec));
388 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800389 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000390 EXPECT_EQ(0, renderer_.num_rendered_frames());
391 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700392 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
jbaucheec21bd2016-03-20 06:15:43 -0700393 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000394 EXPECT_EQ(codec.id, GetPayloadType(p.get()));
395 }
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000396 void SendReceiveManyAndGetStats(const cricket::VideoCodec& codec,
397 int duration_sec, int fps) {
398 EXPECT_TRUE(SetOneCodec(codec));
399 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800400 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000401 EXPECT_EQ(0, renderer_.num_rendered_frames());
402 for (int i = 0; i < duration_sec; ++i) {
403 for (int frame = 1; frame <= fps; ++frame) {
404 EXPECT_TRUE(WaitAndSendFrame(1000 / fps));
perkj26752742016-10-24 01:21:16 -0700405 EXPECT_FRAME_WAIT(frame + i * fps, kVideoWidth, kVideoHeight, kTimeout);
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000406 }
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000407 }
jbaucheec21bd2016-03-20 06:15:43 -0700408 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000409 EXPECT_EQ(codec.id, GetPayloadType(p.get()));
410 }
411
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000412 // Test that stats work properly for a 1-1 call.
413 void GetStats() {
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000414 const int kDurationSec = 3;
415 const int kFps = 10;
416 SendReceiveManyAndGetStats(DefaultCodec(), kDurationSec, kFps);
417
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000418 cricket::VideoMediaInfo info;
pbos@webrtc.org058b1f12015-03-04 08:54:32 +0000419 EXPECT_TRUE(channel_->GetStats(&info));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000420
421 ASSERT_EQ(1U, info.senders.size());
422 // TODO(whyuan): bytes_sent and bytes_rcvd are different. Are both payload?
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000423 // For webrtc, bytes_sent does not include the RTP header length.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000424 EXPECT_GT(info.senders[0].bytes_sent, 0);
425 EXPECT_EQ(NumRtpPackets(), info.senders[0].packets_sent);
426 EXPECT_EQ(0.0, info.senders[0].fraction_lost);
hbosa65704b2016-11-14 02:28:16 -0800427 ASSERT_TRUE(info.senders[0].codec_payload_type);
428 EXPECT_EQ(DefaultCodec().id, *info.senders[0].codec_payload_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000429 EXPECT_EQ(0, info.senders[0].firs_rcvd);
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000430 EXPECT_EQ(0, info.senders[0].plis_rcvd);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000431 EXPECT_EQ(0, info.senders[0].nacks_rcvd);
perkj26752742016-10-24 01:21:16 -0700432 EXPECT_EQ(kVideoWidth, info.senders[0].send_frame_width);
433 EXPECT_EQ(kVideoHeight, info.senders[0].send_frame_height);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000434 EXPECT_GT(info.senders[0].framerate_input, 0);
435 EXPECT_GT(info.senders[0].framerate_sent, 0);
436
hbosa65704b2016-11-14 02:28:16 -0800437 EXPECT_EQ(1U, info.send_codecs.count(DefaultCodec().id));
438 EXPECT_EQ(DefaultCodec().ToCodecParameters(),
439 info.send_codecs[DefaultCodec().id]);
440
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000441 ASSERT_EQ(1U, info.receivers.size());
sergeyu@chromium.org5bc25c42013-12-05 00:24:06 +0000442 EXPECT_EQ(1U, info.senders[0].ssrcs().size());
443 EXPECT_EQ(1U, info.receivers[0].ssrcs().size());
444 EXPECT_EQ(info.senders[0].ssrcs()[0], info.receivers[0].ssrcs()[0]);
hbosa65704b2016-11-14 02:28:16 -0800445 ASSERT_TRUE(info.receivers[0].codec_payload_type);
446 EXPECT_EQ(DefaultCodec().id, *info.receivers[0].codec_payload_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000447 EXPECT_EQ(NumRtpBytes(), info.receivers[0].bytes_rcvd);
448 EXPECT_EQ(NumRtpPackets(), info.receivers[0].packets_rcvd);
449 EXPECT_EQ(0.0, info.receivers[0].fraction_lost);
450 EXPECT_EQ(0, info.receivers[0].packets_lost);
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000451 // TODO(asapersson): Not set for webrtc. Handle missing stats.
452 // EXPECT_EQ(0, info.receivers[0].packets_concealed);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000453 EXPECT_EQ(0, info.receivers[0].firs_sent);
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000454 EXPECT_EQ(0, info.receivers[0].plis_sent);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000455 EXPECT_EQ(0, info.receivers[0].nacks_sent);
perkj26752742016-10-24 01:21:16 -0700456 EXPECT_EQ(kVideoWidth, info.receivers[0].frame_width);
457 EXPECT_EQ(kVideoHeight, info.receivers[0].frame_height);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000458 EXPECT_GT(info.receivers[0].framerate_rcvd, 0);
459 EXPECT_GT(info.receivers[0].framerate_decoded, 0);
460 EXPECT_GT(info.receivers[0].framerate_output, 0);
hbosa65704b2016-11-14 02:28:16 -0800461
462 EXPECT_EQ(1U, info.receive_codecs.count(DefaultCodec().id));
463 EXPECT_EQ(DefaultCodec().ToCodecParameters(),
464 info.receive_codecs[DefaultCodec().id]);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000465 }
Stefan Holmer586b19b2015-09-18 11:14:31 +0200466
467 cricket::VideoSenderInfo GetSenderStats(size_t i) {
468 cricket::VideoMediaInfo info;
469 EXPECT_TRUE(channel_->GetStats(&info));
470 return info.senders[i];
471 }
472
473 cricket::VideoReceiverInfo GetReceiverStats(size_t i) {
474 cricket::VideoMediaInfo info;
475 EXPECT_TRUE(channel_->GetStats(&info));
476 return info.receivers[i];
477 }
478
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000479 // Test that stats work properly for a conf call with multiple recv streams.
480 void GetStatsMultipleRecvStreams() {
481 cricket::FakeVideoRenderer renderer1, renderer2;
482 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200483 cricket::VideoSendParameters parameters;
484 parameters.codecs.push_back(DefaultCodec());
nisse4b4dc862016-02-17 05:25:36 -0800485 parameters.conference_mode = true;
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200486 EXPECT_TRUE(channel_->SetSendParameters(parameters));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000487 EXPECT_TRUE(SetSend(true));
488 EXPECT_TRUE(channel_->AddRecvStream(
489 cricket::StreamParams::CreateLegacy(1)));
490 EXPECT_TRUE(channel_->AddRecvStream(
491 cricket::StreamParams::CreateLegacy(2)));
nisse08582ff2016-02-04 01:24:52 -0800492 EXPECT_TRUE(channel_->SetSink(1, &renderer1));
493 EXPECT_TRUE(channel_->SetSink(2, &renderer2));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000494 EXPECT_EQ(0, renderer1.num_rendered_frames());
495 EXPECT_EQ(0, renderer2.num_rendered_frames());
Peter Boström0c4e06b2015-10-07 12:23:21 +0200496 std::vector<uint32_t> ssrcs;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000497 ssrcs.push_back(1);
498 ssrcs.push_back(2);
499 network_interface_.SetConferenceMode(true, ssrcs);
500 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700501 EXPECT_FRAME_ON_RENDERER_WAIT(renderer1, 1, kVideoWidth, kVideoHeight,
502 kTimeout);
503 EXPECT_FRAME_ON_RENDERER_WAIT(renderer2, 1, kVideoWidth, kVideoHeight,
504 kTimeout);
Stefan Holmer586b19b2015-09-18 11:14:31 +0200505
506 EXPECT_TRUE(channel_->SetSend(false));
507
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000508 cricket::VideoMediaInfo info;
pbos@webrtc.org058b1f12015-03-04 08:54:32 +0000509 EXPECT_TRUE(channel_->GetStats(&info));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000510 ASSERT_EQ(1U, info.senders.size());
511 // TODO(whyuan): bytes_sent and bytes_rcvd are different. Are both payload?
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000512 // For webrtc, bytes_sent does not include the RTP header length.
Stefan Holmer586b19b2015-09-18 11:14:31 +0200513 EXPECT_GT(GetSenderStats(0).bytes_sent, 0);
514 EXPECT_EQ_WAIT(NumRtpPackets(), GetSenderStats(0).packets_sent, kTimeout);
perkj26752742016-10-24 01:21:16 -0700515 EXPECT_EQ(kVideoWidth, GetSenderStats(0).send_frame_width);
516 EXPECT_EQ(kVideoHeight, GetSenderStats(0).send_frame_height);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000517
518 ASSERT_EQ(2U, info.receivers.size());
519 for (size_t i = 0; i < info.receivers.size(); ++i) {
Stefan Holmer586b19b2015-09-18 11:14:31 +0200520 EXPECT_EQ(1U, GetReceiverStats(i).ssrcs().size());
521 EXPECT_EQ(i + 1, GetReceiverStats(i).ssrcs()[0]);
522 EXPECT_EQ_WAIT(NumRtpBytes(), GetReceiverStats(i).bytes_rcvd, kTimeout);
523 EXPECT_EQ_WAIT(NumRtpPackets(), GetReceiverStats(i).packets_rcvd,
524 kTimeout);
perkj26752742016-10-24 01:21:16 -0700525 EXPECT_EQ_WAIT(kVideoWidth, GetReceiverStats(i).frame_width, kTimeout);
526 EXPECT_EQ_WAIT(kVideoHeight, GetReceiverStats(i).frame_height, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000527 }
528 }
529 // Test that stats work properly for a conf call with multiple send streams.
530 void GetStatsMultipleSendStreams() {
531 // Normal setup; note that we set the SSRC explicitly to ensure that
532 // it will come first in the senders map.
533 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200534 cricket::VideoSendParameters parameters;
535 parameters.codecs.push_back(DefaultCodec());
nisse4b4dc862016-02-17 05:25:36 -0800536 parameters.conference_mode = true;
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200537 EXPECT_TRUE(channel_->SetSendParameters(parameters));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000538 EXPECT_TRUE(channel_->AddRecvStream(
buildbot@webrtc.org99f63082014-07-18 23:31:30 +0000539 cricket::StreamParams::CreateLegacy(kSsrc)));
nisse08582ff2016-02-04 01:24:52 -0800540 EXPECT_TRUE(channel_->SetSink(kSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000541 EXPECT_TRUE(SetSend(true));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000542 EXPECT_TRUE(SendFrame());
543 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
perkj26752742016-10-24 01:21:16 -0700544 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000545
546 // Add an additional capturer, and hook up a renderer to receive it.
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000547 cricket::FakeVideoRenderer renderer2;
kwiberg686a8ef2016-02-26 03:00:35 -0800548 std::unique_ptr<cricket::FakeVideoCapturer> capturer(
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000549 CreateFakeVideoCapturer());
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000550 const int kTestWidth = 160;
551 const int kTestHeight = 120;
552 cricket::VideoFormat format(kTestWidth, kTestHeight,
553 cricket::VideoFormat::FpsToInterval(5),
554 cricket::FOURCC_I420);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000555 EXPECT_EQ(cricket::CS_RUNNING, capturer->Start(format));
556 EXPECT_TRUE(channel_->AddSendStream(
557 cricket::StreamParams::CreateLegacy(5678)));
deadbeef5a4a75a2016-06-02 16:23:38 -0700558 EXPECT_TRUE(channel_->SetVideoSend(5678, true, nullptr, capturer.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000559 EXPECT_TRUE(channel_->AddRecvStream(
560 cricket::StreamParams::CreateLegacy(5678)));
nisse08582ff2016-02-04 01:24:52 -0800561 EXPECT_TRUE(channel_->SetSink(5678, &renderer2));
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000562 EXPECT_TRUE(capturer->CaptureCustomFrame(
563 kTestWidth, kTestHeight, cricket::FOURCC_I420));
564 EXPECT_FRAME_ON_RENDERER_WAIT(
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000565 renderer2, 1, kTestWidth, kTestHeight, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000566
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000567 // Get stats, and make sure they are correct for two senders. We wait until
568 // the number of expected packets have been sent to avoid races where we
569 // check stats before it has been updated.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000570 cricket::VideoMediaInfo info;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200571 for (uint32_t i = 0; i < kTimeout; ++i) {
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000572 rtc::Thread::Current()->ProcessMessages(1);
pbos@webrtc.org058b1f12015-03-04 08:54:32 +0000573 EXPECT_TRUE(channel_->GetStats(&info));
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000574 ASSERT_EQ(2U, info.senders.size());
575 if (info.senders[0].packets_sent + info.senders[1].packets_sent ==
576 NumRtpPackets()) {
577 // Stats have been updated for both sent frames, expectations can be
578 // checked now.
579 break;
580 }
581 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000582 EXPECT_EQ(NumRtpPackets(),
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000583 info.senders[0].packets_sent + info.senders[1].packets_sent)
584 << "Timed out while waiting for packet counts for all sent packets.";
sergeyu@chromium.org5bc25c42013-12-05 00:24:06 +0000585 EXPECT_EQ(1U, info.senders[0].ssrcs().size());
586 EXPECT_EQ(1234U, info.senders[0].ssrcs()[0]);
perkj26752742016-10-24 01:21:16 -0700587 EXPECT_EQ(kVideoWidth, info.senders[0].send_frame_width);
588 EXPECT_EQ(kVideoHeight, info.senders[0].send_frame_height);
sergeyu@chromium.org5bc25c42013-12-05 00:24:06 +0000589 EXPECT_EQ(1U, info.senders[1].ssrcs().size());
590 EXPECT_EQ(5678U, info.senders[1].ssrcs()[0]);
wu@webrtc.org987f2c92014-03-28 16:22:19 +0000591 EXPECT_EQ(kTestWidth, info.senders[1].send_frame_width);
592 EXPECT_EQ(kTestHeight, info.senders[1].send_frame_height);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000593 // The capturer must be unregistered here as it runs out of it's scope next.
deadbeef5a4a75a2016-06-02 16:23:38 -0700594 channel_->SetVideoSend(5678, true, nullptr, nullptr);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000595 }
596
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000597 // Test that we can set the bandwidth.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000598 void SetSendBandwidth() {
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200599 cricket::VideoSendParameters parameters;
600 parameters.codecs.push_back(DefaultCodec());
601 parameters.max_bandwidth_bps = -1; // <= 0 means unlimited.
602 EXPECT_TRUE(channel_->SetSendParameters(parameters));
603 parameters.max_bandwidth_bps = 128 * 1024;
604 EXPECT_TRUE(channel_->SetSendParameters(parameters));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000605 }
606 // Test that we can set the SSRC for the default send source.
607 void SetSendSsrc() {
608 EXPECT_TRUE(SetDefaultCodec());
609 EXPECT_TRUE(SetSend(true));
610 EXPECT_TRUE(SendFrame());
611 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
Peter Boström0c4e06b2015-10-07 12:23:21 +0200612 uint32_t ssrc = 0;
jbaucheec21bd2016-03-20 06:15:43 -0700613 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000614 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL);
615 EXPECT_EQ(kSsrc, ssrc);
Peter Boström718b6c72015-11-11 16:19:33 +0000616 // Packets are being paced out, so these can mismatch between the first and
617 // second call to NumRtpPackets until pending packets are paced out.
618 EXPECT_EQ_WAIT(NumRtpPackets(), NumRtpPackets(ssrc), kTimeout);
619 EXPECT_EQ_WAIT(NumRtpBytes(), NumRtpBytes(ssrc), kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000620 EXPECT_EQ(1, NumSentSsrcs());
621 EXPECT_EQ(0, NumRtpPackets(kSsrc - 1));
622 EXPECT_EQ(0, NumRtpBytes(kSsrc - 1));
623 }
624 // Test that we can set the SSRC even after codecs are set.
625 void SetSendSsrcAfterSetCodecs() {
626 // Remove stream added in Setup.
627 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc));
628 EXPECT_TRUE(SetDefaultCodec());
629 EXPECT_TRUE(channel_->AddSendStream(
630 cricket::StreamParams::CreateLegacy(999)));
deadbeef5a4a75a2016-06-02 16:23:38 -0700631 EXPECT_TRUE(
632 channel_->SetVideoSend(999u, true, nullptr, video_capturer_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000633 EXPECT_TRUE(SetSend(true));
634 EXPECT_TRUE(WaitAndSendFrame(0));
635 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
Peter Boström0c4e06b2015-10-07 12:23:21 +0200636 uint32_t ssrc = 0;
jbaucheec21bd2016-03-20 06:15:43 -0700637 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000638 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL);
639 EXPECT_EQ(999u, ssrc);
Peter Boström718b6c72015-11-11 16:19:33 +0000640 // Packets are being paced out, so these can mismatch between the first and
641 // second call to NumRtpPackets until pending packets are paced out.
642 EXPECT_EQ_WAIT(NumRtpPackets(), NumRtpPackets(ssrc), kTimeout);
643 EXPECT_EQ_WAIT(NumRtpBytes(), NumRtpBytes(ssrc), kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000644 EXPECT_EQ(1, NumSentSsrcs());
645 EXPECT_EQ(0, NumRtpPackets(kSsrc));
646 EXPECT_EQ(0, NumRtpBytes(kSsrc));
647 }
648 // Test that we can set the default video renderer before and after
649 // media is received.
nisse08582ff2016-02-04 01:24:52 -0800650 void SetSink() {
Peter Boström0c4e06b2015-10-07 12:23:21 +0200651 uint8_t data1[] = {
652 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000653
jbaucheec21bd2016-03-20 06:15:43 -0700654 rtc::CopyOnWriteBuffer packet1(data1, sizeof(data1));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000655 rtc::SetBE32(packet1.data() + 8, kSsrc);
nisse08582ff2016-02-04 01:24:52 -0800656 channel_->SetSink(kDefaultReceiveSsrc, NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000657 EXPECT_TRUE(SetDefaultCodec());
658 EXPECT_TRUE(SetSend(true));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000659 EXPECT_EQ(0, renderer_.num_rendered_frames());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000660 channel_->OnPacketReceived(&packet1, rtc::PacketTime());
nisse08582ff2016-02-04 01:24:52 -0800661 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000662 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700663 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000664 }
665
666 // Tests empty StreamParams is rejected.
667 void RejectEmptyStreamParams() {
668 // Remove the send stream that was added during Setup.
669 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc));
670
671 cricket::StreamParams empty;
672 EXPECT_FALSE(channel_->AddSendStream(empty));
673 EXPECT_TRUE(channel_->AddSendStream(
674 cricket::StreamParams::CreateLegacy(789u)));
675 }
676
677 // Tests setting up and configuring a send stream.
678 void AddRemoveSendStreams() {
679 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
680 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800681 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000682 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700683 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
Stefan Holmer586b19b2015-09-18 11:14:31 +0200684 EXPECT_GT(NumRtpPackets(), 0);
Peter Boström0c4e06b2015-10-07 12:23:21 +0200685 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000686 size_t last_packet = NumRtpPackets() - 1;
jbaucheec21bd2016-03-20 06:15:43 -0700687 std::unique_ptr<const rtc::CopyOnWriteBuffer>
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000688 p(GetRtpPacket(static_cast<int>(last_packet)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000689 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL);
690 EXPECT_EQ(kSsrc, ssrc);
691
692 // Remove the send stream that was added during Setup.
693 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc));
694 int rtp_packets = NumRtpPackets();
695
696 EXPECT_TRUE(channel_->AddSendStream(
697 cricket::StreamParams::CreateLegacy(789u)));
deadbeef5a4a75a2016-06-02 16:23:38 -0700698 EXPECT_TRUE(
699 channel_->SetVideoSend(789u, true, nullptr, video_capturer_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000700 EXPECT_EQ(rtp_packets, NumRtpPackets());
701 // Wait 30ms to guarantee the engine does not drop the frame.
702 EXPECT_TRUE(WaitAndSendFrame(30));
703 EXPECT_TRUE_WAIT(NumRtpPackets() > rtp_packets, kTimeout);
704
705 last_packet = NumRtpPackets() - 1;
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000706 p.reset(GetRtpPacket(static_cast<int>(last_packet)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000707 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL);
708 EXPECT_EQ(789u, ssrc);
709 }
710
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000711 // Tests the behavior of incoming streams in a conference scenario.
712 void SimulateConference() {
713 cricket::FakeVideoRenderer renderer1, renderer2;
714 EXPECT_TRUE(SetDefaultCodec());
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200715 cricket::VideoSendParameters parameters;
716 parameters.codecs.push_back(DefaultCodec());
nisse4b4dc862016-02-17 05:25:36 -0800717 parameters.conference_mode = true;
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200718 EXPECT_TRUE(channel_->SetSendParameters(parameters));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000719 EXPECT_TRUE(SetSend(true));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000720 EXPECT_TRUE(channel_->AddRecvStream(
721 cricket::StreamParams::CreateLegacy(1)));
722 EXPECT_TRUE(channel_->AddRecvStream(
723 cricket::StreamParams::CreateLegacy(2)));
nisse08582ff2016-02-04 01:24:52 -0800724 EXPECT_TRUE(channel_->SetSink(1, &renderer1));
725 EXPECT_TRUE(channel_->SetSink(2, &renderer2));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000726 EXPECT_EQ(0, renderer1.num_rendered_frames());
727 EXPECT_EQ(0, renderer2.num_rendered_frames());
Peter Boström0c4e06b2015-10-07 12:23:21 +0200728 std::vector<uint32_t> ssrcs;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000729 ssrcs.push_back(1);
730 ssrcs.push_back(2);
731 network_interface_.SetConferenceMode(true, ssrcs);
732 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700733 EXPECT_FRAME_ON_RENDERER_WAIT(renderer1, 1, kVideoWidth, kVideoHeight,
734 kTimeout);
735 EXPECT_FRAME_ON_RENDERER_WAIT(renderer2, 1, kVideoWidth, kVideoHeight,
736 kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000737
jbaucheec21bd2016-03-20 06:15:43 -0700738 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000739 EXPECT_EQ(DefaultCodec().id, GetPayloadType(p.get()));
perkj26752742016-10-24 01:21:16 -0700740 EXPECT_EQ(kVideoWidth, renderer1.width());
741 EXPECT_EQ(kVideoHeight, renderer1.height());
742 EXPECT_EQ(kVideoWidth, renderer2.width());
743 EXPECT_EQ(kVideoHeight, renderer2.height());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000744 EXPECT_TRUE(channel_->RemoveRecvStream(2));
745 EXPECT_TRUE(channel_->RemoveRecvStream(1));
746 }
747
748 // Tests that we can add and remove capturers and frames are sent out properly
749 void AddRemoveCapturer() {
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000750 cricket::VideoCodec codec = DefaultCodec();
perkj26752742016-10-24 01:21:16 -0700751 const int time_between_send_ms =
752 cricket::VideoFormat::FpsToInterval(kFramerate);
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000753 EXPECT_TRUE(SetOneCodec(codec));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000754 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800755 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000756 EXPECT_EQ(0, renderer_.num_rendered_frames());
757 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700758 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
kwiberg686a8ef2016-02-26 03:00:35 -0800759 std::unique_ptr<cricket::FakeVideoCapturer> capturer(
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000760 CreateFakeVideoCapturer());
Niels Möller60653ba2016-03-02 11:41:36 +0100761
762 // TODO(nisse): This testcase fails if we don't configure
763 // screencast. It's unclear why, I see nothing obvious in this
764 // test which is related to screencast logic.
765 cricket::VideoOptions video_options;
766 video_options.is_screencast = rtc::Optional<bool>(true);
deadbeef5a4a75a2016-06-02 16:23:38 -0700767 channel_->SetVideoSend(kSsrc, true, &video_options, nullptr);
Niels Möller60653ba2016-03-02 11:41:36 +0100768
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000769 cricket::VideoFormat format(480, 360,
770 cricket::VideoFormat::FpsToInterval(30),
771 cricket::FOURCC_I420);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000772 EXPECT_EQ(cricket::CS_RUNNING, capturer->Start(format));
773 // All capturers start generating frames with the same timestamp. ViE does
774 // not allow the same timestamp to be used. Capture one frame before
775 // associating the capturer with the channel.
776 EXPECT_TRUE(capturer->CaptureCustomFrame(format.width, format.height,
777 cricket::FOURCC_I420));
778
779 int captured_frames = 1;
780 for (int iterations = 0; iterations < 2; ++iterations) {
deadbeef5a4a75a2016-06-02 16:23:38 -0700781 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, capturer.get()));
perkj26752742016-10-24 01:21:16 -0700782 rtc::Thread::Current()->ProcessMessages(time_between_send_ms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000783 EXPECT_TRUE(capturer->CaptureCustomFrame(format.width, format.height,
784 cricket::FOURCC_I420));
785 ++captured_frames;
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000786 // Wait until frame of right size is captured.
787 EXPECT_TRUE_WAIT(renderer_.num_rendered_frames() >= captured_frames &&
788 format.width == renderer_.width() &&
henrike@webrtc.org1e09a712013-07-26 19:17:59 +0000789 format.height == renderer_.height() &&
790 !renderer_.black_frame(), kTimeout);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000791 EXPECT_GE(renderer_.num_rendered_frames(), captured_frames);
792 EXPECT_EQ(format.width, renderer_.width());
793 EXPECT_EQ(format.height, renderer_.height());
794 captured_frames = renderer_.num_rendered_frames() + 1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000795 EXPECT_FALSE(renderer_.black_frame());
deadbeef5a4a75a2016-06-02 16:23:38 -0700796 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, nullptr));
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000797 // Make sure a black frame is generated within the specified timeout.
pbos@webrtc.orgb4987bf2015-02-18 10:13:09 +0000798 // The black frame should be the resolution of the previous frame to
799 // prevent expensive encoder reconfigurations.
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000800 EXPECT_TRUE_WAIT(renderer_.num_rendered_frames() >= captured_frames &&
pbos@webrtc.orgb4987bf2015-02-18 10:13:09 +0000801 format.width == renderer_.width() &&
802 format.height == renderer_.height() &&
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000803 renderer_.black_frame(), kTimeout);
804 EXPECT_GE(renderer_.num_rendered_frames(), captured_frames);
pbos@webrtc.orgb4987bf2015-02-18 10:13:09 +0000805 EXPECT_EQ(format.width, renderer_.width());
806 EXPECT_EQ(format.height, renderer_.height());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000807 EXPECT_TRUE(renderer_.black_frame());
808
809 // The black frame has the same timestamp as the next frame since it's
810 // timestamp is set to the last frame's timestamp + interval. WebRTC will
811 // not render a frame with the same timestamp so capture another frame
812 // with the frame capturer to increment the next frame's timestamp.
813 EXPECT_TRUE(capturer->CaptureCustomFrame(format.width, format.height,
814 cricket::FOURCC_I420));
815 }
816 }
817
deadbeef5a4a75a2016-06-02 16:23:38 -0700818 // Tests that if SetVideoSend is called with a NULL capturer after the
819 // capturer was already removed, the application doesn't crash (and no black
820 // frame is sent).
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000821 void RemoveCapturerWithoutAdd() {
822 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
823 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800824 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000825 EXPECT_EQ(0, renderer_.num_rendered_frames());
826 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700827 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
pbos@webrtc.org776e6f22014-10-29 15:28:39 +0000828 // Wait for one frame so they don't get dropped because we send frames too
829 // tightly.
830 rtc::Thread::Current()->ProcessMessages(30);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000831 // Remove the capturer.
deadbeef5a4a75a2016-06-02 16:23:38 -0700832 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, nullptr));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000833 // Wait for one black frame for removing the capturer.
perkj26752742016-10-24 01:21:16 -0700834 EXPECT_FRAME_WAIT(2, kVideoWidth, kVideoHeight, kTimeout);
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000835
deadbeef5a4a75a2016-06-02 16:23:38 -0700836 // No capturer was added, so this SetVideoSend shouldn't do anything.
837 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, nullptr));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000838 rtc::Thread::Current()->ProcessMessages(300);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000839 // Verify no more frames were sent.
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000840 EXPECT_EQ(2, renderer_.num_rendered_frames());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000841 }
842
843 // Tests that we can add and remove capturer as unique sources.
844 void AddRemoveCapturerMultipleSources() {
845 // WebRTC implementation will drop frames if pushed to quickly. Wait the
846 // interval time to avoid that.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000847 // WebRTC implementation will drop frames if pushed to quickly. Wait the
848 // interval time to avoid that.
849 // Set up the stream associated with the engine.
850 EXPECT_TRUE(channel_->AddRecvStream(
851 cricket::StreamParams::CreateLegacy(kSsrc)));
nisse08582ff2016-02-04 01:24:52 -0800852 EXPECT_TRUE(channel_->SetSink(kSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000853 cricket::VideoFormat capture_format; // default format
perkj26752742016-10-24 01:21:16 -0700854 capture_format.interval = cricket::VideoFormat::FpsToInterval(kFramerate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000855 // Set up additional stream 1.
856 cricket::FakeVideoRenderer renderer1;
nisse08582ff2016-02-04 01:24:52 -0800857 EXPECT_FALSE(channel_->SetSink(1, &renderer1));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000858 EXPECT_TRUE(channel_->AddRecvStream(
859 cricket::StreamParams::CreateLegacy(1)));
nisse08582ff2016-02-04 01:24:52 -0800860 EXPECT_TRUE(channel_->SetSink(1, &renderer1));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000861 EXPECT_TRUE(channel_->AddSendStream(
862 cricket::StreamParams::CreateLegacy(1)));
kwiberg686a8ef2016-02-26 03:00:35 -0800863 std::unique_ptr<cricket::FakeVideoCapturer> capturer1(
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000864 CreateFakeVideoCapturer());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000865 EXPECT_EQ(cricket::CS_RUNNING, capturer1->Start(capture_format));
866 // Set up additional stream 2.
867 cricket::FakeVideoRenderer renderer2;
nisse08582ff2016-02-04 01:24:52 -0800868 EXPECT_FALSE(channel_->SetSink(2, &renderer2));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000869 EXPECT_TRUE(channel_->AddRecvStream(
870 cricket::StreamParams::CreateLegacy(2)));
nisse08582ff2016-02-04 01:24:52 -0800871 EXPECT_TRUE(channel_->SetSink(2, &renderer2));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000872 EXPECT_TRUE(channel_->AddSendStream(
873 cricket::StreamParams::CreateLegacy(2)));
kwiberg686a8ef2016-02-26 03:00:35 -0800874 std::unique_ptr<cricket::FakeVideoCapturer> capturer2(
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000875 CreateFakeVideoCapturer());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000876 EXPECT_EQ(cricket::CS_RUNNING, capturer2->Start(capture_format));
877 // State for all the streams.
878 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
deadbeef5a4a75a2016-06-02 16:23:38 -0700879 // A limitation in the lmi implementation requires that SetVideoSend() is
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000880 // called after SetOneCodec().
881 // TODO(hellner): this seems like an unnecessary constraint, fix it.
deadbeef5a4a75a2016-06-02 16:23:38 -0700882 EXPECT_TRUE(channel_->SetVideoSend(1, true, nullptr, capturer1.get()));
883 EXPECT_TRUE(channel_->SetVideoSend(2, true, nullptr, capturer2.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000884 EXPECT_TRUE(SetSend(true));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000885 // Test capturer associated with engine.
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000886 const int kTestWidth = 160;
887 const int kTestHeight = 120;
888 EXPECT_TRUE(capturer1->CaptureCustomFrame(
889 kTestWidth, kTestHeight, cricket::FOURCC_I420));
890 EXPECT_FRAME_ON_RENDERER_WAIT(
891 renderer1, 1, kTestWidth, kTestHeight, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000892 // Capture a frame with additional capturer2, frames should be received
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000893 EXPECT_TRUE(capturer2->CaptureCustomFrame(
894 kTestWidth, kTestHeight, cricket::FOURCC_I420));
895 EXPECT_FRAME_ON_RENDERER_WAIT(
896 renderer2, 1, kTestWidth, kTestHeight, kTimeout);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000897 // Successfully remove the capturer.
deadbeef5a4a75a2016-06-02 16:23:38 -0700898 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, nullptr));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000899 // The capturers must be unregistered here as it runs out of it's scope
900 // next.
deadbeef5a4a75a2016-06-02 16:23:38 -0700901 EXPECT_TRUE(channel_->SetVideoSend(1, true, nullptr, nullptr));
902 EXPECT_TRUE(channel_->SetVideoSend(2, true, nullptr, nullptr));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000903 }
904
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000905 // Test that multiple send streams can be created and deleted properly.
906 void MultipleSendStreams() {
907 // Remove stream added in Setup. I.e. remove stream corresponding to default
908 // channel.
909 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc));
910 const unsigned int kSsrcsSize = sizeof(kSsrcs4)/sizeof(kSsrcs4[0]);
911 for (unsigned int i = 0; i < kSsrcsSize; ++i) {
912 EXPECT_TRUE(channel_->AddSendStream(
913 cricket::StreamParams::CreateLegacy(kSsrcs4[i])));
914 }
915 // Delete one of the non default channel streams, let the destructor delete
916 // the remaining ones.
917 EXPECT_TRUE(channel_->RemoveSendStream(kSsrcs4[kSsrcsSize - 1]));
918 // Stream should already be deleted.
919 EXPECT_FALSE(channel_->RemoveSendStream(kSsrcs4[kSsrcsSize - 1]));
920 }
921
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000922 // Two streams one channel tests.
923
924 // Tests that we can send and receive frames.
925 void TwoStreamsSendAndReceive(const cricket::VideoCodec& codec) {
926 SetUpSecondStream();
927 // Test sending and receiving on first stream.
Peter Boströmd1f584b2016-04-20 16:31:53 +0200928 SendAndReceive(codec);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000929 // Test sending and receiving on second stream.
930 EXPECT_EQ_WAIT(1, renderer2_.num_rendered_frames(), kTimeout);
Stefan Holmer586b19b2015-09-18 11:14:31 +0200931 EXPECT_GT(NumRtpPackets(), 0);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000932 EXPECT_EQ(1, renderer2_.num_rendered_frames());
933 }
934
skvlad11a9cbf2016-10-07 11:53:05 -0700935 webrtc::RtcEventLogNullImpl event_log_;
kwiberg686a8ef2016-02-26 03:00:35 -0800936 const std::unique_ptr<webrtc::Call> call_;
nisse2ded9b12016-04-08 02:23:55 -0700937 E engine_;
kwiberg686a8ef2016-02-26 03:00:35 -0800938 std::unique_ptr<cricket::FakeVideoCapturer> video_capturer_;
939 std::unique_ptr<cricket::FakeVideoCapturer> video_capturer_2_;
940 std::unique_ptr<C> channel_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000941 cricket::FakeNetworkInterface network_interface_;
942 cricket::FakeVideoRenderer renderer_;
943 cricket::VideoMediaChannel::Error media_error_;
944
945 // Used by test cases where 2 streams are run on the same channel.
946 cricket::FakeVideoRenderer renderer2_;
947};
948
kjellandera96e2d72016-02-04 23:52:28 -0800949#endif // WEBRTC_MEDIA_BASE_VIDEOENGINE_UNITTEST_H_ NOLINT