1 // libjingle 2 // Copyright 2004 Google Inc. All rights reserved. 3 // 4 // Redistribution and use in source and binary forms, with or without 5 // modification, are permitted provided that the following conditions are met: 6 // 7 // 1. Redistributions of source code must retain the above copyright notice, 8 // this list of conditions and the following disclaimer. 9 // 2. Redistributions in binary form must reproduce the above copyright notice, 10 // this list of conditions and the following disclaimer in the documentation 11 // and/or other materials provided with the distribution. 12 // 3. The name of the author may not be used to endorse or promote products 13 // derived from this software without specific prior written permission. 14 // 15 // THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED 16 // WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 17 // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO 18 // EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 19 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 20 // PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; 21 // OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, 22 // WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR 23 // OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF 24 // ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 25 26 #ifndef TALK_MEDIA_BASE_VIDEOENGINE_UNITTEST_H_ // NOLINT 27 #define TALK_MEDIA_BASE_VIDEOENGINE_UNITTEST_H_ 28 29 #include <string> 30 #include <vector> 31 32 #include "talk/base/bytebuffer.h" 33 #include "talk/base/gunit.h" 34 #include "talk/base/timeutils.h" 35 #include "talk/media/base/fakenetworkinterface.h" 36 #include "talk/media/base/fakevideocapturer.h" 37 #include "talk/media/base/fakevideorenderer.h" 38 #include "talk/media/base/mediachannel.h" 39 #include "talk/media/base/streamparams.h" 40 41 #ifdef WIN32 42 #include <objbase.h> // NOLINT 43 #endif 44 45 #define EXPECT_FRAME_WAIT(c, w, h, t) \ 46 EXPECT_EQ_WAIT((c), renderer_.num_rendered_frames(), (t)); \ 47 EXPECT_EQ((w), renderer_.width()); \ 48 EXPECT_EQ((h), renderer_.height()); \ 49 EXPECT_EQ(0, renderer_.errors()); \ 50 51 #define EXPECT_FRAME_ON_RENDERER_WAIT(r, c, w, h, t) \ 52 EXPECT_EQ_WAIT((c), (r).num_rendered_frames(), (t)); \ 53 EXPECT_EQ((w), (r).width()); \ 54 EXPECT_EQ((h), (r).height()); \ 55 EXPECT_EQ(0, (r).errors()); \ 56 57 #define EXPECT_GT_FRAME_ON_RENDERER_WAIT(r, c, w, h, t) \ 58 EXPECT_TRUE_WAIT((r).num_rendered_frames() >= (c) && \ 59 (w) == (r).width() && \ 60 (h) == (r).height(), (t)); \ 61 EXPECT_EQ(0, (r).errors()); \ 62 63 static const uint32 kTimeout = 5000U; 64 static const uint32 kSsrc = 1234u; 65 static const uint32 kRtxSsrc = 4321u; 66 static const uint32 kSsrcs4[] = {1, 2, 3, 4}; 67 68 inline bool IsEqualRes(const cricket::VideoCodec& a, int w, int h, int fps) { 69 return a.width == w && a.height == h && a.framerate == fps; 70 } 71 72 inline bool IsEqualCodec(const cricket::VideoCodec& a, 73 const cricket::VideoCodec& b) { 74 return a.id == b.id && a.name == b.name && 75 IsEqualRes(a, b.width, b.height, b.framerate); 76 } 77 78 namespace std { 79 inline std::ostream& operator<<(std::ostream& s, const cricket::VideoCodec& c) { 80 s << "{" << c.name << "(" << c.id << "), " 81 << c.width << "x" << c.height << "x" << c.framerate << "}"; 82 return s; 83 } 84 } // namespace std 85 86 inline int TimeBetweenSend(const cricket::VideoCodec& codec) { 87 return static_cast<int>( 88 cricket::VideoFormat::FpsToInterval(codec.framerate) / 89 talk_base::kNumNanosecsPerMillisec); 90 } 91 92 // Fake video engine that makes it possible to test enabling and disabling 93 // capturer (checking that the engine state is updated and that the capturer 94 // is indeed capturing) without having to create a channel. It also makes it 95 // possible to test that the media processors are indeed being called when 96 // registered. 97 template<class T> 98 class VideoEngineOverride : public T { 99 public: 100 VideoEngineOverride() { 101 } 102 virtual ~VideoEngineOverride() { 103 } 104 bool is_camera_on() const { return T::GetVideoCapturer()->IsRunning(); } 105 void set_has_senders(bool has_senders) { 106 cricket::VideoCapturer* video_capturer = T::GetVideoCapturer(); 107 if (has_senders) { 108 video_capturer->SignalVideoFrame.connect(this, 109 &VideoEngineOverride<T>::OnLocalFrame); 110 } else { 111 video_capturer->SignalVideoFrame.disconnect(this); 112 } 113 } 114 void OnLocalFrame(cricket::VideoCapturer*, 115 const cricket::VideoFrame*) { 116 } 117 void OnLocalFrameFormat(cricket::VideoCapturer*, 118 const cricket::VideoFormat*) { 119 } 120 121 void TriggerMediaFrame( 122 uint32 ssrc, cricket::VideoFrame* frame, bool* drop_frame) { 123 T::SignalMediaFrame(ssrc, frame, drop_frame); 124 } 125 }; 126 127 // Macroes that declare test functions for a given test class, before and after 128 // Init(). 129 // To use, define a test function called FooBody and pass Foo to the macro. 130 #define TEST_PRE_VIDEOENGINE_INIT(TestClass, func) \ 131 TEST_F(TestClass, func##PreInit) { \ 132 func##Body(); \ 133 } 134 #define TEST_POST_VIDEOENGINE_INIT(TestClass, func) \ 135 TEST_F(TestClass, func##PostInit) { \ 136 EXPECT_TRUE(engine_.Init(talk_base::Thread::Current())); \ 137 func##Body(); \ 138 engine_.Terminate(); \ 139 } 140 141 template<class E> 142 class VideoEngineTest : public testing::Test { 143 protected: 144 // Tests starting and stopping the engine, and creating a channel. 145 void StartupShutdown() { 146 EXPECT_TRUE(engine_.Init(talk_base::Thread::Current())); 147 cricket::VideoMediaChannel* channel = engine_.CreateChannel(NULL); 148 EXPECT_TRUE(channel != NULL); 149 delete channel; 150 engine_.Terminate(); 151 } 152 153 #ifdef WIN32 154 // Tests that the COM reference count is not munged by the engine. 155 // Test to make sure LMI does not munge the CoInitialize reference count. 156 void CheckCoInitialize() { 157 // Initial refcount should be 0. 158 EXPECT_EQ(S_OK, CoInitializeEx(NULL, COINIT_MULTITHREADED)); 159 160 // Engine should start even with COM already inited. 161 EXPECT_TRUE(engine_.Init(talk_base::Thread::Current())); 162 engine_.Terminate(); 163 // Refcount after terminate should be 1; this tests if it is nonzero. 164 EXPECT_EQ(S_FALSE, CoInitializeEx(NULL, COINIT_MULTITHREADED)); 165 // Decrement refcount to (hopefully) 0. 166 CoUninitialize(); 167 CoUninitialize(); 168 169 // Ensure refcount is 0. 170 EXPECT_EQ(S_OK, CoInitializeEx(NULL, COINIT_MULTITHREADED)); 171 CoUninitialize(); 172 } 173 #endif 174 175 void ConstrainNewCodecBody() { 176 cricket::VideoCodec empty, in, out; 177 cricket::VideoCodec max_settings(engine_.codecs()[0].id, 178 engine_.codecs()[0].name, 179 1280, 800, 30, 0); 180 181 // set max settings of 1280x960x30 182 EXPECT_TRUE(engine_.SetDefaultEncoderConfig( 183 cricket::VideoEncoderConfig(max_settings))); 184 185 // don't constrain the max resolution 186 in = max_settings; 187 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 188 EXPECT_PRED2(IsEqualCodec, out, in); 189 190 // constrain resolution greater than the max and wider aspect, 191 // picking best aspect (16:10) 192 in.width = 1380; 193 in.height = 800; 194 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 195 EXPECT_PRED4(IsEqualRes, out, 1280, 720, 30); 196 197 // constrain resolution greater than the max and narrow aspect, 198 // picking best aspect (16:9) 199 in.width = 1280; 200 in.height = 740; 201 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 202 EXPECT_PRED4(IsEqualRes, out, 1280, 720, 30); 203 204 // constrain resolution greater than the max, picking equal aspect (4:3) 205 in.width = 1280; 206 in.height = 960; 207 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 208 EXPECT_PRED4(IsEqualRes, out, 1280, 800, 30); 209 210 // constrain resolution greater than the max, picking equal aspect (16:10) 211 in.width = 1280; 212 in.height = 800; 213 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 214 EXPECT_PRED4(IsEqualRes, out, 1280, 800, 30); 215 216 // reduce max settings to 640x480x30 217 max_settings.width = 640; 218 max_settings.height = 480; 219 EXPECT_TRUE(engine_.SetDefaultEncoderConfig( 220 cricket::VideoEncoderConfig(max_settings))); 221 222 // don't constrain the max resolution 223 in = max_settings; 224 in.width = 640; 225 in.height = 480; 226 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 227 EXPECT_PRED2(IsEqualCodec, out, in); 228 229 // keep 16:10 if they request it 230 in.height = 400; 231 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 232 EXPECT_PRED2(IsEqualCodec, out, in); 233 234 // don't constrain lesser 4:3 resolutions 235 in.width = 320; 236 in.height = 240; 237 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 238 EXPECT_PRED2(IsEqualCodec, out, in); 239 240 // don't constrain lesser 16:10 resolutions 241 in.width = 320; 242 in.height = 200; 243 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 244 EXPECT_PRED2(IsEqualCodec, out, in); 245 246 // requested resolution of 0x0 succeeds 247 in.width = 0; 248 in.height = 0; 249 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 250 EXPECT_PRED2(IsEqualCodec, out, in); 251 252 // constrain resolution lesser than the max and wider aspect, 253 // picking best aspect (16:9) 254 in.width = 350; 255 in.height = 201; 256 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 257 EXPECT_PRED4(IsEqualRes, out, 320, 180, 30); 258 259 // constrain resolution greater than the max and narrow aspect, 260 // picking best aspect (4:3) 261 in.width = 350; 262 in.height = 300; 263 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 264 EXPECT_PRED4(IsEqualRes, out, 320, 240, 30); 265 266 // constrain resolution greater than the max and wider aspect, 267 // picking best aspect (16:9) 268 in.width = 1380; 269 in.height = 800; 270 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 271 EXPECT_PRED4(IsEqualRes, out, 640, 360, 30); 272 273 // constrain resolution greater than the max and narrow aspect, 274 // picking best aspect (4:3) 275 in.width = 1280; 276 in.height = 900; 277 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 278 EXPECT_PRED4(IsEqualRes, out, 640, 480, 30); 279 280 // constrain resolution greater than the max, picking equal aspect (4:3) 281 in.width = 1280; 282 in.height = 960; 283 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 284 EXPECT_PRED4(IsEqualRes, out, 640, 480, 30); 285 286 // constrain resolution greater than the max, picking equal aspect (16:10) 287 in.width = 1280; 288 in.height = 800; 289 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 290 EXPECT_PRED4(IsEqualRes, out, 640, 400, 30); 291 292 // constrain res & fps greater than the max 293 in.framerate = 50; 294 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 295 EXPECT_PRED4(IsEqualRes, out, 640, 400, 30); 296 297 // reduce max settings to 160x100x10 298 max_settings.width = 160; 299 max_settings.height = 100; 300 max_settings.framerate = 10; 301 EXPECT_TRUE(engine_.SetDefaultEncoderConfig( 302 cricket::VideoEncoderConfig(max_settings))); 303 304 // constrain res & fps to new max 305 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 306 EXPECT_PRED4(IsEqualRes, out, 160, 100, 10); 307 308 // allow 4:3 "comparable" resolutions 309 in.width = 160; 310 in.height = 120; 311 in.framerate = 10; 312 EXPECT_TRUE(engine_.CanSendCodec(in, empty, &out)); 313 EXPECT_PRED4(IsEqualRes, out, 160, 120, 10); 314 } 315 316 void ConstrainRunningCodecBody() { 317 cricket::VideoCodec in, out, current; 318 cricket::VideoCodec max_settings(engine_.codecs()[0].id, 319 engine_.codecs()[0].name, 320 1280, 800, 30, 0); 321 322 // set max settings of 1280x960x30 323 EXPECT_TRUE(engine_.SetDefaultEncoderConfig( 324 cricket::VideoEncoderConfig(max_settings))); 325 326 // establish current call at 1280x800x30 (16:10) 327 current = max_settings; 328 current.height = 800; 329 330 // Don't constrain current resolution 331 in = current; 332 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 333 EXPECT_PRED2(IsEqualCodec, out, in); 334 335 // requested resolution of 0x0 succeeds 336 in.width = 0; 337 in.height = 0; 338 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 339 EXPECT_PRED2(IsEqualCodec, out, in); 340 341 // Reduce an intermediate resolution down to the next lowest one, preserving 342 // aspect ratio. 343 in.width = 800; 344 in.height = 600; 345 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 346 EXPECT_PRED4(IsEqualRes, out, 640, 400, 30); 347 348 // Clamping by aspect ratio, but still never return a dimension higher than 349 // requested. 350 in.width = 1280; 351 in.height = 720; 352 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 353 EXPECT_PRED4(IsEqualRes, out, 1280, 720, 30); 354 355 in.width = 1279; 356 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 357 EXPECT_PRED4(IsEqualRes, out, 960, 600, 30); 358 359 in.width = 1281; 360 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 361 EXPECT_PRED4(IsEqualRes, out, 1280, 720, 30); 362 363 // Clamp large resolutions down, always preserving aspect 364 in.width = 1920; 365 in.height = 1080; 366 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 367 EXPECT_PRED4(IsEqualRes, out, 1280, 800, 30); 368 369 in.width = 1921; 370 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 371 EXPECT_PRED4(IsEqualRes, out, 1280, 800, 30); 372 373 in.width = 1919; 374 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 375 EXPECT_PRED4(IsEqualRes, out, 1280, 800, 30); 376 377 // reduce max settings to 640x480x30 378 max_settings.width = 640; 379 max_settings.height = 480; 380 EXPECT_TRUE(engine_.SetDefaultEncoderConfig( 381 cricket::VideoEncoderConfig(max_settings))); 382 383 // establish current call at 640x400x30 (16:10) 384 current = max_settings; 385 current.height = 400; 386 387 // Don't constrain current resolution 388 in = current; 389 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 390 EXPECT_PRED2(IsEqualCodec, out, in); 391 392 // requested resolution of 0x0 succeeds 393 in.width = 0; 394 in.height = 0; 395 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 396 EXPECT_PRED2(IsEqualCodec, out, in); 397 398 // Reduce an intermediate resolution down to the next lowest one, preserving 399 // aspect ratio. 400 in.width = 400; 401 in.height = 300; 402 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 403 EXPECT_PRED4(IsEqualRes, out, 320, 200, 30); 404 405 // Clamping by aspect ratio, but still never return a dimension higher than 406 // requested. 407 in.width = 640; 408 in.height = 360; 409 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 410 EXPECT_PRED4(IsEqualRes, out, 640, 360, 30); 411 412 in.width = 639; 413 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 414 EXPECT_PRED4(IsEqualRes, out, 480, 300, 30); 415 416 in.width = 641; 417 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 418 EXPECT_PRED4(IsEqualRes, out, 640, 360, 30); 419 420 // Clamp large resolutions down, always preserving aspect 421 in.width = 1280; 422 in.height = 800; 423 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 424 EXPECT_PRED4(IsEqualRes, out, 640, 400, 30); 425 426 in.width = 1281; 427 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 428 EXPECT_PRED4(IsEqualRes, out, 640, 400, 30); 429 430 in.width = 1279; 431 EXPECT_TRUE(engine_.CanSendCodec(in, current, &out)); 432 EXPECT_PRED4(IsEqualRes, out, 640, 400, 30); 433 434 // Should fail for any that are smaller than our supported formats 435 in.width = 80; 436 in.height = 80; 437 EXPECT_FALSE(engine_.CanSendCodec(in, current, &out)); 438 439 in.height = 50; 440 EXPECT_FALSE(engine_.CanSendCodec(in, current, &out)); 441 } 442 443 VideoEngineOverride<E> engine_; 444 talk_base::scoped_ptr<cricket::FakeVideoCapturer> video_capturer_; 445 }; 446 447 template<class E, class C> 448 class VideoMediaChannelTest : public testing::Test, 449 public sigslot::has_slots<> { 450 protected: 451 virtual cricket::VideoCodec DefaultCodec() = 0; 452 453 virtual cricket::StreamParams DefaultSendStreamParams() { 454 return cricket::StreamParams::CreateLegacy(kSsrc); 455 } 456 457 virtual void SetUp() { 458 cricket::Device device("test", "device"); 459 EXPECT_TRUE(engine_.Init(talk_base::Thread::Current())); 460 channel_.reset(engine_.CreateChannel(NULL)); 461 EXPECT_TRUE(channel_.get() != NULL); 462 ConnectVideoChannelError(); 463 network_interface_.SetDestination(channel_.get()); 464 channel_->SetInterface(&network_interface_); 465 SetRendererAsDefault(); 466 media_error_ = cricket::VideoMediaChannel::ERROR_NONE; 467 channel_->SetRecvCodecs(engine_.codecs()); 468 EXPECT_TRUE(channel_->AddSendStream(DefaultSendStreamParams())); 469 470 video_capturer_.reset(new cricket::FakeVideoCapturer); 471 cricket::VideoFormat format(640, 480, 472 cricket::VideoFormat::FpsToInterval(30), 473 cricket::FOURCC_I420); 474 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_->Start(format)); 475 EXPECT_TRUE(channel_->SetCapturer(kSsrc, video_capturer_.get())); 476 } 477 void SetUpSecondStream() { 478 EXPECT_TRUE(channel_->AddRecvStream( 479 cricket::StreamParams::CreateLegacy(kSsrc))); 480 EXPECT_TRUE(channel_->AddRecvStream( 481 cricket::StreamParams::CreateLegacy(kSsrc + 2))); 482 // SetUp() already added kSsrc make sure duplicate SSRCs cant be added. 483 EXPECT_FALSE(channel_->AddSendStream( 484 cricket::StreamParams::CreateLegacy(kSsrc))); 485 EXPECT_TRUE(channel_->AddSendStream( 486 cricket::StreamParams::CreateLegacy(kSsrc + 2))); 487 488 video_capturer_2_.reset(new cricket::FakeVideoCapturer()); 489 cricket::VideoFormat format(640, 480, 490 cricket::VideoFormat::FpsToInterval(30), 491 cricket::FOURCC_I420); 492 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_2_->Start(format)); 493 494 EXPECT_TRUE(channel_->SetCapturer(kSsrc + 2, video_capturer_2_.get())); 495 // Make the second renderer available for use by a new stream. 496 EXPECT_TRUE(channel_->SetRenderer(kSsrc + 2, &renderer2_)); 497 } 498 virtual void TearDown() { 499 channel_.reset(); 500 engine_.Terminate(); 501 } 502 void ConnectVideoChannelError() { 503 channel_->SignalMediaError.connect(this, 504 &VideoMediaChannelTest<E, C>::OnVideoChannelError); 505 } 506 bool SetDefaultCodec() { 507 return SetOneCodec(DefaultCodec()); 508 } 509 void SetRendererAsDefault() { 510 EXPECT_TRUE(channel_->SetRenderer(0, &renderer_)); 511 } 512 513 bool SetOneCodec(int pt, const char* name, int w, int h, int fr) { 514 return SetOneCodec(cricket::VideoCodec(pt, name, w, h, fr, 0)); 515 } 516 bool SetOneCodec(const cricket::VideoCodec& codec) { 517 std::vector<cricket::VideoCodec> codecs; 518 codecs.push_back(codec); 519 520 cricket::VideoFormat capture_format(codec.width, codec.height, 521 cricket::VideoFormat::FpsToInterval(codec.framerate), 522 cricket::FOURCC_I420); 523 524 if (video_capturer_) { 525 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_->Start(capture_format)); 526 } 527 528 if (video_capturer_2_) { 529 EXPECT_EQ(cricket::CS_RUNNING, video_capturer_2_->Start(capture_format)); 530 } 531 532 bool sending = channel_->sending(); 533 bool success = SetSend(false); 534 if (success) 535 success = channel_->SetSendCodecs(codecs); 536 if (success) 537 success = SetSend(sending); 538 return success; 539 } 540 bool SetSend(bool send) { 541 return channel_->SetSend(send); 542 } 543 int DrainOutgoingPackets() { 544 int packets = 0; 545 do { 546 packets = NumRtpPackets(); 547 // 100 ms should be long enough. 548 talk_base::Thread::Current()->ProcessMessages(100); 549 } while (NumRtpPackets() > packets); 550 return NumRtpPackets(); 551 } 552 bool SendFrame() { 553 if (video_capturer_2_) { 554 video_capturer_2_->CaptureFrame(); 555 } 556 return video_capturer_.get() && 557 video_capturer_->CaptureFrame(); 558 } 559 bool WaitAndSendFrame(int wait_ms) { 560 bool ret = talk_base::Thread::Current()->ProcessMessages(wait_ms); 561 ret &= SendFrame(); 562 return ret; 563 } 564 // Sends frames and waits for the decoder to be fully initialized. 565 // Returns the number of frames that were sent. 566 int WaitForDecoder() { 567 #if defined(HAVE_OPENMAX) 568 // Send enough frames for the OpenMAX decoder to continue processing, and 569 // return the number of frames sent. 570 // Send frames for a full kTimeout's worth of 15fps video. 571 int frame_count = 0; 572 while (frame_count < static_cast<int>(kTimeout) / 66) { 573 EXPECT_TRUE(WaitAndSendFrame(66)); 574 ++frame_count; 575 } 576 return frame_count; 577 #else 578 return 0; 579 #endif 580 } 581 bool SendCustomVideoFrame(int w, int h) { 582 if (!video_capturer_.get()) return false; 583 return video_capturer_->CaptureCustomFrame(w, h, cricket::FOURCC_I420); 584 } 585 int NumRtpBytes() { 586 return network_interface_.NumRtpBytes(); 587 } 588 int NumRtpBytes(uint32 ssrc) { 589 return network_interface_.NumRtpBytes(ssrc); 590 } 591 int NumRtpPackets() { 592 return network_interface_.NumRtpPackets(); 593 } 594 int NumRtpPackets(uint32 ssrc) { 595 return network_interface_.NumRtpPackets(ssrc); 596 } 597 int NumSentSsrcs() { 598 return network_interface_.NumSentSsrcs(); 599 } 600 const talk_base::Buffer* GetRtpPacket(int index) { 601 return network_interface_.GetRtpPacket(index); 602 } 603 int NumRtcpPackets() { 604 return network_interface_.NumRtcpPackets(); 605 } 606 const talk_base::Buffer* GetRtcpPacket(int index) { 607 return network_interface_.GetRtcpPacket(index); 608 } 609 static int GetPayloadType(const talk_base::Buffer* p) { 610 int pt = -1; 611 ParseRtpPacket(p, NULL, &pt, NULL, NULL, NULL, NULL); 612 return pt; 613 } 614 static bool ParseRtpPacket(const talk_base::Buffer* p, bool* x, int* pt, 615 int* seqnum, uint32* tstamp, uint32* ssrc, 616 std::string* payload) { 617 talk_base::ByteBuffer buf(p->data(), p->length()); 618 uint8 u08 = 0; 619 uint16 u16 = 0; 620 uint32 u32 = 0; 621 622 // Read X and CC fields. 623 if (!buf.ReadUInt8(&u08)) return false; 624 bool extension = ((u08 & 0x10) != 0); 625 uint8 cc = (u08 & 0x0F); 626 if (x) *x = extension; 627 628 // Read PT field. 629 if (!buf.ReadUInt8(&u08)) return false; 630 if (pt) *pt = (u08 & 0x7F); 631 632 // Read Sequence Number field. 633 if (!buf.ReadUInt16(&u16)) return false; 634 if (seqnum) *seqnum = u16; 635 636 // Read Timestamp field. 637 if (!buf.ReadUInt32(&u32)) return false; 638 if (tstamp) *tstamp = u32; 639 640 // Read SSRC field. 641 if (!buf.ReadUInt32(&u32)) return false; 642 if (ssrc) *ssrc = u32; 643 644 // Skip CSRCs. 645 for (uint8 i = 0; i < cc; ++i) { 646 if (!buf.ReadUInt32(&u32)) return false; 647 } 648 649 // Skip extension header. 650 if (extension) { 651 // Read Profile-specific extension header ID 652 if (!buf.ReadUInt16(&u16)) return false; 653 654 // Read Extension header length 655 if (!buf.ReadUInt16(&u16)) return false; 656 uint16 ext_header_len = u16; 657 658 // Read Extension header 659 for (uint16 i = 0; i < ext_header_len; ++i) { 660 if (!buf.ReadUInt32(&u32)) return false; 661 } 662 } 663 664 if (payload) { 665 return buf.ReadString(payload, buf.Length()); 666 } 667 return true; 668 } 669 670 // Parse all RTCP packet, from start_index to stop_index, and count how many 671 // FIR (PT=206 and FMT=4 according to RFC 5104). If successful, set the count 672 // and return true. 673 bool CountRtcpFir(int start_index, int stop_index, int* fir_count) { 674 int count = 0; 675 for (int i = start_index; i < stop_index; ++i) { 676 talk_base::scoped_ptr<const talk_base::Buffer> p(GetRtcpPacket(i)); 677 talk_base::ByteBuffer buf(p->data(), p->length()); 678 size_t total_len = 0; 679 // The packet may be a compound RTCP packet. 680 while (total_len < p->length()) { 681 // Read FMT, type and length. 682 uint8 fmt = 0; 683 uint8 type = 0; 684 uint16 length = 0; 685 if (!buf.ReadUInt8(&fmt)) return false; 686 fmt &= 0x1F; 687 if (!buf.ReadUInt8(&type)) return false; 688 if (!buf.ReadUInt16(&length)) return false; 689 buf.Consume(length * 4); // Skip RTCP data. 690 total_len += (length + 1) * 4; 691 if ((192 == type) || ((206 == type) && (4 == fmt))) { 692 ++count; 693 } 694 } 695 } 696 697 if (fir_count) { 698 *fir_count = count; 699 } 700 return true; 701 } 702 703 void OnVideoChannelError(uint32 ssrc, 704 cricket::VideoMediaChannel::Error error) { 705 media_error_ = error; 706 } 707 708 // Test that SetSend works. 709 void SetSend() { 710 EXPECT_FALSE(channel_->sending()); 711 EXPECT_TRUE(channel_->SetCapturer(kSsrc, video_capturer_.get())); 712 EXPECT_TRUE(SetOneCodec(DefaultCodec())); 713 EXPECT_FALSE(channel_->sending()); 714 EXPECT_TRUE(SetSend(true)); 715 EXPECT_TRUE(channel_->sending()); 716 EXPECT_TRUE(SendFrame()); 717 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout); 718 EXPECT_TRUE(SetSend(false)); 719 EXPECT_FALSE(channel_->sending()); 720 } 721 // Test that SetSend fails without codecs being set. 722 void SetSendWithoutCodecs() { 723 EXPECT_FALSE(channel_->sending()); 724 EXPECT_FALSE(SetSend(true)); 725 EXPECT_FALSE(channel_->sending()); 726 } 727 // Test that we properly set the send and recv buffer sizes by the time 728 // SetSend is called. 729 void SetSendSetsTransportBufferSizes() { 730 EXPECT_TRUE(SetOneCodec(DefaultCodec())); 731 EXPECT_TRUE(SetSend(true)); 732 // TODO(sriniv): Remove or re-enable this. 733 // As part of b/8030474, send-buffer is size now controlled through 734 // portallocator flags. Its not set by channels. 735 // EXPECT_EQ(64 * 1024, network_interface_.sendbuf_size()); 736 EXPECT_EQ(64 * 1024, network_interface_.recvbuf_size()); 737 } 738 // Tests that we can send frames and the right payload type is used. 739 void Send(const cricket::VideoCodec& codec) { 740 EXPECT_TRUE(SetOneCodec(codec)); 741 EXPECT_TRUE(SetSend(true)); 742 EXPECT_TRUE(SendFrame()); 743 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout); 744 talk_base::scoped_ptr<const talk_base::Buffer> p(GetRtpPacket(0)); 745 EXPECT_EQ(codec.id, GetPayloadType(p.get())); 746 } 747 // Tests that we can send and receive frames. 748 void SendAndReceive(const cricket::VideoCodec& codec) { 749 EXPECT_TRUE(SetOneCodec(codec)); 750 EXPECT_TRUE(SetSend(true)); 751 EXPECT_TRUE(channel_->SetRender(true)); 752 EXPECT_EQ(0, renderer_.num_rendered_frames()); 753 EXPECT_TRUE(SendFrame()); 754 EXPECT_FRAME_WAIT(1, codec.width, codec.height, kTimeout); 755 talk_base::scoped_ptr<const talk_base::Buffer> p(GetRtpPacket(0)); 756 EXPECT_EQ(codec.id, GetPayloadType(p.get())); 757 } 758 // Tests that we only get a VideoRenderer::SetSize() callback when needed. 759 void SendManyResizeOnce() { 760 cricket::VideoCodec codec(DefaultCodec()); 761 EXPECT_TRUE(SetOneCodec(codec)); 762 EXPECT_TRUE(SetSend(true)); 763 EXPECT_TRUE(channel_->SetRender(true)); 764 EXPECT_EQ(0, renderer_.num_rendered_frames()); 765 EXPECT_TRUE(WaitAndSendFrame(30)); 766 EXPECT_FRAME_WAIT(1, codec.width, codec.height, kTimeout); 767 EXPECT_TRUE(WaitAndSendFrame(30)); 768 EXPECT_FRAME_WAIT(2, codec.width, codec.height, kTimeout); 769 talk_base::scoped_ptr<const talk_base::Buffer> p(GetRtpPacket(0)); 770 EXPECT_EQ(codec.id, GetPayloadType(p.get())); 771 EXPECT_EQ(1, renderer_.num_set_sizes()); 772 773 codec.width /= 2; 774 codec.height /= 2; 775 EXPECT_TRUE(SetOneCodec(codec)); 776 EXPECT_TRUE(WaitAndSendFrame(30)); 777 EXPECT_FRAME_WAIT(3, codec.width, codec.height, kTimeout); 778 EXPECT_EQ(2, renderer_.num_set_sizes()); 779 } 780 // Test that stats work properly for a 1-1 call. 781 void GetStats() { 782 SendAndReceive(DefaultCodec()); 783 cricket::VideoMediaInfo info; 784 EXPECT_TRUE(channel_->GetStats(&info)); 785 786 ASSERT_EQ(1U, info.senders.size()); 787 // TODO(whyuan): bytes_sent and bytes_rcvd are different. Are both payload? 788 EXPECT_GT(info.senders[0].bytes_sent, 0); 789 EXPECT_EQ(NumRtpPackets(), info.senders[0].packets_sent); 790 EXPECT_EQ(0.0, info.senders[0].fraction_lost); 791 EXPECT_EQ(0, info.senders[0].firs_rcvd); 792 EXPECT_EQ(0, info.senders[0].nacks_rcvd); 793 EXPECT_EQ(DefaultCodec().width, info.senders[0].frame_width); 794 EXPECT_EQ(DefaultCodec().height, info.senders[0].frame_height); 795 EXPECT_GT(info.senders[0].framerate_input, 0); 796 EXPECT_GT(info.senders[0].framerate_sent, 0); 797 798 ASSERT_EQ(1U, info.receivers.size()); 799 EXPECT_EQ(1U, info.senders[0].ssrcs().size()); 800 EXPECT_EQ(1U, info.receivers[0].ssrcs().size()); 801 EXPECT_EQ(info.senders[0].ssrcs()[0], info.receivers[0].ssrcs()[0]); 802 EXPECT_EQ(NumRtpBytes(), info.receivers[0].bytes_rcvd); 803 EXPECT_EQ(NumRtpPackets(), info.receivers[0].packets_rcvd); 804 EXPECT_EQ(0.0, info.receivers[0].fraction_lost); 805 EXPECT_EQ(0, info.receivers[0].packets_lost); 806 EXPECT_EQ(0, info.receivers[0].packets_concealed); 807 EXPECT_EQ(0, info.receivers[0].firs_sent); 808 EXPECT_EQ(0, info.receivers[0].nacks_sent); 809 EXPECT_EQ(DefaultCodec().width, info.receivers[0].frame_width); 810 EXPECT_EQ(DefaultCodec().height, info.receivers[0].frame_height); 811 EXPECT_GT(info.receivers[0].framerate_rcvd, 0); 812 EXPECT_GT(info.receivers[0].framerate_decoded, 0); 813 EXPECT_GT(info.receivers[0].framerate_output, 0); 814 } 815 // Test that stats work properly for a conf call with multiple recv streams. 816 void GetStatsMultipleRecvStreams() { 817 cricket::FakeVideoRenderer renderer1, renderer2; 818 EXPECT_TRUE(SetOneCodec(DefaultCodec())); 819 cricket::VideoOptions vmo; 820 vmo.conference_mode.Set(true); 821 EXPECT_TRUE(channel_->SetOptions(vmo)); 822 EXPECT_TRUE(SetSend(true)); 823 EXPECT_TRUE(channel_->AddRecvStream( 824 cricket::StreamParams::CreateLegacy(1))); 825 EXPECT_TRUE(channel_->AddRecvStream( 826 cricket::StreamParams::CreateLegacy(2))); 827 EXPECT_TRUE(channel_->SetRenderer(1, &renderer1)); 828 EXPECT_TRUE(channel_->SetRenderer(2, &renderer2)); 829 EXPECT_TRUE(channel_->SetRender(true)); 830 EXPECT_EQ(0, renderer1.num_rendered_frames()); 831 EXPECT_EQ(0, renderer2.num_rendered_frames()); 832 std::vector<uint32> ssrcs; 833 ssrcs.push_back(1); 834 ssrcs.push_back(2); 835 network_interface_.SetConferenceMode(true, ssrcs); 836 EXPECT_TRUE(SendFrame()); 837 EXPECT_FRAME_ON_RENDERER_WAIT( 838 renderer1, 1, DefaultCodec().width, DefaultCodec().height, kTimeout); 839 EXPECT_FRAME_ON_RENDERER_WAIT( 840 renderer2, 1, DefaultCodec().width, DefaultCodec().height, kTimeout); 841 cricket::VideoMediaInfo info; 842 EXPECT_TRUE(channel_->GetStats(&info)); 843 844 ASSERT_EQ(1U, info.senders.size()); 845 // TODO(whyuan): bytes_sent and bytes_rcvd are different. Are both payload? 846 EXPECT_GT(info.senders[0].bytes_sent, 0); 847 EXPECT_EQ(NumRtpPackets(), info.senders[0].packets_sent); 848 EXPECT_EQ(0.0, info.senders[0].fraction_lost); 849 EXPECT_EQ(0, info.senders[0].firs_rcvd); 850 EXPECT_EQ(0, info.senders[0].nacks_rcvd); 851 EXPECT_EQ(DefaultCodec().width, info.senders[0].frame_width); 852 EXPECT_EQ(DefaultCodec().height, info.senders[0].frame_height); 853 EXPECT_GT(info.senders[0].framerate_input, 0); 854 EXPECT_GT(info.senders[0].framerate_sent, 0); 855 856 ASSERT_EQ(2U, info.receivers.size()); 857 for (size_t i = 0; i < info.receivers.size(); ++i) { 858 EXPECT_EQ(1U, info.receivers[i].ssrcs().size()); 859 EXPECT_EQ(i + 1, info.receivers[i].ssrcs()[0]); 860 EXPECT_EQ(NumRtpBytes(), info.receivers[i].bytes_rcvd); 861 EXPECT_EQ(NumRtpPackets(), info.receivers[i].packets_rcvd); 862 EXPECT_EQ(0.0, info.receivers[i].fraction_lost); 863 EXPECT_EQ(0, info.receivers[i].packets_lost); 864 EXPECT_EQ(0, info.receivers[i].packets_concealed); 865 EXPECT_EQ(0, info.receivers[i].firs_sent); 866 EXPECT_EQ(0, info.receivers[i].nacks_sent); 867 EXPECT_EQ(DefaultCodec().width, info.receivers[i].frame_width); 868 EXPECT_EQ(DefaultCodec().height, info.receivers[i].frame_height); 869 EXPECT_GT(info.receivers[i].framerate_rcvd, 0); 870 EXPECT_GT(info.receivers[i].framerate_decoded, 0); 871 EXPECT_GT(info.receivers[i].framerate_output, 0); 872 } 873 } 874 // Test that stats work properly for a conf call with multiple send streams. 875 void GetStatsMultipleSendStreams() { 876 // Normal setup; note that we set the SSRC explicitly to ensure that 877 // it will come first in the senders map. 878 EXPECT_TRUE(SetOneCodec(DefaultCodec())); 879 cricket::VideoOptions vmo; 880 vmo.conference_mode.Set(true); 881 EXPECT_TRUE(channel_->SetOptions(vmo)); 882 EXPECT_TRUE(channel_->AddRecvStream( 883 cricket::StreamParams::CreateLegacy(1234))); 884 channel_->UpdateAspectRatio(640, 400); 885 EXPECT_TRUE(SetSend(true)); 886 EXPECT_TRUE(channel_->SetRender(true)); 887 EXPECT_TRUE(SendFrame()); 888 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout); 889 EXPECT_FRAME_WAIT(1, DefaultCodec().width, DefaultCodec().height, kTimeout); 890 891 // Add an additional capturer, and hook up a renderer to receive it. 892 cricket::FakeVideoRenderer renderer1; 893 talk_base::scoped_ptr<cricket::FakeVideoCapturer> capturer( 894 new cricket::FakeVideoCapturer); 895 capturer->SetScreencast(true); 896 cricket::VideoFormat format(1024, 768, 897 cricket::VideoFormat::FpsToInterval(5), 0); 898 EXPECT_EQ(cricket::CS_RUNNING, capturer->Start(format)); 899 EXPECT_TRUE(channel_->AddSendStream( 900 cricket::StreamParams::CreateLegacy(5678))); 901 EXPECT_TRUE(channel_->SetCapturer(5678, capturer.get())); 902 EXPECT_TRUE(channel_->AddRecvStream( 903 cricket::StreamParams::CreateLegacy(5678))); 904 EXPECT_TRUE(channel_->SetRenderer(5678, &renderer1)); 905 EXPECT_TRUE(capturer->CaptureCustomFrame(1024, 768, cricket::FOURCC_I420)); 906 EXPECT_FRAME_ON_RENDERER_WAIT(renderer1, 1, 1024, 768, kTimeout); 907 908 // Get stats, and make sure they are correct for two senders. 909 cricket::VideoMediaInfo info; 910 EXPECT_TRUE(channel_->GetStats(&info)); 911 ASSERT_EQ(2U, info.senders.size()); 912 EXPECT_EQ(NumRtpPackets(), 913 info.senders[0].packets_sent + info.senders[1].packets_sent); 914 EXPECT_EQ(1U, info.senders[0].ssrcs().size()); 915 EXPECT_EQ(1234U, info.senders[0].ssrcs()[0]); 916 EXPECT_EQ(DefaultCodec().width, info.senders[0].frame_width); 917 EXPECT_EQ(DefaultCodec().height, info.senders[0].frame_height); 918 EXPECT_EQ(1U, info.senders[1].ssrcs().size()); 919 EXPECT_EQ(5678U, info.senders[1].ssrcs()[0]); 920 EXPECT_EQ(1024, info.senders[1].frame_width); 921 EXPECT_EQ(768, info.senders[1].frame_height); 922 // The capturer must be unregistered here as it runs out of it's scope next. 923 EXPECT_TRUE(channel_->SetCapturer(5678, NULL)); 924 } 925 926 // Test that we can set the bandwidth to auto or a specific value. 927 void SetSendBandwidth() { 928 EXPECT_TRUE(channel_->SetSendBandwidth(true, -1)); 929 EXPECT_TRUE(channel_->SetSendBandwidth(true, 128 * 1024)); 930 EXPECT_TRUE(channel_->SetSendBandwidth(false, -1)); 931 EXPECT_TRUE(channel_->SetSendBandwidth(false, 128 * 1024)); 932 } 933 // Test that we can set the SSRC for the default send source. 934 void SetSendSsrc() { 935 EXPECT_TRUE(SetDefaultCodec()); 936 EXPECT_TRUE(SetSend(true)); 937 EXPECT_TRUE(SendFrame()); 938 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout); 939 uint32 ssrc = 0; 940 talk_base::scoped_ptr<const talk_base::Buffer> p(GetRtpPacket(0)); 941 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL); 942 EXPECT_EQ(kSsrc, ssrc); 943 EXPECT_EQ(NumRtpPackets(), NumRtpPackets(ssrc)); 944 EXPECT_EQ(NumRtpBytes(), NumRtpBytes(ssrc)); 945 EXPECT_EQ(1, NumSentSsrcs()); 946 EXPECT_EQ(0, NumRtpPackets(kSsrc - 1)); 947 EXPECT_EQ(0, NumRtpBytes(kSsrc - 1)); 948 } 949 // Test that we can set the SSRC even after codecs are set. 950 void SetSendSsrcAfterSetCodecs() { 951 // Remove stream added in Setup. 952 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc)); 953 EXPECT_TRUE(SetDefaultCodec()); 954 EXPECT_TRUE(channel_->AddSendStream( 955 cricket::StreamParams::CreateLegacy(999))); 956 EXPECT_TRUE(channel_->SetCapturer(999u, video_capturer_.get())); 957 EXPECT_TRUE(SetSend(true)); 958 EXPECT_TRUE(WaitAndSendFrame(0)); 959 EXPECT_TRUE_WAIT(NumRtpPackets() > 0, kTimeout); 960 uint32 ssrc = 0; 961 talk_base::scoped_ptr<const talk_base::Buffer> p(GetRtpPacket(0)); 962 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL); 963 EXPECT_EQ(999u, ssrc); 964 EXPECT_EQ(NumRtpPackets(), NumRtpPackets(ssrc)); 965 EXPECT_EQ(NumRtpBytes(), NumRtpBytes(ssrc)); 966 EXPECT_EQ(1, NumSentSsrcs()); 967 EXPECT_EQ(0, NumRtpPackets(kSsrc)); 968 EXPECT_EQ(0, NumRtpBytes(kSsrc)); 969 } 970 // Test that we can set the default video renderer before and after 971 // media is received. 972 void SetRenderer() { 973 uint8 data1[] = { 974 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 975 }; 976 977 talk_base::Buffer packet1(data1, sizeof(data1)); 978 talk_base::SetBE32(packet1.data() + 8, kSsrc); 979 channel_->SetRenderer(0, NULL); 980 EXPECT_TRUE(SetDefaultCodec()); 981 EXPECT_TRUE(SetSend(true)); 982 EXPECT_TRUE(channel_->SetRender(true)); 983 EXPECT_EQ(0, renderer_.num_rendered_frames()); 984 channel_->OnPacketReceived(&packet1, talk_base::PacketTime()); 985 SetRendererAsDefault(); 986 EXPECT_TRUE(SendFrame()); 987 EXPECT_FRAME_WAIT(1, DefaultCodec().width, DefaultCodec().height, kTimeout); 988 } 989 990 // Tests empty StreamParams is rejected. 991 void RejectEmptyStreamParams() { 992 // Remove the send stream that was added during Setup. 993 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc)); 994 995 cricket::StreamParams empty; 996 EXPECT_FALSE(channel_->AddSendStream(empty)); 997 EXPECT_TRUE(channel_->AddSendStream( 998 cricket::StreamParams::CreateLegacy(789u))); 999 } 1000 1001 // Tests setting up and configuring a send stream. 1002 void AddRemoveSendStreams() { 1003 EXPECT_TRUE(SetOneCodec(DefaultCodec())); 1004 EXPECT_TRUE(SetSend(true)); 1005 EXPECT_TRUE(channel_->SetRender(true)); 1006 EXPECT_TRUE(SendFrame()); 1007 EXPECT_FRAME_WAIT(1, DefaultCodec().width, DefaultCodec().height, kTimeout); 1008 EXPECT_GE(2, NumRtpPackets()); 1009 uint32 ssrc = 0; 1010 size_t last_packet = NumRtpPackets() - 1; 1011 talk_base::scoped_ptr<const talk_base::Buffer> 1012 p(GetRtpPacket(static_cast<int>(last_packet))); 1013 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL); 1014 EXPECT_EQ(kSsrc, ssrc); 1015 1016 // Remove the send stream that was added during Setup. 1017 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc)); 1018 int rtp_packets = NumRtpPackets(); 1019 1020 EXPECT_TRUE(channel_->AddSendStream( 1021 cricket::StreamParams::CreateLegacy(789u))); 1022 EXPECT_TRUE(channel_->SetCapturer(789u, video_capturer_.get())); 1023 EXPECT_EQ(rtp_packets, NumRtpPackets()); 1024 // Wait 30ms to guarantee the engine does not drop the frame. 1025 EXPECT_TRUE(WaitAndSendFrame(30)); 1026 EXPECT_TRUE_WAIT(NumRtpPackets() > rtp_packets, kTimeout); 1027 1028 last_packet = NumRtpPackets() - 1; 1029 p.reset(GetRtpPacket(static_cast<int>(last_packet))); 1030 ParseRtpPacket(p.get(), NULL, NULL, NULL, NULL, &ssrc, NULL); 1031 EXPECT_EQ(789u, ssrc); 1032 } 1033 1034 // Tests adding streams already exists returns false. 1035 void AddRecvStreamsAlreadyExist() { 1036 cricket::VideoOptions vmo; 1037 vmo.conference_mode.Set(true); 1038 EXPECT_TRUE(channel_->SetOptions(vmo)); 1039 1040 EXPECT_FALSE(channel_->AddRecvStream( 1041 cricket::StreamParams::CreateLegacy(0))); 1042 1043 EXPECT_TRUE(channel_->AddRecvStream( 1044 cricket::StreamParams::CreateLegacy(1))); 1045 EXPECT_FALSE(channel_->AddRecvStream( 1046 cricket::StreamParams::CreateLegacy(1))); 1047 1048 EXPECT_TRUE(channel_->RemoveRecvStream(1)); 1049 EXPECT_FALSE(channel_->AddRecvStream( 1050 cricket::StreamParams::CreateLegacy(0))); 1051 EXPECT_TRUE(channel_->AddRecvStream( 1052 cricket::StreamParams::CreateLegacy(1))); 1053 } 1054 1055 // Tests setting up and configuring multiple incoming streams. 1056 void AddRemoveRecvStreams() { 1057 cricket::FakeVideoRenderer renderer1, renderer2; 1058 cricket::VideoOptions vmo; 1059 vmo.conference_mode.Set(true); 1060 EXPECT_TRUE(channel_->SetOptions(vmo)); 1061 // Ensure we can't set the renderer on a non-existent stream. 1062 EXPECT_FALSE(channel_->SetRenderer(1, &renderer1)); 1063 EXPECT_FALSE(channel_->SetRenderer(2, &renderer2)); 1064 cricket::VideoRenderer* renderer; 1065 EXPECT_FALSE(channel_->GetRenderer(1, &renderer)); 1066 EXPECT_FALSE(channel_->GetRenderer(2, &renderer)); 1067 1068 // Ensure we can add streams. 1069 EXPECT_TRUE(channel_->AddRecvStream( 1070 cricket::StreamParams::CreateLegacy(1))); 1071 EXPECT_TRUE(channel_->AddRecvStream( 1072 cricket::StreamParams::CreateLegacy(2))); 1073 EXPECT_TRUE(channel_->GetRenderer(1, &renderer)); 1074 // Verify the first AddRecvStream hook up to the default renderer. 1075 EXPECT_EQ(&renderer_, renderer); 1076 EXPECT_TRUE(channel_->GetRenderer(2, &renderer)); 1077 EXPECT_TRUE(NULL == renderer); 1078 1079 // Ensure we can now set the renderers. 1080 EXPECT_TRUE(channel_->SetRenderer(1, &renderer1)); 1081 EXPECT_TRUE(channel_->SetRenderer(2, &renderer2)); 1082 EXPECT_TRUE(channel_->GetRenderer(1, &renderer)); 1083 EXPECT_TRUE(&renderer1 == renderer); 1084 EXPECT_TRUE(channel_->GetRenderer(2, &renderer)); 1085 EXPECT_TRUE(&renderer2 == renderer); 1086 1087 // Ensure we can change the renderers if needed. 1088 EXPECT_TRUE(channel_->SetRenderer(1, &renderer2)); 1089 EXPECT_TRUE(channel_->SetRenderer(2, &renderer1)); 1090 EXPECT_TRUE(channel_->GetRenderer(1, &renderer)); 1091 EXPECT_TRUE(&renderer2 == renderer); 1092 EXPECT_TRUE(channel_->GetRenderer(2, &renderer)); 1093 EXPECT_TRUE(&renderer1 == renderer); 1094 1095 EXPECT_TRUE(channel_->RemoveRecvStream(2)); 1096 EXPECT_TRUE(channel_->RemoveRecvStream(1)); 1097 EXPECT_FALSE(channel_->GetRenderer(1, &renderer)); 1098 EXPECT_FALSE(channel_->GetRenderer(2, &renderer)); 1099 } 1100 1101 // Tests setting up and configuring multiple incoming streams in a 1102 // non-conference call. 1103 void AddRemoveRecvStreamsNoConference() { 1104 cricket::FakeVideoRenderer renderer1, renderer2; 1105 // Ensure we can't set the renderer on a non-existent stream. 1106 EXPECT_FALSE(channel_->SetRenderer(1, &renderer1)); 1107 EXPECT_FALSE(channel_->SetRenderer(2, &renderer2)); 1108 cricket::VideoRenderer* renderer; 1109 EXPECT_FALSE(channel_->GetRenderer(1, &renderer)); 1110 EXPECT_FALSE(channel_->GetRenderer(2, &renderer)); 1111 1112 // Ensure we can add streams. 1113 EXPECT_TRUE(channel_->AddRecvStream( 1114 cricket::StreamParams::CreateLegacy(1))); 1115 EXPECT_TRUE(channel_->AddRecvStream( 1116 cricket::StreamParams::CreateLegacy(2))); 1117 EXPECT_TRUE(channel_->GetRenderer(1, &renderer)); 1118 // Verify the first AddRecvStream hook up to the default renderer. 1119 EXPECT_EQ(&renderer_, renderer); 1120 EXPECT_TRUE(channel_->GetRenderer(2, &renderer)); 1121 EXPECT_TRUE(NULL == renderer); 1122 1123 // Ensure we can now set the renderers. 1124 EXPECT_TRUE(channel_->SetRenderer(1, &renderer1)); 1125 EXPECT_TRUE(channel_->SetRenderer(2, &renderer2)); 1126 EXPECT_TRUE(channel_->GetRenderer(1, &renderer)); 1127 EXPECT_TRUE(&renderer1 == renderer); 1128 EXPECT_TRUE(channel_->GetRenderer(2, &renderer)); 1129 EXPECT_TRUE(&renderer2 == renderer); 1130 1131 // Ensure we can change the renderers if needed. 1132 EXPECT_TRUE(channel_->SetRenderer(1, &renderer2)); 1133 EXPECT_TRUE(channel_->SetRenderer(2, &renderer1)); 1134 EXPECT_TRUE(channel_->GetRenderer(1, &renderer)); 1135 EXPECT_TRUE(&renderer2 == renderer); 1136 EXPECT_TRUE(channel_->GetRenderer(2, &renderer)); 1137 EXPECT_TRUE(&renderer1 == renderer); 1138 1139 EXPECT_TRUE(channel_->RemoveRecvStream(2)); 1140 EXPECT_TRUE(channel_->RemoveRecvStream(1)); 1141 EXPECT_FALSE(channel_->GetRenderer(1, &renderer)); 1142 EXPECT_FALSE(channel_->GetRenderer(2, &renderer)); 1143 } 1144 1145 // Test that no frames are rendered after the receive stream have been 1146 // removed. 1147 void AddRemoveRecvStreamAndRender() { 1148 cricket::FakeVideoRenderer renderer1; 1149 EXPECT_TRUE(SetDefaultCodec()); 1150 EXPECT_TRUE(SetSend(true)); 1151 EXPECT_TRUE(channel_->SetRender(true)); 1152 EXPECT_TRUE(channel_->AddRecvStream( 1153 cricket::StreamParams::CreateLegacy(kSsrc))); 1154 EXPECT_TRUE(channel_->SetRenderer(kSsrc, &renderer1)); 1155 1156 EXPECT_TRUE(SendFrame()); 1157 EXPECT_FRAME_ON_RENDERER_WAIT( 1158 renderer1, 1, DefaultCodec().width, DefaultCodec().height, kTimeout); 1159 EXPECT_TRUE(channel_->RemoveRecvStream(kSsrc)); 1160 // Send three more frames. This is to avoid that the test might be flaky 1161 // due to frame dropping. 1162 for (size_t i = 0; i < 3; ++i) 1163 EXPECT_TRUE(WaitAndSendFrame(100)); 1164 1165 // Test that no more frames have been rendered. 1166 EXPECT_EQ(1, renderer1.num_rendered_frames()); 1167 1168 // Re-add the stream again and make sure it renders. 1169 EXPECT_TRUE(channel_->AddRecvStream( 1170 cricket::StreamParams::CreateLegacy(kSsrc))); 1171 // Force the next frame to be a key frame to make the receiving 1172 // decoder happy. 1173 EXPECT_TRUE(channel_->SendIntraFrame()); 1174 1175 EXPECT_TRUE(channel_->SetRenderer(kSsrc, &renderer1)); 1176 EXPECT_TRUE(SendFrame()); 1177 // Because the default channel is used, RemoveRecvStream above is not going 1178 // to delete the channel. As a result the engine will continue to receive 1179 // and decode the 3 frames sent above. So it is possible we will receive 1180 // some (e.g. 1) of these 3 frames after the renderer is set again. 1181 EXPECT_GT_FRAME_ON_RENDERER_WAIT( 1182 renderer1, 2, DefaultCodec().width, DefaultCodec().height, kTimeout); 1183 } 1184 1185 // Tests the behavior of incoming streams in a conference scenario. 1186 void SimulateConference() { 1187 cricket::FakeVideoRenderer renderer1, renderer2; 1188 EXPECT_TRUE(SetDefaultCodec()); 1189 cricket::VideoOptions vmo; 1190 vmo.conference_mode.Set(true); 1191 EXPECT_TRUE(channel_->SetOptions(vmo)); 1192 EXPECT_TRUE(SetSend(true)); 1193 EXPECT_TRUE(channel_->SetRender(true)); 1194 EXPECT_TRUE(channel_->AddRecvStream( 1195 cricket::StreamParams::CreateLegacy(1))); 1196 EXPECT_TRUE(channel_->AddRecvStream( 1197 cricket::StreamParams::CreateLegacy(2))); 1198 EXPECT_TRUE(channel_->SetRenderer(1, &renderer1)); 1199 EXPECT_TRUE(channel_->SetRenderer(2, &renderer2)); 1200 EXPECT_EQ(0, renderer1.num_rendered_frames()); 1201 EXPECT_EQ(0, renderer2.num_rendered_frames()); 1202 std::vector<uint32> ssrcs; 1203 ssrcs.push_back(1); 1204 ssrcs.push_back(2); 1205 network_interface_.SetConferenceMode(true, ssrcs); 1206 EXPECT_TRUE(SendFrame()); 1207 EXPECT_FRAME_ON_RENDERER_WAIT( 1208 renderer1, 1, DefaultCodec().width, DefaultCodec().height, kTimeout); 1209 EXPECT_FRAME_ON_RENDERER_WAIT( 1210 renderer2, 1, DefaultCodec().width, DefaultCodec().height, kTimeout); 1211 1212 talk_base::scoped_ptr<const talk_base::Buffer> p(GetRtpPacket(0)); 1213 EXPECT_EQ(DefaultCodec().id, GetPayloadType(p.get())); 1214 EXPECT_EQ(DefaultCodec().width, renderer1.width()); 1215 EXPECT_EQ(DefaultCodec().height, renderer1.height()); 1216 EXPECT_EQ(DefaultCodec().width, renderer2.width()); 1217 EXPECT_EQ(DefaultCodec().height, renderer2.height()); 1218 EXPECT_TRUE(channel_->RemoveRecvStream(2)); 1219 EXPECT_TRUE(channel_->RemoveRecvStream(1)); 1220 } 1221 1222 // Tests that we can add and remove capturers and frames are sent out properly 1223 void AddRemoveCapturer() { 1224 const cricket::VideoCodec codec(DefaultCodec()); 1225 const int time_between_send = TimeBetweenSend(codec); 1226 EXPECT_TRUE(SetDefaultCodec()); 1227 EXPECT_TRUE(SetSend(true)); 1228 EXPECT_TRUE(channel_->SetRender(true)); 1229 EXPECT_EQ(0, renderer_.num_rendered_frames()); 1230 EXPECT_TRUE(SendFrame()); 1231 EXPECT_FRAME_WAIT(1, codec.width, codec.height, kTimeout); 1232 talk_base::scoped_ptr<cricket::FakeVideoCapturer> capturer( 1233 new cricket::FakeVideoCapturer); 1234 capturer->SetScreencast(true); 1235 cricket::VideoFormat format(1024, 768, 1236 cricket::VideoFormat::FpsToInterval(30), 0); 1237 EXPECT_EQ(cricket::CS_RUNNING, capturer->Start(format)); 1238 // All capturers start generating frames with the same timestamp. ViE does 1239 // not allow the same timestamp to be used. Capture one frame before 1240 // associating the capturer with the channel. 1241 EXPECT_TRUE(capturer->CaptureCustomFrame(format.width, format.height, 1242 cricket::FOURCC_I420)); 1243 1244 int captured_frames = 1; 1245 for (int iterations = 0; iterations < 2; ++iterations) { 1246 EXPECT_TRUE(channel_->SetCapturer(kSsrc, capturer.get())); 1247 talk_base::Thread::Current()->ProcessMessages(time_between_send); 1248 EXPECT_TRUE(capturer->CaptureCustomFrame(format.width, format.height, 1249 cricket::FOURCC_I420)); 1250 ++captured_frames; 1251 // Wait until frame of right size is captured. 1252 EXPECT_TRUE_WAIT(renderer_.num_rendered_frames() >= captured_frames && 1253 format.width == renderer_.width() && 1254 format.height == renderer_.height() && 1255 !renderer_.black_frame(), kTimeout); 1256 EXPECT_GE(renderer_.num_rendered_frames(), captured_frames); 1257 EXPECT_EQ(format.width, renderer_.width()); 1258 EXPECT_EQ(format.height, renderer_.height()); 1259 captured_frames = renderer_.num_rendered_frames() + 1; 1260 EXPECT_FALSE(renderer_.black_frame()); 1261 EXPECT_TRUE(channel_->SetCapturer(kSsrc, NULL)); 1262 // Make sure a black frame is generated within the specified timeout. 1263 // The black frame should be the resolution of the send codec. 1264 EXPECT_TRUE_WAIT(renderer_.num_rendered_frames() >= captured_frames && 1265 codec.width == renderer_.width() && 1266 codec.height == renderer_.height() && 1267 renderer_.black_frame(), kTimeout); 1268 EXPECT_GE(renderer_.num_rendered_frames(), captured_frames); 1269 EXPECT_EQ(codec.width, renderer_.width()); 1270 EXPECT_EQ(codec.height, renderer_.height()); 1271 EXPECT_TRUE(renderer_.black_frame()); 1272 1273 // The black frame has the same timestamp as the next frame since it's 1274 // timestamp is set to the last frame's timestamp + interval. WebRTC will 1275 // not render a frame with the same timestamp so capture another frame 1276 // with the frame capturer to increment the next frame's timestamp. 1277 EXPECT_TRUE(capturer->CaptureCustomFrame(format.width, format.height, 1278 cricket::FOURCC_I420)); 1279 } 1280 } 1281 1282 // Tests that if RemoveCapturer is called without a capturer ever being 1283 // added, the plugin shouldn't crash (and no black frame should be sent). 1284 void RemoveCapturerWithoutAdd() { 1285 EXPECT_TRUE(SetOneCodec(DefaultCodec())); 1286 EXPECT_TRUE(SetSend(true)); 1287 EXPECT_TRUE(channel_->SetRender(true)); 1288 EXPECT_EQ(0, renderer_.num_rendered_frames()); 1289 EXPECT_TRUE(SendFrame()); 1290 EXPECT_FRAME_WAIT(1, 640, 400, kTimeout); 1291 // Remove the capturer. 1292 EXPECT_TRUE(channel_->SetCapturer(kSsrc, NULL)); 1293 // Wait for one black frame for removing the capturer. 1294 EXPECT_FRAME_WAIT(2, 640, 400, kTimeout); 1295 1296 // No capturer was added, so this RemoveCapturer should 1297 // fail. 1298 EXPECT_FALSE(channel_->SetCapturer(kSsrc, NULL)); 1299 talk_base::Thread::Current()->ProcessMessages(300); 1300 // Verify no more frames were sent. 1301 EXPECT_EQ(2, renderer_.num_rendered_frames()); 1302 } 1303 1304 // Tests that we can add and remove capturer as unique sources. 1305 void AddRemoveCapturerMultipleSources() { 1306 // WebRTC implementation will drop frames if pushed to quickly. Wait the 1307 // interval time to avoid that. 1308 const cricket::VideoFormat send_format( 1309 1024, 1310 768, 1311 cricket::VideoFormat::FpsToInterval(30), 1312 0); 1313 // WebRTC implementation will drop frames if pushed to quickly. Wait the 1314 // interval time to avoid that. 1315 // Set up the stream associated with the engine. 1316 EXPECT_TRUE(channel_->AddRecvStream( 1317 cricket::StreamParams::CreateLegacy(kSsrc))); 1318 EXPECT_TRUE(channel_->SetRenderer(kSsrc, &renderer_)); 1319 cricket::VideoFormat capture_format; // default format 1320 capture_format.interval = cricket::VideoFormat::FpsToInterval(30); 1321 // Set up additional stream 1. 1322 cricket::FakeVideoRenderer renderer1; 1323 EXPECT_FALSE(channel_->SetRenderer(1, &renderer1)); 1324 EXPECT_TRUE(channel_->AddRecvStream( 1325 cricket::StreamParams::CreateLegacy(1))); 1326 EXPECT_TRUE(channel_->SetRenderer(1, &renderer1)); 1327 EXPECT_TRUE(channel_->AddSendStream( 1328 cricket::StreamParams::CreateLegacy(1))); 1329 talk_base::scoped_ptr<cricket::FakeVideoCapturer> capturer1( 1330 new cricket::FakeVideoCapturer); 1331 capturer1->SetScreencast(true); 1332 EXPECT_EQ(cricket::CS_RUNNING, capturer1->Start(capture_format)); 1333 // Set up additional stream 2. 1334 cricket::FakeVideoRenderer renderer2; 1335 EXPECT_FALSE(channel_->SetRenderer(2, &renderer2)); 1336 EXPECT_TRUE(channel_->AddRecvStream( 1337 cricket::StreamParams::CreateLegacy(2))); 1338 EXPECT_TRUE(channel_->SetRenderer(2, &renderer2)); 1339 EXPECT_TRUE(channel_->AddSendStream( 1340 cricket::StreamParams::CreateLegacy(2))); 1341 talk_base::scoped_ptr<cricket::FakeVideoCapturer> capturer2( 1342 new cricket::FakeVideoCapturer); 1343 capturer2->SetScreencast(true); 1344 EXPECT_EQ(cricket::CS_RUNNING, capturer2->Start(capture_format)); 1345 // State for all the streams. 1346 EXPECT_TRUE(SetOneCodec(DefaultCodec())); 1347 // A limitation in the lmi implementation requires that SetCapturer() is 1348 // called after SetOneCodec(). 1349 // TODO(hellner): this seems like an unnecessary constraint, fix it. 1350 EXPECT_TRUE(channel_->SetCapturer(1, capturer1.get())); 1351 EXPECT_TRUE(channel_->SetCapturer(2, capturer2.get())); 1352 EXPECT_TRUE(SetSend(true)); 1353 EXPECT_TRUE(channel_->SetRender(true)); 1354 // Test capturer associated with engine. 1355 EXPECT_TRUE(capturer1->CaptureCustomFrame(1024, 768, cricket::FOURCC_I420)); 1356 EXPECT_FRAME_ON_RENDERER_WAIT(renderer1, 1, 1024, 768, kTimeout); 1357 // Capture a frame with additional capturer2, frames should be received 1358 EXPECT_TRUE(capturer2->CaptureCustomFrame(1024, 768, cricket::FOURCC_I420)); 1359 EXPECT_FRAME_ON_RENDERER_WAIT(renderer2, 1, 1024, 768, kTimeout); 1360 // Successfully remove the capturer. 1361 EXPECT_TRUE(channel_->SetCapturer(kSsrc, NULL)); 1362 // Fail to re-remove the capturer. 1363 EXPECT_FALSE(channel_->SetCapturer(kSsrc, NULL)); 1364 // The capturers must be unregistered here as it runs out of it's scope 1365 // next. 1366 EXPECT_TRUE(channel_->SetCapturer(1, NULL)); 1367 EXPECT_TRUE(channel_->SetCapturer(2, NULL)); 1368 } 1369 1370 void HighAspectHighHeightCapturer() { 1371 const int kWidth = 80; 1372 const int kHeight = 10000; 1373 const int kScaledWidth = 20; 1374 const int kScaledHeight = 2500; 1375 1376 cricket::VideoCodec codec(DefaultCodec()); 1377 EXPECT_TRUE(SetOneCodec(codec)); 1378 EXPECT_TRUE(SetSend(true)); 1379 1380 cricket::FakeVideoRenderer renderer; 1381 EXPECT_TRUE(channel_->AddRecvStream( 1382 cricket::StreamParams::CreateLegacy(kSsrc))); 1383 EXPECT_TRUE(channel_->SetRenderer(kSsrc, &renderer)); 1384 EXPECT_TRUE(channel_->SetRender(true)); 1385 EXPECT_EQ(0, renderer.num_rendered_frames()); 1386 1387 EXPECT_TRUE(SendFrame()); 1388 EXPECT_GT_FRAME_ON_RENDERER_WAIT( 1389 renderer, 1, codec.width, codec.height, kTimeout); 1390 1391 // Registering an external capturer is currently the same as screen casting 1392 // (update the test when this changes). 1393 talk_base::scoped_ptr<cricket::FakeVideoCapturer> capturer( 1394 new cricket::FakeVideoCapturer); 1395 capturer->SetScreencast(true); 1396 const std::vector<cricket::VideoFormat>* formats = 1397 capturer->GetSupportedFormats(); 1398 cricket::VideoFormat capture_format = (*formats)[0]; 1399 EXPECT_EQ(cricket::CS_RUNNING, capturer->Start(capture_format)); 1400 // Capture frame to not get same frame timestamps as previous capturer. 1401 capturer->CaptureFrame(); 1402 EXPECT_TRUE(channel_->SetCapturer(kSsrc, capturer.get())); 1403 EXPECT_TRUE(talk_base::Thread::Current()->ProcessMessages(30)); 1404 EXPECT_TRUE(capturer->CaptureCustomFrame(kWidth, kHeight, 1405 cricket::FOURCC_ARGB)); 1406 EXPECT_TRUE(capturer->CaptureFrame()); 1407 EXPECT_GT_FRAME_ON_RENDERER_WAIT( 1408 renderer, 2, kScaledWidth, kScaledHeight, kTimeout); 1409 EXPECT_TRUE(channel_->SetCapturer(kSsrc, NULL)); 1410 } 1411 1412 // Tests that we can adapt video resolution with 16:10 aspect ratio properly. 1413 void AdaptResolution16x10() { 1414 cricket::VideoCodec codec(DefaultCodec()); 1415 codec.width = 640; 1416 codec.height = 400; 1417 SendAndReceive(codec); 1418 codec.width /= 2; 1419 codec.height /= 2; 1420 // Adapt the resolution. 1421 EXPECT_TRUE(SetOneCodec(codec)); 1422 EXPECT_TRUE(WaitAndSendFrame(30)); 1423 EXPECT_FRAME_WAIT(2, codec.width, codec.height, kTimeout); 1424 } 1425 // Tests that we can adapt video resolution with 4:3 aspect ratio properly. 1426 void AdaptResolution4x3() { 1427 cricket::VideoCodec codec(DefaultCodec()); 1428 codec.width = 640; 1429 codec.height = 400; 1430 SendAndReceive(codec); 1431 codec.width /= 2; 1432 codec.height /= 2; 1433 // Adapt the resolution. 1434 EXPECT_TRUE(SetOneCodec(codec)); 1435 EXPECT_TRUE(WaitAndSendFrame(30)); 1436 EXPECT_FRAME_WAIT(2, codec.width, codec.height, kTimeout); 1437 } 1438 // Tests that we can drop all frames properly. 1439 void AdaptDropAllFrames() { 1440 // Set the channel codec's resolution to 0, which will require the adapter 1441 // to drop all frames. 1442 cricket::VideoCodec codec(DefaultCodec()); 1443 codec.width = codec.height = codec.framerate = 0; 1444 EXPECT_TRUE(SetOneCodec(codec)); 1445 EXPECT_TRUE(SetSend(true)); 1446 EXPECT_TRUE(channel_->SetRender(true)); 1447 EXPECT_EQ(0, renderer_.num_rendered_frames()); 1448 EXPECT_TRUE(SendFrame()); 1449 EXPECT_TRUE(SendFrame()); 1450 talk_base::Thread::Current()->ProcessMessages(500); 1451 EXPECT_EQ(0, renderer_.num_rendered_frames()); 1452 } 1453 // Tests that we can reduce the frame rate on demand properly. 1454 // TODO(fbarchard): This test is flakey on pulse. Fix and re-enable 1455 void AdaptFramerate() { 1456 cricket::VideoCodec codec(DefaultCodec()); 1457 int frame_count = 0; 1458 // The capturer runs at 30 fps. The channel requires 30 fps. 1459 EXPECT_TRUE(SetOneCodec(codec)); 1460 EXPECT_TRUE(SetSend(true)); 1461 EXPECT_TRUE(channel_->SetRender(true)); 1462 EXPECT_EQ(frame_count, renderer_.num_rendered_frames()); 1463 EXPECT_TRUE(WaitAndSendFrame(0)); // Should be rendered. 1464 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be rendered. 1465 frame_count += 2; 1466 EXPECT_FRAME_WAIT(frame_count, codec.width, codec.height, kTimeout); 1467 talk_base::scoped_ptr<const talk_base::Buffer> p(GetRtpPacket(0)); 1468 EXPECT_EQ(codec.id, GetPayloadType(p.get())); 1469 1470 // The channel requires 15 fps. 1471 codec.framerate = 15; 1472 EXPECT_TRUE(SetOneCodec(codec)); 1473 EXPECT_TRUE(WaitAndSendFrame(0)); // Should be rendered. 1474 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be dropped. 1475 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be rendered. 1476 frame_count += 2; 1477 EXPECT_EQ_WAIT(frame_count, renderer_.num_rendered_frames(), kTimeout); 1478 1479 // The channel requires 10 fps. 1480 codec.framerate = 10; 1481 EXPECT_TRUE(SetOneCodec(codec)); 1482 EXPECT_TRUE(WaitAndSendFrame(0)); // Should be rendered. 1483 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be dropped. 1484 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be dropped. 1485 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be rendered. 1486 frame_count += 2; 1487 EXPECT_EQ_WAIT(frame_count, renderer_.num_rendered_frames(), kTimeout); 1488 1489 // The channel requires 8 fps. The adapter adapts to 10 fps, which is the 1490 // closest factor of 30. 1491 codec.framerate = 8; 1492 EXPECT_TRUE(SetOneCodec(codec)); 1493 EXPECT_TRUE(WaitAndSendFrame(0)); // Should be rendered. 1494 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be dropped. 1495 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be dropped. 1496 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be rendered. 1497 frame_count += 2; 1498 EXPECT_EQ_WAIT(frame_count, renderer_.num_rendered_frames(), kTimeout); 1499 } 1500 // Tests that we can set the send stream format properly. 1501 void SetSendStreamFormat() { 1502 cricket::VideoCodec codec(DefaultCodec()); 1503 SendAndReceive(codec); 1504 int frame_count = 1; 1505 EXPECT_FRAME_WAIT(frame_count, codec.width, codec.height, kTimeout); 1506 1507 // Adapt the resolution and frame rate to half. 1508 cricket::VideoFormat format( 1509 codec.width / 2, 1510 codec.height / 2, 1511 cricket::VideoFormat::FpsToInterval(codec.framerate / 2), 1512 cricket::FOURCC_I420); 1513 // The SSRC differs from the send SSRC. 1514 EXPECT_FALSE(channel_->SetSendStreamFormat(kSsrc - 1, format)); 1515 EXPECT_TRUE(channel_->SetSendStreamFormat(kSsrc, format)); 1516 1517 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be dropped. 1518 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be rendered. 1519 EXPECT_TRUE(WaitAndSendFrame(30)); // Should be dropped. 1520 frame_count += 1; 1521 EXPECT_FRAME_WAIT(frame_count, format.width, format.height, kTimeout); 1522 1523 // Adapt the resolution to 0x0, which should drop all frames. 1524 format.width = 0; 1525 format.height = 0; 1526 EXPECT_TRUE(channel_->SetSendStreamFormat(kSsrc, format)); 1527 EXPECT_TRUE(SendFrame()); 1528 EXPECT_TRUE(SendFrame()); 1529 talk_base::Thread::Current()->ProcessMessages(500); 1530 EXPECT_EQ(frame_count, renderer_.num_rendered_frames()); 1531 } 1532 // Test that setting send stream format to 0x0 resolution will result in 1533 // frames being dropped. 1534 void SetSendStreamFormat0x0() { 1535 EXPECT_TRUE(SetOneCodec(DefaultCodec())); 1536 EXPECT_TRUE(SetSend(true)); 1537 EXPECT_TRUE(channel_->SetRender(true)); 1538 EXPECT_EQ(0, renderer_.num_rendered_frames()); 1539 // This frame should be received. 1540 EXPECT_TRUE(SendFrame()); 1541 EXPECT_FRAME_WAIT(1, DefaultCodec().width, DefaultCodec().height, kTimeout); 1542 const int64 interval = cricket::VideoFormat::FpsToInterval( 1543 DefaultCodec().framerate); 1544 cricket::VideoFormat format( 1545 0, 1546 0, 1547 interval, 1548 cricket::FOURCC_I420); 1549 EXPECT_TRUE(channel_->SetSendStreamFormat(kSsrc, format)); 1550 // This frame should not be received. 1551 EXPECT_TRUE(WaitAndSendFrame( 1552 static_cast<int>(interval/talk_base::kNumNanosecsPerMillisec))); 1553 talk_base::Thread::Current()->ProcessMessages(500); 1554 EXPECT_EQ(1, renderer_.num_rendered_frames()); 1555 } 1556 1557 // Tests that we can mute and unmute the channel properly. 1558 void MuteStream() { 1559 int frame_count = 0; 1560 EXPECT_TRUE(SetDefaultCodec()); 1561 cricket::FakeVideoCapturer video_capturer; 1562 video_capturer.Start( 1563 cricket::VideoFormat( 1564 640, 480, 1565 cricket::VideoFormat::FpsToInterval(30), 1566 cricket::FOURCC_I420)); 1567 EXPECT_TRUE(channel_->SetCapturer(kSsrc, &video_capturer)); 1568 EXPECT_TRUE(SetSend(true)); 1569 EXPECT_TRUE(channel_->SetRender(true)); 1570 EXPECT_EQ(frame_count, renderer_.num_rendered_frames()); 1571 1572 // Mute the channel and expect black output frame. 1573 EXPECT_TRUE(channel_->MuteStream(kSsrc, true)); 1574 EXPECT_TRUE(video_capturer.CaptureFrame()); 1575 ++frame_count; 1576 EXPECT_EQ_WAIT(frame_count, renderer_.num_rendered_frames(), kTimeout); 1577 EXPECT_TRUE(renderer_.black_frame()); 1578 1579 // Unmute the channel and expect non-black output frame. 1580 EXPECT_TRUE(channel_->MuteStream(kSsrc, false)); 1581 EXPECT_TRUE(talk_base::Thread::Current()->ProcessMessages(30)); 1582 EXPECT_TRUE(video_capturer.CaptureFrame()); 1583 ++frame_count; 1584 EXPECT_EQ_WAIT(frame_count, renderer_.num_rendered_frames(), kTimeout); 1585 EXPECT_FALSE(renderer_.black_frame()); 1586 1587 // Test that we can also Mute using the correct send stream SSRC. 1588 EXPECT_TRUE(channel_->MuteStream(kSsrc, true)); 1589 EXPECT_TRUE(talk_base::Thread::Current()->ProcessMessages(30)); 1590 EXPECT_TRUE(video_capturer.CaptureFrame()); 1591 ++frame_count; 1592 EXPECT_EQ_WAIT(frame_count, renderer_.num_rendered_frames(), kTimeout); 1593 EXPECT_TRUE(renderer_.black_frame()); 1594 1595 EXPECT_TRUE(channel_->MuteStream(kSsrc, false)); 1596 EXPECT_TRUE(talk_base::Thread::Current()->ProcessMessages(30)); 1597 EXPECT_TRUE(video_capturer.CaptureFrame()); 1598 ++frame_count; 1599 EXPECT_EQ_WAIT(frame_count, renderer_.num_rendered_frames(), kTimeout); 1600 EXPECT_FALSE(renderer_.black_frame()); 1601 1602 // Test that muting an invalid stream fails. 1603 EXPECT_FALSE(channel_->MuteStream(kSsrc+1, true)); 1604 EXPECT_TRUE(channel_->SetCapturer(kSsrc, NULL)); 1605 } 1606 1607 // Test that multiple send streams can be created and deleted properly. 1608 void MultipleSendStreams() { 1609 // Remove stream added in Setup. I.e. remove stream corresponding to default 1610 // channel. 1611 EXPECT_TRUE(channel_->RemoveSendStream(kSsrc)); 1612 const unsigned int kSsrcsSize = sizeof(kSsrcs4)/sizeof(kSsrcs4[0]); 1613 for (unsigned int i = 0; i < kSsrcsSize; ++i) { 1614 EXPECT_TRUE(channel_->AddSendStream( 1615 cricket::StreamParams::CreateLegacy(kSsrcs4[i]))); 1616 } 1617 // Delete one of the non default channel streams, let the destructor delete 1618 // the remaining ones. 1619 EXPECT_TRUE(channel_->RemoveSendStream(kSsrcs4[kSsrcsSize - 1])); 1620 // Stream should already be deleted. 1621 EXPECT_FALSE(channel_->RemoveSendStream(kSsrcs4[kSsrcsSize - 1])); 1622 } 1623 1624 1625 // Two streams one channel tests. 1626 1627 // Tests that we can send and receive frames. 1628 void TwoStreamsSendAndReceive(const cricket::VideoCodec& codec) { 1629 SetUpSecondStream(); 1630 // Test sending and receiving on first stream. 1631 SendAndReceive(codec); 1632 // Test sending and receiving on second stream. 1633 EXPECT_EQ_WAIT(1, renderer2_.num_rendered_frames(), kTimeout); 1634 EXPECT_EQ(2, NumRtpPackets()); 1635 EXPECT_EQ(1, renderer2_.num_rendered_frames()); 1636 } 1637 1638 // Disconnect the first stream and re-use it with another SSRC 1639 void TwoStreamsReUseFirstStream(const cricket::VideoCodec& codec) { 1640 SetUpSecondStream(); 1641 EXPECT_TRUE(channel_->RemoveRecvStream(kSsrc)); 1642 EXPECT_FALSE(channel_->RemoveRecvStream(kSsrc)); 1643 // SSRC 0 should map to the "default" stream. I.e. the first added stream. 1644 EXPECT_TRUE(channel_->RemoveSendStream(0)); 1645 // Make sure that the first added stream was indeed the "default" stream. 1646 EXPECT_FALSE(channel_->RemoveSendStream(kSsrc)); 1647 // Make sure that the "default" stream is indeed removed and that removing 1648 // the default stream has an effect. 1649 EXPECT_FALSE(channel_->RemoveSendStream(0)); 1650 1651 SetRendererAsDefault(); 1652 EXPECT_TRUE(channel_->AddSendStream( 1653 cricket::StreamParams::CreateLegacy(kSsrc))); 1654 EXPECT_FALSE(channel_->AddSendStream( 1655 cricket::StreamParams::CreateLegacy(kSsrc))); 1656 EXPECT_TRUE(channel_->AddRecvStream( 1657 cricket::StreamParams::CreateLegacy(kSsrc))); 1658 EXPECT_FALSE(channel_->AddRecvStream( 1659 cricket::StreamParams::CreateLegacy(kSsrc))); 1660 1661 EXPECT_TRUE(channel_->SetCapturer(kSsrc, video_capturer_.get())); 1662 1663 SendAndReceive(codec); 1664 EXPECT_TRUE(channel_->RemoveSendStream(0)); 1665 } 1666 1667 VideoEngineOverride<E> engine_; 1668 talk_base::scoped_ptr<cricket::FakeVideoCapturer> video_capturer_; 1669 talk_base::scoped_ptr<cricket::FakeVideoCapturer> video_capturer_2_; 1670 talk_base::scoped_ptr<C> channel_; 1671 cricket::FakeNetworkInterface network_interface_; 1672 cricket::FakeVideoRenderer renderer_; 1673 cricket::VideoMediaChannel::Error media_error_; 1674 1675 // Used by test cases where 2 streams are run on the same channel. 1676 cricket::FakeVideoRenderer renderer2_; 1677 }; 1678 1679 #endif // TALK_MEDIA_BASE_VIDEOENGINE_UNITTEST_H_ NOLINT 1680