blob: 1cbb2453aaabe9bc7a5cdad36d17af8ce5039679 [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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#ifndef MEDIA_BASE_VIDEOENGINE_UNITTEST_H_ // NOLINT
12#define 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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020018#include "call/call.h"
19#include "logging/rtc_event_log/rtc_event_log.h"
20#include "media/base/fakenetworkinterface.h"
21#include "media/base/fakevideocapturer.h"
22#include "media/base/fakevideorenderer.h"
23#include "media/base/mediachannel.h"
24#include "media/base/streamparams.h"
25#include "media/engine/fakewebrtccall.h"
26#include "rtc_base/bytebuffer.h"
27#include "rtc_base/gunit.h"
28#include "rtc_base/timeutils.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000029
Magnus Jedvert02e7a192017-09-23 17:21:32 +020030namespace cricket {
31class WebRtcVideoEncoderFactory;
32class WebRtcVideoDecoderFactory;
33} // namespace cricket
34
henrike@webrtc.org28e20752013-07-10 00:45:36 +000035#define EXPECT_FRAME_WAIT(c, w, h, t) \
36 EXPECT_EQ_WAIT((c), renderer_.num_rendered_frames(), (t)); \
37 EXPECT_EQ((w), renderer_.width()); \
38 EXPECT_EQ((h), renderer_.height()); \
39 EXPECT_EQ(0, renderer_.errors()); \
40
41#define EXPECT_FRAME_ON_RENDERER_WAIT(r, c, w, h, t) \
42 EXPECT_EQ_WAIT((c), (r).num_rendered_frames(), (t)); \
43 EXPECT_EQ((w), (r).width()); \
44 EXPECT_EQ((h), (r).height()); \
45 EXPECT_EQ(0, (r).errors()); \
46
wu@webrtc.org9caf2762013-12-11 18:25:07 +000047#define EXPECT_GT_FRAME_ON_RENDERER_WAIT(r, c, w, h, t) \
48 EXPECT_TRUE_WAIT((r).num_rendered_frames() >= (c) && \
49 (w) == (r).width() && \
50 (h) == (r).height(), (t)); \
Peter Boström0c4e06b2015-10-07 12:23:21 +020051 EXPECT_EQ(0, (r).errors());
wu@webrtc.org9caf2762013-12-11 18:25:07 +000052
Peter Boström0c4e06b2015-10-07 12:23:21 +020053static const uint32_t kTimeout = 5000U;
54static const uint32_t kDefaultReceiveSsrc = 0;
55static const uint32_t kSsrc = 1234u;
56static const uint32_t kRtxSsrc = 4321u;
57static const uint32_t kSsrcs4[] = {1, 2, 3, 4};
perkj26752742016-10-24 01:21:16 -070058static const int kVideoWidth = 640;
59static const int kVideoHeight = 360;
60static const int kFramerate = 30;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000061
62inline bool IsEqualCodec(const cricket::VideoCodec& a,
63 const cricket::VideoCodec& b) {
perkj26752742016-10-24 01:21:16 -070064 return a.id == b.id && a.name == b.name;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000065}
66
henrike@webrtc.org28e20752013-07-10 00:45:36 +000067template<class E, class C>
68class VideoMediaChannelTest : public testing::Test,
69 public sigslot::has_slots<> {
70 protected:
Fredrik Solenberg709ed672015-09-15 12:26:33 +020071 VideoMediaChannelTest<E, C>()
magjed2475ae22017-09-12 04:42:15 -070072 : call_(webrtc::Call::Create(webrtc::Call::Config(&event_log_))),
Magnus Jedvert02e7a192017-09-23 17:21:32 +020073 engine_(std::unique_ptr<cricket::WebRtcVideoEncoderFactory>(),
74 std::unique_ptr<cricket::WebRtcVideoDecoderFactory>()) {}
Fredrik Solenberg709ed672015-09-15 12:26:33 +020075
henrike@webrtc.org28e20752013-07-10 00:45:36 +000076 virtual cricket::VideoCodec DefaultCodec() = 0;
77
78 virtual cricket::StreamParams DefaultSendStreamParams() {
79 return cricket::StreamParams::CreateLegacy(kSsrc);
80 }
81
82 virtual void SetUp() {
nisse2013e292017-01-17 05:45:40 -080083 cricket::MediaConfig media_config;
84 // Disabling cpu overuse detection actually disables quality scaling too; it
85 // implies DegradationPreference kMaintainResolution. Automatic scaling
86 // needs to be disabled, otherwise, tests which check the size of received
87 // frames become flaky.
88 media_config.video.enable_cpu_overuse_detection = false;
89 channel_.reset(engine_.CreateChannel(call_.get(), media_config,
nisse51542be2016-02-12 02:27:06 -080090 cricket::VideoOptions()));
Sergey Ulanove2b15012016-11-22 16:08:30 -080091 channel_->OnReadyToSend(true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000092 EXPECT_TRUE(channel_.get() != NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000093 network_interface_.SetDestination(channel_.get());
94 channel_->SetInterface(&network_interface_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000095 media_error_ = cricket::VideoMediaChannel::ERROR_NONE;
Fredrik Solenbergb071a192015-09-17 16:42:56 +020096 cricket::VideoRecvParameters parameters;
97 parameters.codecs = engine_.codecs();
98 channel_->SetRecvParameters(parameters);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000099 EXPECT_TRUE(channel_->AddSendStream(DefaultSendStreamParams()));
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000100 video_capturer_.reset(CreateFakeVideoCapturer());
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000101 cricket::VideoFormat format(640, 480,
perkj26752742016-10-24 01:21:16 -0700102 cricket::VideoFormat::FpsToInterval(kFramerate),
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000103 cricket::FOURCC_I420);
104 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_->Start(format));
deadbeef5a4a75a2016-06-02 16:23:38 -0700105 EXPECT_TRUE(
106 channel_->SetVideoSend(kSsrc, true, nullptr, video_capturer_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000107 }
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000108
109 virtual cricket::FakeVideoCapturer* CreateFakeVideoCapturer() {
110 return new cricket::FakeVideoCapturer();
111 }
112
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000113 // Utility method to setup an additional stream to send and receive video.
114 // Used to test send and recv between two streams.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000115 void SetUpSecondStream() {
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000116 SetUpSecondStreamWithNoRecv();
117 // Setup recv for second stream.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000118 EXPECT_TRUE(channel_->AddRecvStream(
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000119 cricket::StreamParams::CreateLegacy(kSsrc + 2)));
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000120 // Make the second renderer available for use by a new stream.
nisse08582ff2016-02-04 01:24:52 -0800121 EXPECT_TRUE(channel_->SetSink(kSsrc + 2, &renderer2_));
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000122 }
123 // Setup an additional stream just to send video. Defer add recv stream.
124 // This is required if you want to test unsignalled recv of video rtp packets.
125 void SetUpSecondStreamWithNoRecv() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000126 // SetUp() already added kSsrc make sure duplicate SSRCs cant be added.
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000127 EXPECT_TRUE(channel_->AddRecvStream(
128 cricket::StreamParams::CreateLegacy(kSsrc)));
nisse08582ff2016-02-04 01:24:52 -0800129 EXPECT_TRUE(channel_->SetSink(kSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000130 EXPECT_FALSE(channel_->AddSendStream(
131 cricket::StreamParams::CreateLegacy(kSsrc)));
132 EXPECT_TRUE(channel_->AddSendStream(
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000133 cricket::StreamParams::CreateLegacy(kSsrc + 2)));
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000134 // We dont add recv for the second stream.
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000135
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000136 // Setup the receive and renderer for second stream after send.
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000137 video_capturer_2_.reset(CreateFakeVideoCapturer());
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000138 cricket::VideoFormat format(640, 480,
perkj26752742016-10-24 01:21:16 -0700139 cricket::VideoFormat::FpsToInterval(kFramerate),
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000140 cricket::FOURCC_I420);
141 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_2_->Start(format));
142
deadbeef5a4a75a2016-06-02 16:23:38 -0700143 EXPECT_TRUE(channel_->SetVideoSend(kSsrc + 2, true, nullptr,
144 video_capturer_2_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000145 }
146 virtual void TearDown() {
147 channel_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000148 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000149 bool SetDefaultCodec() {
150 return SetOneCodec(DefaultCodec());
151 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000152
perkj26752742016-10-24 01:21:16 -0700153 bool SetOneCodec(int pt, const char* name) {
154 return SetOneCodec(cricket::VideoCodec(pt, name));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000155 }
156 bool SetOneCodec(const cricket::VideoCodec& codec) {
perkj26752742016-10-24 01:21:16 -0700157 cricket::VideoFormat capture_format(
158 kVideoWidth, kVideoHeight,
159 cricket::VideoFormat::FpsToInterval(kFramerate), cricket::FOURCC_I420);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000160
161 if (video_capturer_) {
162 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_->Start(capture_format));
163 }
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000164 if (video_capturer_2_) {
165 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_2_->Start(capture_format));
166 }
167
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000168 bool sending = channel_->sending();
169 bool success = SetSend(false);
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200170 if (success) {
171 cricket::VideoSendParameters parameters;
172 parameters.codecs.push_back(codec);
173 success = channel_->SetSendParameters(parameters);
174 }
175 if (success) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000176 success = SetSend(sending);
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200177 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000178 return success;
179 }
180 bool SetSend(bool send) {
181 return channel_->SetSend(send);
182 }
183 int DrainOutgoingPackets() {
184 int packets = 0;
185 do {
186 packets = NumRtpPackets();
187 // 100 ms should be long enough.
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000188 rtc::Thread::Current()->ProcessMessages(100);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000189 } while (NumRtpPackets() > packets);
190 return NumRtpPackets();
191 }
192 bool SendFrame() {
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000193 if (video_capturer_2_) {
194 video_capturer_2_->CaptureFrame();
195 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000196 return video_capturer_.get() &&
197 video_capturer_->CaptureFrame();
198 }
199 bool WaitAndSendFrame(int wait_ms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000200 bool ret = rtc::Thread::Current()->ProcessMessages(wait_ms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000201 ret &= SendFrame();
202 return ret;
203 }
204 // Sends frames and waits for the decoder to be fully initialized.
205 // Returns the number of frames that were sent.
206 int WaitForDecoder() {
207#if defined(HAVE_OPENMAX)
208 // Send enough frames for the OpenMAX decoder to continue processing, and
209 // return the number of frames sent.
210 // Send frames for a full kTimeout's worth of 15fps video.
211 int frame_count = 0;
212 while (frame_count < static_cast<int>(kTimeout) / 66) {
213 EXPECT_TRUE(WaitAndSendFrame(66));
214 ++frame_count;
215 }
216 return frame_count;
217#else
218 return 0;
219#endif
220 }
221 bool SendCustomVideoFrame(int w, int h) {
222 if (!video_capturer_.get()) return false;
223 return video_capturer_->CaptureCustomFrame(w, h, cricket::FOURCC_I420);
224 }
225 int NumRtpBytes() {
226 return network_interface_.NumRtpBytes();
227 }
Peter Boström0c4e06b2015-10-07 12:23:21 +0200228 int NumRtpBytes(uint32_t ssrc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000229 return network_interface_.NumRtpBytes(ssrc);
230 }
231 int NumRtpPackets() {
232 return network_interface_.NumRtpPackets();
233 }
Peter Boström0c4e06b2015-10-07 12:23:21 +0200234 int NumRtpPackets(uint32_t ssrc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000235 return network_interface_.NumRtpPackets(ssrc);
236 }
237 int NumSentSsrcs() {
238 return network_interface_.NumSentSsrcs();
239 }
jbaucheec21bd2016-03-20 06:15:43 -0700240 const rtc::CopyOnWriteBuffer* GetRtpPacket(int index) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000241 return network_interface_.GetRtpPacket(index);
242 }
243 int NumRtcpPackets() {
244 return network_interface_.NumRtcpPackets();
245 }
jbaucheec21bd2016-03-20 06:15:43 -0700246 const rtc::CopyOnWriteBuffer* GetRtcpPacket(int index) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000247 return network_interface_.GetRtcpPacket(index);
248 }
jbaucheec21bd2016-03-20 06:15:43 -0700249 static int GetPayloadType(const rtc::CopyOnWriteBuffer* p) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000250 int pt = -1;
251 ParseRtpPacket(p, NULL, &pt, NULL, NULL, NULL, NULL);
252 return pt;
253 }
jbaucheec21bd2016-03-20 06:15:43 -0700254 static bool ParseRtpPacket(const rtc::CopyOnWriteBuffer* p,
Peter Boström0c4e06b2015-10-07 12:23:21 +0200255 bool* x,
256 int* pt,
257 int* seqnum,
258 uint32_t* tstamp,
259 uint32_t* ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000260 std::string* payload) {
jbauchf1f87202016-03-30 06:43:37 -0700261 rtc::ByteBufferReader buf(p->data<char>(), p->size());
Peter Boström0c4e06b2015-10-07 12:23:21 +0200262 uint8_t u08 = 0;
263 uint16_t u16 = 0;
264 uint32_t u32 = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000265
266 // Read X and CC fields.
267 if (!buf.ReadUInt8(&u08)) return false;
268 bool extension = ((u08 & 0x10) != 0);
Peter Boström0c4e06b2015-10-07 12:23:21 +0200269 uint8_t cc = (u08 & 0x0F);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000270 if (x) *x = extension;
271
272 // Read PT field.
273 if (!buf.ReadUInt8(&u08)) return false;
274 if (pt) *pt = (u08 & 0x7F);
275
276 // Read Sequence Number field.
277 if (!buf.ReadUInt16(&u16)) return false;
278 if (seqnum) *seqnum = u16;
279
280 // Read Timestamp field.
281 if (!buf.ReadUInt32(&u32)) return false;
282 if (tstamp) *tstamp = u32;
283
284 // Read SSRC field.
285 if (!buf.ReadUInt32(&u32)) return false;
286 if (ssrc) *ssrc = u32;
287
288 // Skip CSRCs.
Peter Boström0c4e06b2015-10-07 12:23:21 +0200289 for (uint8_t i = 0; i < cc; ++i) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000290 if (!buf.ReadUInt32(&u32)) return false;
291 }
292
293 // Skip extension header.
294 if (extension) {
295 // Read Profile-specific extension header ID
296 if (!buf.ReadUInt16(&u16)) return false;
297
298 // Read Extension header length
299 if (!buf.ReadUInt16(&u16)) return false;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200300 uint16_t ext_header_len = u16;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000301
302 // Read Extension header
Peter Boström0c4e06b2015-10-07 12:23:21 +0200303 for (uint16_t i = 0; i < ext_header_len; ++i) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000304 if (!buf.ReadUInt32(&u32)) return false;
305 }
306 }
307
308 if (payload) {
309 return buf.ReadString(payload, buf.Length());
310 }
311 return true;
312 }
313
314 // Parse all RTCP packet, from start_index to stop_index, and count how many
315 // FIR (PT=206 and FMT=4 according to RFC 5104). If successful, set the count
316 // and return true.
317 bool CountRtcpFir(int start_index, int stop_index, int* fir_count) {
318 int count = 0;
319 for (int i = start_index; i < stop_index; ++i) {
jbaucheec21bd2016-03-20 06:15:43 -0700320 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtcpPacket(i));
jbauchf1f87202016-03-30 06:43:37 -0700321 rtc::ByteBufferReader buf(p->data<char>(), p->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000322 size_t total_len = 0;
323 // The packet may be a compound RTCP packet.
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000324 while (total_len < p->size()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000325 // Read FMT, type and length.
Peter Boström0c4e06b2015-10-07 12:23:21 +0200326 uint8_t fmt = 0;
327 uint8_t type = 0;
328 uint16_t length = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000329 if (!buf.ReadUInt8(&fmt)) return false;
330 fmt &= 0x1F;
331 if (!buf.ReadUInt8(&type)) return false;
332 if (!buf.ReadUInt16(&length)) return false;
333 buf.Consume(length * 4); // Skip RTCP data.
334 total_len += (length + 1) * 4;
335 if ((192 == type) || ((206 == type) && (4 == fmt))) {
336 ++count;
337 }
338 }
339 }
340
341 if (fir_count) {
342 *fir_count = count;
343 }
344 return true;
345 }
346
Peter Boström0c4e06b2015-10-07 12:23:21 +0200347 void OnVideoChannelError(uint32_t ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000348 cricket::VideoMediaChannel::Error error) {
349 media_error_ = error;
350 }
351
352 // Test that SetSend works.
353 void SetSend() {
354 EXPECT_FALSE(channel_->sending());
deadbeef5a4a75a2016-06-02 16:23:38 -0700355 EXPECT_TRUE(
356 channel_->SetVideoSend(kSsrc, true, nullptr, video_capturer_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000357 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
358 EXPECT_FALSE(channel_->sending());
359 EXPECT_TRUE(SetSend(true));
360 EXPECT_TRUE(channel_->sending());
361 EXPECT_TRUE(SendFrame());
362 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
363 EXPECT_TRUE(SetSend(false));
364 EXPECT_FALSE(channel_->sending());
365 }
366 // Test that SetSend fails without codecs being set.
367 void SetSendWithoutCodecs() {
368 EXPECT_FALSE(channel_->sending());
369 EXPECT_FALSE(SetSend(true));
370 EXPECT_FALSE(channel_->sending());
371 }
372 // Test that we properly set the send and recv buffer sizes by the time
373 // SetSend is called.
374 void SetSendSetsTransportBufferSizes() {
375 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
376 EXPECT_TRUE(SetSend(true));
buildbot@webrtc.orgae694ef2014-10-28 17:37:17 +0000377 EXPECT_EQ(64 * 1024, network_interface_.sendbuf_size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000378 EXPECT_EQ(64 * 1024, network_interface_.recvbuf_size());
379 }
380 // Tests that we can send frames and the right payload type is used.
381 void Send(const cricket::VideoCodec& codec) {
382 EXPECT_TRUE(SetOneCodec(codec));
383 EXPECT_TRUE(SetSend(true));
384 EXPECT_TRUE(SendFrame());
385 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
jbaucheec21bd2016-03-20 06:15:43 -0700386 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000387 EXPECT_EQ(codec.id, GetPayloadType(p.get()));
388 }
389 // Tests that we can send and receive frames.
390 void SendAndReceive(const cricket::VideoCodec& codec) {
391 EXPECT_TRUE(SetOneCodec(codec));
392 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800393 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000394 EXPECT_EQ(0, renderer_.num_rendered_frames());
395 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700396 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
jbaucheec21bd2016-03-20 06:15:43 -0700397 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000398 EXPECT_EQ(codec.id, GetPayloadType(p.get()));
399 }
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000400 void SendReceiveManyAndGetStats(const cricket::VideoCodec& codec,
401 int duration_sec, int fps) {
402 EXPECT_TRUE(SetOneCodec(codec));
403 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800404 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000405 EXPECT_EQ(0, renderer_.num_rendered_frames());
406 for (int i = 0; i < duration_sec; ++i) {
407 for (int frame = 1; frame <= fps; ++frame) {
408 EXPECT_TRUE(WaitAndSendFrame(1000 / fps));
perkj26752742016-10-24 01:21:16 -0700409 EXPECT_FRAME_WAIT(frame + i * fps, kVideoWidth, kVideoHeight, kTimeout);
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000410 }
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000411 }
jbaucheec21bd2016-03-20 06:15:43 -0700412 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000413 EXPECT_EQ(codec.id, GetPayloadType(p.get()));
414 }
415
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000416 // Test that stats work properly for a 1-1 call.
417 void GetStats() {
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000418 const int kDurationSec = 3;
419 const int kFps = 10;
420 SendReceiveManyAndGetStats(DefaultCodec(), kDurationSec, kFps);
421
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000422 cricket::VideoMediaInfo info;
pbos@webrtc.org058b1f12015-03-04 08:54:32 +0000423 EXPECT_TRUE(channel_->GetStats(&info));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000424
425 ASSERT_EQ(1U, info.senders.size());
426 // TODO(whyuan): bytes_sent and bytes_rcvd are different. Are both payload?
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000427 // For webrtc, bytes_sent does not include the RTP header length.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000428 EXPECT_GT(info.senders[0].bytes_sent, 0);
429 EXPECT_EQ(NumRtpPackets(), info.senders[0].packets_sent);
430 EXPECT_EQ(0.0, info.senders[0].fraction_lost);
hbosa65704b2016-11-14 02:28:16 -0800431 ASSERT_TRUE(info.senders[0].codec_payload_type);
432 EXPECT_EQ(DefaultCodec().id, *info.senders[0].codec_payload_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000433 EXPECT_EQ(0, info.senders[0].firs_rcvd);
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000434 EXPECT_EQ(0, info.senders[0].plis_rcvd);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000435 EXPECT_EQ(0, info.senders[0].nacks_rcvd);
perkj26752742016-10-24 01:21:16 -0700436 EXPECT_EQ(kVideoWidth, info.senders[0].send_frame_width);
437 EXPECT_EQ(kVideoHeight, info.senders[0].send_frame_height);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000438 EXPECT_GT(info.senders[0].framerate_input, 0);
439 EXPECT_GT(info.senders[0].framerate_sent, 0);
440
hbosa65704b2016-11-14 02:28:16 -0800441 EXPECT_EQ(1U, info.send_codecs.count(DefaultCodec().id));
442 EXPECT_EQ(DefaultCodec().ToCodecParameters(),
443 info.send_codecs[DefaultCodec().id]);
444
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000445 ASSERT_EQ(1U, info.receivers.size());
sergeyu@chromium.org5bc25c42013-12-05 00:24:06 +0000446 EXPECT_EQ(1U, info.senders[0].ssrcs().size());
447 EXPECT_EQ(1U, info.receivers[0].ssrcs().size());
448 EXPECT_EQ(info.senders[0].ssrcs()[0], info.receivers[0].ssrcs()[0]);
hbosa65704b2016-11-14 02:28:16 -0800449 ASSERT_TRUE(info.receivers[0].codec_payload_type);
450 EXPECT_EQ(DefaultCodec().id, *info.receivers[0].codec_payload_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000451 EXPECT_EQ(NumRtpBytes(), info.receivers[0].bytes_rcvd);
452 EXPECT_EQ(NumRtpPackets(), info.receivers[0].packets_rcvd);
453 EXPECT_EQ(0.0, info.receivers[0].fraction_lost);
454 EXPECT_EQ(0, info.receivers[0].packets_lost);
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000455 // TODO(asapersson): Not set for webrtc. Handle missing stats.
456 // EXPECT_EQ(0, info.receivers[0].packets_concealed);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000457 EXPECT_EQ(0, info.receivers[0].firs_sent);
henrike@webrtc.org704bf9e2014-02-27 17:52:04 +0000458 EXPECT_EQ(0, info.receivers[0].plis_sent);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000459 EXPECT_EQ(0, info.receivers[0].nacks_sent);
perkj0b2d3e22017-01-17 05:56:53 -0800460 EXPECT_EQ(kVideoWidth, info.receivers[0].frame_width);
461 EXPECT_EQ(kVideoHeight, info.receivers[0].frame_height);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000462 EXPECT_GT(info.receivers[0].framerate_rcvd, 0);
463 EXPECT_GT(info.receivers[0].framerate_decoded, 0);
464 EXPECT_GT(info.receivers[0].framerate_output, 0);
hbosa65704b2016-11-14 02:28:16 -0800465
466 EXPECT_EQ(1U, info.receive_codecs.count(DefaultCodec().id));
467 EXPECT_EQ(DefaultCodec().ToCodecParameters(),
468 info.receive_codecs[DefaultCodec().id]);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000469 }
Stefan Holmer586b19b2015-09-18 11:14:31 +0200470
471 cricket::VideoSenderInfo GetSenderStats(size_t i) {
472 cricket::VideoMediaInfo info;
473 EXPECT_TRUE(channel_->GetStats(&info));
474 return info.senders[i];
475 }
476
477 cricket::VideoReceiverInfo GetReceiverStats(size_t i) {
478 cricket::VideoMediaInfo info;
479 EXPECT_TRUE(channel_->GetStats(&info));
480 return info.receivers[i];
481 }
482
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000483 // Test that stats work properly for a conf call with multiple recv streams.
484 void GetStatsMultipleRecvStreams() {
485 cricket::FakeVideoRenderer renderer1, renderer2;
486 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200487 cricket::VideoSendParameters parameters;
488 parameters.codecs.push_back(DefaultCodec());
nisse4b4dc862016-02-17 05:25:36 -0800489 parameters.conference_mode = true;
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200490 EXPECT_TRUE(channel_->SetSendParameters(parameters));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000491 EXPECT_TRUE(SetSend(true));
492 EXPECT_TRUE(channel_->AddRecvStream(
493 cricket::StreamParams::CreateLegacy(1)));
494 EXPECT_TRUE(channel_->AddRecvStream(
495 cricket::StreamParams::CreateLegacy(2)));
nisse08582ff2016-02-04 01:24:52 -0800496 EXPECT_TRUE(channel_->SetSink(1, &renderer1));
497 EXPECT_TRUE(channel_->SetSink(2, &renderer2));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000498 EXPECT_EQ(0, renderer1.num_rendered_frames());
499 EXPECT_EQ(0, renderer2.num_rendered_frames());
Peter Boström0c4e06b2015-10-07 12:23:21 +0200500 std::vector<uint32_t> ssrcs;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000501 ssrcs.push_back(1);
502 ssrcs.push_back(2);
503 network_interface_.SetConferenceMode(true, ssrcs);
504 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700505 EXPECT_FRAME_ON_RENDERER_WAIT(renderer1, 1, kVideoWidth, kVideoHeight,
506 kTimeout);
507 EXPECT_FRAME_ON_RENDERER_WAIT(renderer2, 1, kVideoWidth, kVideoHeight,
508 kTimeout);
Stefan Holmer586b19b2015-09-18 11:14:31 +0200509
510 EXPECT_TRUE(channel_->SetSend(false));
511
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000512 cricket::VideoMediaInfo info;
pbos@webrtc.org058b1f12015-03-04 08:54:32 +0000513 EXPECT_TRUE(channel_->GetStats(&info));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000514 ASSERT_EQ(1U, info.senders.size());
515 // TODO(whyuan): bytes_sent and bytes_rcvd are different. Are both payload?
buildbot@webrtc.org3eb2c2f2014-06-10 09:39:06 +0000516 // For webrtc, bytes_sent does not include the RTP header length.
Stefan Holmer586b19b2015-09-18 11:14:31 +0200517 EXPECT_GT(GetSenderStats(0).bytes_sent, 0);
518 EXPECT_EQ_WAIT(NumRtpPackets(), GetSenderStats(0).packets_sent, kTimeout);
perkj26752742016-10-24 01:21:16 -0700519 EXPECT_EQ(kVideoWidth, GetSenderStats(0).send_frame_width);
520 EXPECT_EQ(kVideoHeight, GetSenderStats(0).send_frame_height);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000521
522 ASSERT_EQ(2U, info.receivers.size());
523 for (size_t i = 0; i < info.receivers.size(); ++i) {
Stefan Holmer586b19b2015-09-18 11:14:31 +0200524 EXPECT_EQ(1U, GetReceiverStats(i).ssrcs().size());
525 EXPECT_EQ(i + 1, GetReceiverStats(i).ssrcs()[0]);
526 EXPECT_EQ_WAIT(NumRtpBytes(), GetReceiverStats(i).bytes_rcvd, kTimeout);
527 EXPECT_EQ_WAIT(NumRtpPackets(), GetReceiverStats(i).packets_rcvd,
528 kTimeout);
perkj26752742016-10-24 01:21:16 -0700529 EXPECT_EQ_WAIT(kVideoWidth, GetReceiverStats(i).frame_width, kTimeout);
530 EXPECT_EQ_WAIT(kVideoHeight, GetReceiverStats(i).frame_height, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000531 }
532 }
533 // Test that stats work properly for a conf call with multiple send streams.
534 void GetStatsMultipleSendStreams() {
535 // Normal setup; note that we set the SSRC explicitly to ensure that
536 // it will come first in the senders map.
537 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200538 cricket::VideoSendParameters parameters;
539 parameters.codecs.push_back(DefaultCodec());
nisse4b4dc862016-02-17 05:25:36 -0800540 parameters.conference_mode = true;
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200541 EXPECT_TRUE(channel_->SetSendParameters(parameters));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000542 EXPECT_TRUE(channel_->AddRecvStream(
buildbot@webrtc.org99f63082014-07-18 23:31:30 +0000543 cricket::StreamParams::CreateLegacy(kSsrc)));
nisse08582ff2016-02-04 01:24:52 -0800544 EXPECT_TRUE(channel_->SetSink(kSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000545 EXPECT_TRUE(SetSend(true));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000546 EXPECT_TRUE(SendFrame());
547 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
perkj26752742016-10-24 01:21:16 -0700548 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000549
550 // Add an additional capturer, and hook up a renderer to receive it.
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000551 cricket::FakeVideoRenderer renderer2;
kwiberg686a8ef2016-02-26 03:00:35 -0800552 std::unique_ptr<cricket::FakeVideoCapturer> capturer(
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000553 CreateFakeVideoCapturer());
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000554 const int kTestWidth = 160;
555 const int kTestHeight = 120;
556 cricket::VideoFormat format(kTestWidth, kTestHeight,
557 cricket::VideoFormat::FpsToInterval(5),
558 cricket::FOURCC_I420);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000559 EXPECT_EQ(cricket::CS_RUNNING, capturer->Start(format));
560 EXPECT_TRUE(channel_->AddSendStream(
561 cricket::StreamParams::CreateLegacy(5678)));
deadbeef5a4a75a2016-06-02 16:23:38 -0700562 EXPECT_TRUE(channel_->SetVideoSend(5678, true, nullptr, capturer.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000563 EXPECT_TRUE(channel_->AddRecvStream(
564 cricket::StreamParams::CreateLegacy(5678)));
nisse08582ff2016-02-04 01:24:52 -0800565 EXPECT_TRUE(channel_->SetSink(5678, &renderer2));
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000566 EXPECT_TRUE(capturer->CaptureCustomFrame(
567 kTestWidth, kTestHeight, cricket::FOURCC_I420));
568 EXPECT_FRAME_ON_RENDERER_WAIT(
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000569 renderer2, 1, kTestWidth, kTestHeight, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000570
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000571 // Get stats, and make sure they are correct for two senders. We wait until
572 // the number of expected packets have been sent to avoid races where we
573 // check stats before it has been updated.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000574 cricket::VideoMediaInfo info;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200575 for (uint32_t i = 0; i < kTimeout; ++i) {
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000576 rtc::Thread::Current()->ProcessMessages(1);
pbos@webrtc.org058b1f12015-03-04 08:54:32 +0000577 EXPECT_TRUE(channel_->GetStats(&info));
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000578 ASSERT_EQ(2U, info.senders.size());
579 if (info.senders[0].packets_sent + info.senders[1].packets_sent ==
580 NumRtpPackets()) {
581 // Stats have been updated for both sent frames, expectations can be
582 // checked now.
583 break;
584 }
585 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000586 EXPECT_EQ(NumRtpPackets(),
pbos@webrtc.orgec45e3b2015-02-20 10:24:53 +0000587 info.senders[0].packets_sent + info.senders[1].packets_sent)
588 << "Timed out while waiting for packet counts for all sent packets.";
sergeyu@chromium.org5bc25c42013-12-05 00:24:06 +0000589 EXPECT_EQ(1U, info.senders[0].ssrcs().size());
590 EXPECT_EQ(1234U, info.senders[0].ssrcs()[0]);
perkj26752742016-10-24 01:21:16 -0700591 EXPECT_EQ(kVideoWidth, info.senders[0].send_frame_width);
592 EXPECT_EQ(kVideoHeight, info.senders[0].send_frame_height);
sergeyu@chromium.org5bc25c42013-12-05 00:24:06 +0000593 EXPECT_EQ(1U, info.senders[1].ssrcs().size());
594 EXPECT_EQ(5678U, info.senders[1].ssrcs()[0]);
wu@webrtc.org987f2c92014-03-28 16:22:19 +0000595 EXPECT_EQ(kTestWidth, info.senders[1].send_frame_width);
596 EXPECT_EQ(kTestHeight, info.senders[1].send_frame_height);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000597 // The capturer must be unregistered here as it runs out of it's scope next.
deadbeef5a4a75a2016-06-02 16:23:38 -0700598 channel_->SetVideoSend(5678, true, nullptr, nullptr);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000599 }
600
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000601 // Test that we can set the bandwidth.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000602 void SetSendBandwidth() {
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200603 cricket::VideoSendParameters parameters;
604 parameters.codecs.push_back(DefaultCodec());
605 parameters.max_bandwidth_bps = -1; // <= 0 means unlimited.
606 EXPECT_TRUE(channel_->SetSendParameters(parameters));
607 parameters.max_bandwidth_bps = 128 * 1024;
608 EXPECT_TRUE(channel_->SetSendParameters(parameters));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000609 }
610 // Test that we can set the SSRC for the default send source.
611 void SetSendSsrc() {
612 EXPECT_TRUE(SetDefaultCodec());
613 EXPECT_TRUE(SetSend(true));
614 EXPECT_TRUE(SendFrame());
615 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
Peter Boström0c4e06b2015-10-07 12:23:21 +0200616 uint32_t ssrc = 0;
jbaucheec21bd2016-03-20 06:15:43 -0700617 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000618 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL);
619 EXPECT_EQ(kSsrc, ssrc);
Peter Boström718b6c72015-11-11 16:19:33 +0000620 // Packets are being paced out, so these can mismatch between the first and
621 // second call to NumRtpPackets until pending packets are paced out.
622 EXPECT_EQ_WAIT(NumRtpPackets(), NumRtpPackets(ssrc), kTimeout);
623 EXPECT_EQ_WAIT(NumRtpBytes(), NumRtpBytes(ssrc), kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000624 EXPECT_EQ(1, NumSentSsrcs());
625 EXPECT_EQ(0, NumRtpPackets(kSsrc - 1));
626 EXPECT_EQ(0, NumRtpBytes(kSsrc - 1));
627 }
628 // Test that we can set the SSRC even after codecs are set.
629 void SetSendSsrcAfterSetCodecs() {
630 // Remove stream added in Setup.
631 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc));
632 EXPECT_TRUE(SetDefaultCodec());
633 EXPECT_TRUE(channel_->AddSendStream(
634 cricket::StreamParams::CreateLegacy(999)));
deadbeef5a4a75a2016-06-02 16:23:38 -0700635 EXPECT_TRUE(
636 channel_->SetVideoSend(999u, true, nullptr, video_capturer_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000637 EXPECT_TRUE(SetSend(true));
638 EXPECT_TRUE(WaitAndSendFrame(0));
639 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout);
Peter Boström0c4e06b2015-10-07 12:23:21 +0200640 uint32_t ssrc = 0;
jbaucheec21bd2016-03-20 06:15:43 -0700641 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000642 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL);
643 EXPECT_EQ(999u, ssrc);
Peter Boström718b6c72015-11-11 16:19:33 +0000644 // Packets are being paced out, so these can mismatch between the first and
645 // second call to NumRtpPackets until pending packets are paced out.
646 EXPECT_EQ_WAIT(NumRtpPackets(), NumRtpPackets(ssrc), kTimeout);
647 EXPECT_EQ_WAIT(NumRtpBytes(), NumRtpBytes(ssrc), kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000648 EXPECT_EQ(1, NumSentSsrcs());
649 EXPECT_EQ(0, NumRtpPackets(kSsrc));
650 EXPECT_EQ(0, NumRtpBytes(kSsrc));
651 }
652 // Test that we can set the default video renderer before and after
653 // media is received.
nisse08582ff2016-02-04 01:24:52 -0800654 void SetSink() {
Peter Boström0c4e06b2015-10-07 12:23:21 +0200655 uint8_t data1[] = {
656 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000657
jbaucheec21bd2016-03-20 06:15:43 -0700658 rtc::CopyOnWriteBuffer packet1(data1, sizeof(data1));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000659 rtc::SetBE32(packet1.data() + 8, kSsrc);
nisse08582ff2016-02-04 01:24:52 -0800660 channel_->SetSink(kDefaultReceiveSsrc, NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000661 EXPECT_TRUE(SetDefaultCodec());
662 EXPECT_TRUE(SetSend(true));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000663 EXPECT_EQ(0, renderer_.num_rendered_frames());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000664 channel_->OnPacketReceived(&packet1, rtc::PacketTime());
nisse08582ff2016-02-04 01:24:52 -0800665 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000666 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700667 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000668 }
669
670 // Tests empty StreamParams is rejected.
671 void RejectEmptyStreamParams() {
672 // Remove the send stream that was added during Setup.
673 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc));
674
675 cricket::StreamParams empty;
676 EXPECT_FALSE(channel_->AddSendStream(empty));
677 EXPECT_TRUE(channel_->AddSendStream(
678 cricket::StreamParams::CreateLegacy(789u)));
679 }
680
681 // Tests setting up and configuring a send stream.
682 void AddRemoveSendStreams() {
683 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
684 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800685 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000686 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700687 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
Stefan Holmer586b19b2015-09-18 11:14:31 +0200688 EXPECT_GT(NumRtpPackets(), 0);
Peter Boström0c4e06b2015-10-07 12:23:21 +0200689 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000690 size_t last_packet = NumRtpPackets() - 1;
jbaucheec21bd2016-03-20 06:15:43 -0700691 std::unique_ptr<const rtc::CopyOnWriteBuffer>
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000692 p(GetRtpPacket(static_cast<int>(last_packet)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000693 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL);
694 EXPECT_EQ(kSsrc, ssrc);
695
696 // Remove the send stream that was added during Setup.
697 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc));
698 int rtp_packets = NumRtpPackets();
699
700 EXPECT_TRUE(channel_->AddSendStream(
701 cricket::StreamParams::CreateLegacy(789u)));
deadbeef5a4a75a2016-06-02 16:23:38 -0700702 EXPECT_TRUE(
703 channel_->SetVideoSend(789u, true, nullptr, video_capturer_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000704 EXPECT_EQ(rtp_packets, NumRtpPackets());
705 // Wait 30ms to guarantee the engine does not drop the frame.
706 EXPECT_TRUE(WaitAndSendFrame(30));
707 EXPECT_TRUE_WAIT(NumRtpPackets() > rtp_packets, kTimeout);
708
709 last_packet = NumRtpPackets() - 1;
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000710 p.reset(GetRtpPacket(static_cast<int>(last_packet)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000711 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL);
712 EXPECT_EQ(789u, ssrc);
713 }
714
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000715 // Tests the behavior of incoming streams in a conference scenario.
716 void SimulateConference() {
717 cricket::FakeVideoRenderer renderer1, renderer2;
718 EXPECT_TRUE(SetDefaultCodec());
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200719 cricket::VideoSendParameters parameters;
720 parameters.codecs.push_back(DefaultCodec());
nisse4b4dc862016-02-17 05:25:36 -0800721 parameters.conference_mode = true;
Fredrik Solenbergb071a192015-09-17 16:42:56 +0200722 EXPECT_TRUE(channel_->SetSendParameters(parameters));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000723 EXPECT_TRUE(SetSend(true));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000724 EXPECT_TRUE(channel_->AddRecvStream(
725 cricket::StreamParams::CreateLegacy(1)));
726 EXPECT_TRUE(channel_->AddRecvStream(
727 cricket::StreamParams::CreateLegacy(2)));
nisse08582ff2016-02-04 01:24:52 -0800728 EXPECT_TRUE(channel_->SetSink(1, &renderer1));
729 EXPECT_TRUE(channel_->SetSink(2, &renderer2));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000730 EXPECT_EQ(0, renderer1.num_rendered_frames());
731 EXPECT_EQ(0, renderer2.num_rendered_frames());
Peter Boström0c4e06b2015-10-07 12:23:21 +0200732 std::vector<uint32_t> ssrcs;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000733 ssrcs.push_back(1);
734 ssrcs.push_back(2);
735 network_interface_.SetConferenceMode(true, ssrcs);
736 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700737 EXPECT_FRAME_ON_RENDERER_WAIT(renderer1, 1, kVideoWidth, kVideoHeight,
738 kTimeout);
739 EXPECT_FRAME_ON_RENDERER_WAIT(renderer2, 1, kVideoWidth, kVideoHeight,
740 kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000741
jbaucheec21bd2016-03-20 06:15:43 -0700742 std::unique_ptr<const rtc::CopyOnWriteBuffer> p(GetRtpPacket(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000743 EXPECT_EQ(DefaultCodec().id, GetPayloadType(p.get()));
perkj26752742016-10-24 01:21:16 -0700744 EXPECT_EQ(kVideoWidth, renderer1.width());
745 EXPECT_EQ(kVideoHeight, renderer1.height());
746 EXPECT_EQ(kVideoWidth, renderer2.width());
747 EXPECT_EQ(kVideoHeight, renderer2.height());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000748 EXPECT_TRUE(channel_->RemoveRecvStream(2));
749 EXPECT_TRUE(channel_->RemoveRecvStream(1));
750 }
751
752 // Tests that we can add and remove capturers and frames are sent out properly
753 void AddRemoveCapturer() {
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000754 cricket::VideoCodec codec = DefaultCodec();
perkj26752742016-10-24 01:21:16 -0700755 const int time_between_send_ms =
756 cricket::VideoFormat::FpsToInterval(kFramerate);
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000757 EXPECT_TRUE(SetOneCodec(codec));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000758 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800759 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000760 EXPECT_EQ(0, renderer_.num_rendered_frames());
761 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700762 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
kwiberg686a8ef2016-02-26 03:00:35 -0800763 std::unique_ptr<cricket::FakeVideoCapturer> capturer(
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000764 CreateFakeVideoCapturer());
Niels Möller60653ba2016-03-02 11:41:36 +0100765
766 // TODO(nisse): This testcase fails if we don't configure
767 // screencast. It's unclear why, I see nothing obvious in this
768 // test which is related to screencast logic.
769 cricket::VideoOptions video_options;
770 video_options.is_screencast = rtc::Optional<bool>(true);
deadbeef5a4a75a2016-06-02 16:23:38 -0700771 channel_->SetVideoSend(kSsrc, true, &video_options, nullptr);
Niels Möller60653ba2016-03-02 11:41:36 +0100772
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000773 cricket::VideoFormat format(480, 360,
774 cricket::VideoFormat::FpsToInterval(30),
775 cricket::FOURCC_I420);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000776 EXPECT_EQ(cricket::CS_RUNNING, capturer->Start(format));
777 // All capturers start generating frames with the same timestamp. ViE does
778 // not allow the same timestamp to be used. Capture one frame before
779 // associating the capturer with the channel.
780 EXPECT_TRUE(capturer->CaptureCustomFrame(format.width, format.height,
781 cricket::FOURCC_I420));
782
783 int captured_frames = 1;
784 for (int iterations = 0; iterations < 2; ++iterations) {
deadbeef5a4a75a2016-06-02 16:23:38 -0700785 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, capturer.get()));
perkj26752742016-10-24 01:21:16 -0700786 rtc::Thread::Current()->ProcessMessages(time_between_send_ms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000787 EXPECT_TRUE(capturer->CaptureCustomFrame(format.width, format.height,
788 cricket::FOURCC_I420));
789 ++captured_frames;
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000790 // Wait until frame of right size is captured.
791 EXPECT_TRUE_WAIT(renderer_.num_rendered_frames() >= captured_frames &&
792 format.width == renderer_.width() &&
henrike@webrtc.org1e09a712013-07-26 19:17:59 +0000793 format.height == renderer_.height() &&
794 !renderer_.black_frame(), kTimeout);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000795 EXPECT_GE(renderer_.num_rendered_frames(), captured_frames);
796 EXPECT_EQ(format.width, renderer_.width());
797 EXPECT_EQ(format.height, renderer_.height());
798 captured_frames = renderer_.num_rendered_frames() + 1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000799 EXPECT_FALSE(renderer_.black_frame());
deadbeef5a4a75a2016-06-02 16:23:38 -0700800 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, nullptr));
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000801 // Make sure a black frame is generated within the specified timeout.
pbos@webrtc.orgb4987bf2015-02-18 10:13:09 +0000802 // The black frame should be the resolution of the previous frame to
803 // prevent expensive encoder reconfigurations.
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000804 EXPECT_TRUE_WAIT(renderer_.num_rendered_frames() >= captured_frames &&
pbos@webrtc.orgb4987bf2015-02-18 10:13:09 +0000805 format.width == renderer_.width() &&
806 format.height == renderer_.height() &&
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000807 renderer_.black_frame(), kTimeout);
808 EXPECT_GE(renderer_.num_rendered_frames(), captured_frames);
pbos@webrtc.orgb4987bf2015-02-18 10:13:09 +0000809 EXPECT_EQ(format.width, renderer_.width());
810 EXPECT_EQ(format.height, renderer_.height());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000811 EXPECT_TRUE(renderer_.black_frame());
812
813 // The black frame has the same timestamp as the next frame since it's
814 // timestamp is set to the last frame's timestamp + interval. WebRTC will
815 // not render a frame with the same timestamp so capture another frame
816 // with the frame capturer to increment the next frame's timestamp.
817 EXPECT_TRUE(capturer->CaptureCustomFrame(format.width, format.height,
818 cricket::FOURCC_I420));
819 }
820 }
821
deadbeef5a4a75a2016-06-02 16:23:38 -0700822 // Tests that if SetVideoSend is called with a NULL capturer after the
823 // capturer was already removed, the application doesn't crash (and no black
824 // frame is sent).
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000825 void RemoveCapturerWithoutAdd() {
826 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
827 EXPECT_TRUE(SetSend(true));
nisse08582ff2016-02-04 01:24:52 -0800828 EXPECT_TRUE(channel_->SetSink(kDefaultReceiveSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000829 EXPECT_EQ(0, renderer_.num_rendered_frames());
830 EXPECT_TRUE(SendFrame());
perkj26752742016-10-24 01:21:16 -0700831 EXPECT_FRAME_WAIT(1, kVideoWidth, kVideoHeight, kTimeout);
pbos@webrtc.org776e6f22014-10-29 15:28:39 +0000832 // Wait for one frame so they don't get dropped because we send frames too
833 // tightly.
834 rtc::Thread::Current()->ProcessMessages(30);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000835 // Remove the capturer.
deadbeef5a4a75a2016-06-02 16:23:38 -0700836 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, nullptr));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000837
deadbeef5a4a75a2016-06-02 16:23:38 -0700838 // No capturer was added, so this SetVideoSend shouldn't do anything.
839 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, nullptr));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000840 rtc::Thread::Current()->ProcessMessages(300);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000841 // Verify no more frames were sent.
perkjd533aec2017-01-13 05:57:25 -0800842 EXPECT_EQ(1, renderer_.num_rendered_frames());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000843 }
844
845 // Tests that we can add and remove capturer as unique sources.
846 void AddRemoveCapturerMultipleSources() {
847 // WebRTC implementation will drop frames if pushed to quickly. Wait the
848 // interval time to avoid that.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000849 // WebRTC implementation will drop frames if pushed to quickly. Wait the
850 // interval time to avoid that.
851 // Set up the stream associated with the engine.
852 EXPECT_TRUE(channel_->AddRecvStream(
853 cricket::StreamParams::CreateLegacy(kSsrc)));
nisse08582ff2016-02-04 01:24:52 -0800854 EXPECT_TRUE(channel_->SetSink(kSsrc, &renderer_));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000855 cricket::VideoFormat capture_format; // default format
perkj26752742016-10-24 01:21:16 -0700856 capture_format.interval = cricket::VideoFormat::FpsToInterval(kFramerate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000857 // Set up additional stream 1.
858 cricket::FakeVideoRenderer renderer1;
nisse08582ff2016-02-04 01:24:52 -0800859 EXPECT_FALSE(channel_->SetSink(1, &renderer1));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000860 EXPECT_TRUE(channel_->AddRecvStream(
861 cricket::StreamParams::CreateLegacy(1)));
nisse08582ff2016-02-04 01:24:52 -0800862 EXPECT_TRUE(channel_->SetSink(1, &renderer1));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000863 EXPECT_TRUE(channel_->AddSendStream(
864 cricket::StreamParams::CreateLegacy(1)));
kwiberg686a8ef2016-02-26 03:00:35 -0800865 std::unique_ptr<cricket::FakeVideoCapturer> capturer1(
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000866 CreateFakeVideoCapturer());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000867 EXPECT_EQ(cricket::CS_RUNNING, capturer1->Start(capture_format));
868 // Set up additional stream 2.
869 cricket::FakeVideoRenderer renderer2;
nisse08582ff2016-02-04 01:24:52 -0800870 EXPECT_FALSE(channel_->SetSink(2, &renderer2));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000871 EXPECT_TRUE(channel_->AddRecvStream(
872 cricket::StreamParams::CreateLegacy(2)));
nisse08582ff2016-02-04 01:24:52 -0800873 EXPECT_TRUE(channel_->SetSink(2, &renderer2));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000874 EXPECT_TRUE(channel_->AddSendStream(
875 cricket::StreamParams::CreateLegacy(2)));
kwiberg686a8ef2016-02-26 03:00:35 -0800876 std::unique_ptr<cricket::FakeVideoCapturer> capturer2(
buildbot@webrtc.org4f0d4012014-08-07 04:47:36 +0000877 CreateFakeVideoCapturer());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000878 EXPECT_EQ(cricket::CS_RUNNING, capturer2->Start(capture_format));
879 // State for all the streams.
880 EXPECT_TRUE(SetOneCodec(DefaultCodec()));
deadbeef5a4a75a2016-06-02 16:23:38 -0700881 // A limitation in the lmi implementation requires that SetVideoSend() is
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000882 // called after SetOneCodec().
883 // TODO(hellner): this seems like an unnecessary constraint, fix it.
deadbeef5a4a75a2016-06-02 16:23:38 -0700884 EXPECT_TRUE(channel_->SetVideoSend(1, true, nullptr, capturer1.get()));
885 EXPECT_TRUE(channel_->SetVideoSend(2, true, nullptr, capturer2.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000886 EXPECT_TRUE(SetSend(true));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000887 // Test capturer associated with engine.
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000888 const int kTestWidth = 160;
889 const int kTestHeight = 120;
890 EXPECT_TRUE(capturer1->CaptureCustomFrame(
891 kTestWidth, kTestHeight, cricket::FOURCC_I420));
892 EXPECT_FRAME_ON_RENDERER_WAIT(
893 renderer1, 1, kTestWidth, kTestHeight, kTimeout);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000894 // Capture a frame with additional capturer2, frames should be received
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +0000895 EXPECT_TRUE(capturer2->CaptureCustomFrame(
896 kTestWidth, kTestHeight, cricket::FOURCC_I420));
897 EXPECT_FRAME_ON_RENDERER_WAIT(
898 renderer2, 1, kTestWidth, kTestHeight, kTimeout);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000899 // Successfully remove the capturer.
deadbeef5a4a75a2016-06-02 16:23:38 -0700900 EXPECT_TRUE(channel_->SetVideoSend(kSsrc, true, nullptr, nullptr));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000901 // The capturers must be unregistered here as it runs out of it's scope
902 // next.
deadbeef5a4a75a2016-06-02 16:23:38 -0700903 EXPECT_TRUE(channel_->SetVideoSend(1, true, nullptr, nullptr));
904 EXPECT_TRUE(channel_->SetVideoSend(2, true, nullptr, nullptr));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000905 }
906
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000907 // Test that multiple send streams can be created and deleted properly.
908 void MultipleSendStreams() {
909 // Remove stream added in Setup. I.e. remove stream corresponding to default
910 // channel.
911 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc));
912 const unsigned int kSsrcsSize = sizeof(kSsrcs4)/sizeof(kSsrcs4[0]);
913 for (unsigned int i = 0; i < kSsrcsSize; ++i) {
914 EXPECT_TRUE(channel_->AddSendStream(
915 cricket::StreamParams::CreateLegacy(kSsrcs4[i])));
916 }
917 // Delete one of the non default channel streams, let the destructor delete
918 // the remaining ones.
919 EXPECT_TRUE(channel_->RemoveSendStream(kSsrcs4[kSsrcsSize - 1]));
920 // Stream should already be deleted.
921 EXPECT_FALSE(channel_->RemoveSendStream(kSsrcs4[kSsrcsSize - 1]));
922 }
923
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000924 // Two streams one channel tests.
925
926 // Tests that we can send and receive frames.
927 void TwoStreamsSendAndReceive(const cricket::VideoCodec& codec) {
928 SetUpSecondStream();
929 // Test sending and receiving on first stream.
Peter Boströmd1f584b2016-04-20 16:31:53 +0200930 SendAndReceive(codec);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000931 // Test sending and receiving on second stream.
932 EXPECT_EQ_WAIT(1, renderer2_.num_rendered_frames(), kTimeout);
Stefan Holmer586b19b2015-09-18 11:14:31 +0200933 EXPECT_GT(NumRtpPackets(), 0);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000934 EXPECT_EQ(1, renderer2_.num_rendered_frames());
935 }
936
skvlad11a9cbf2016-10-07 11:53:05 -0700937 webrtc::RtcEventLogNullImpl event_log_;
kwiberg686a8ef2016-02-26 03:00:35 -0800938 const std::unique_ptr<webrtc::Call> call_;
nisse2ded9b12016-04-08 02:23:55 -0700939 E engine_;
kwiberg686a8ef2016-02-26 03:00:35 -0800940 std::unique_ptr<cricket::FakeVideoCapturer> video_capturer_;
941 std::unique_ptr<cricket::FakeVideoCapturer> video_capturer_2_;
942 std::unique_ptr<C> channel_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000943 cricket::FakeNetworkInterface network_interface_;
944 cricket::FakeVideoRenderer renderer_;
945 cricket::VideoMediaChannel::Error media_error_;
946
947 // Used by test cases where 2 streams are run on the same channel.
948 cricket::FakeVideoRenderer renderer2_;
949};
950
Mirko Bonadei92ea95e2017-09-15 06:47:31 +0200951#endif // MEDIA_BASE_VIDEOENGINE_UNITTEST_H_ NOLINT