blob: b0bf81361d547c31cf8265d9469af304f474f67b [file] [log] [blame] [edit]
/*
* libjingle
* Copyright 2004--2011, Google Inc.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
* 3. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
* EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "talk/base/gunit.h"
#include "talk/base/scoped_ptr.h"
#include "talk/session/phone/fakemediaprocessor.h"
#include "talk/session/phone/fakewebrtcvideocapturemodule.h"
#include "talk/session/phone/fakewebrtcvideoengine.h"
#include "talk/session/phone/fakewebrtcvoiceengine.h"
#include "talk/session/phone/mediasession.h"
#include "talk/session/phone/videoengine_unittest.h"
#include "talk/session/phone/webrtcvideocapturer.h"
#include "talk/session/phone/webrtcvideoengine.h"
#include "talk/session/phone/webrtcvideoframe.h"
#include "talk/session/phone/webrtcvoiceengine.h"
// Tests for the WebRtcVideoEngine/VideoChannel code.
static const cricket::VideoCodec kVP8Codec(100, "VP8", 640, 400, 30, 0);
static const cricket::VideoCodec kRedCodec(101, "red", 0, 0, 0, 0);
static const cricket::VideoCodec kUlpFecCodec(102, "ulpfec", 0, 0, 0, 0);
static const cricket::VideoCodec* const kVideoCodecs[] = {
&kVP8Codec,
&kRedCodec,
&kUlpFecCodec
};
static const unsigned int kMinBandwidthKbps = 100;
static const unsigned int kStartBandwidthKbps = 300;
static const unsigned int kMaxBandwidthKbps = 2000;
static const unsigned int kConferenceModeMaxBandwidthKbps = 500;
class FakeViEWrapper : public cricket::ViEWrapper {
public:
explicit FakeViEWrapper(cricket::FakeWebRtcVideoEngine* engine)
: cricket::ViEWrapper(engine, // base
engine, // codec
engine, // capture
engine, // network
engine, // render
engine, // rtp
engine) { // image
}
};
// Test fixture to test WebRtcVideoEngine with a fake webrtc::VideoEngine.
// Useful for testing failure paths.
class WebRtcVideoEngineTestFake : public testing::Test {
public:
WebRtcVideoEngineTestFake()
: vie_(kVideoCodecs, ARRAY_SIZE(kVideoCodecs)),
engine_(NULL, // cricket::WebRtcVoiceEngine
new FakeViEWrapper(&vie_)),
channel_(NULL),
voice_channel_(NULL) {
}
bool SetupEngine() {
bool result = engine_.Init();
if (result) {
channel_ = engine_.CreateChannel(voice_channel_);
result = (channel_ != NULL);
}
return result;
}
bool SendI420Frame(int width, int height) {
if (NULL == channel_) {
return false;
}
cricket::WebRtcVideoFrame frame;
size_t size = width * height * 3 / 2; // I420
talk_base::scoped_ptr<uint8> pixel(new uint8[size]);
if (!frame.Init(cricket::FOURCC_I420,
width, height, width, height,
pixel.get(), size, 1, 1, 0, 0, 0)) {
return false;
}
return channel_->SendFrame(0, &frame);
}
virtual void TearDown() {
delete channel_;
engine_.Terminate();
}
protected:
cricket::FakeWebRtcVideoEngine vie_;
cricket::WebRtcVideoEngine engine_;
cricket::WebRtcVideoMediaChannel* channel_;
cricket::WebRtcVoiceMediaChannel* voice_channel_;
};
// Test fixtures to test WebRtcVideoEngine with a real webrtc::VideoEngine.
class WebRtcVideoEngineTest
: public VideoEngineTest<cricket::WebRtcVideoEngine> {
protected:
typedef VideoEngineTest<cricket::WebRtcVideoEngine> Base;
};
class WebRtcVideoMediaChannelTest
: public VideoMediaChannelTest<
cricket::WebRtcVideoEngine, cricket::WebRtcVideoMediaChannel> {
protected:
typedef VideoMediaChannelTest<cricket::WebRtcVideoEngine,
cricket::WebRtcVideoMediaChannel> Base;
virtual cricket::VideoCodec DefaultCodec() { return kVP8Codec; }
virtual void SetUp() {
Base::SetUp();
// Need to start the capturer to allow us to pump in frames.
engine_.SetCapture(true);
}
virtual void TearDown() {
engine_.SetCapture(false);
Base::TearDown();
}
};
/////////////////////////
// Tests with fake ViE //
/////////////////////////
// Tests that our stub library "works".
TEST_F(WebRtcVideoEngineTestFake, StartupShutdown) {
EXPECT_FALSE(vie_.IsInited());
EXPECT_TRUE(engine_.Init());
EXPECT_TRUE(vie_.IsInited());
engine_.Terminate();
}
// Tests that we can create and destroy a channel.
TEST_F(WebRtcVideoEngineTestFake, CreateChannel) {
EXPECT_TRUE(engine_.Init());
channel_ = engine_.CreateChannel(voice_channel_);
EXPECT_TRUE(channel_ != NULL);
EXPECT_EQ(1, engine_.GetNumOfChannels());
delete channel_;
channel_ = NULL;
EXPECT_EQ(0, engine_.GetNumOfChannels());
}
// Tests that we properly handle failures in CreateChannel.
TEST_F(WebRtcVideoEngineTestFake, CreateChannelFail) {
vie_.set_fail_create_channel(true);
EXPECT_TRUE(engine_.Init());
channel_ = engine_.CreateChannel(voice_channel_);
EXPECT_TRUE(channel_ == NULL);
}
// Tests that we properly handle failures in AllocateExternalCaptureDevice.
TEST_F(WebRtcVideoEngineTestFake, AllocateExternalCaptureDeviceFail) {
vie_.set_fail_alloc_capturer(true);
EXPECT_TRUE(engine_.Init());
channel_ = engine_.CreateChannel(voice_channel_);
EXPECT_TRUE(channel_ == NULL);
}
// Test that we apply our default codecs properly.
TEST_F(WebRtcVideoEngineTestFake, SetSendCodecs) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
std::vector<cricket::VideoCodec> codecs(engine_.codecs());
EXPECT_TRUE(channel_->SetSendCodecs(codecs));
webrtc::VideoCodec gcodec;
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.id, gcodec.plType);
EXPECT_EQ(kVP8Codec.width, gcodec.width);
EXPECT_EQ(kVP8Codec.height, gcodec.height);
EXPECT_STREQ(kVP8Codec.name.c_str(), gcodec.plName);
EXPECT_EQ(kMinBandwidthKbps, gcodec.minBitrate);
EXPECT_EQ(kStartBandwidthKbps, gcodec.startBitrate);
EXPECT_EQ(kMaxBandwidthKbps, gcodec.maxBitrate);
EXPECT_TRUE(vie_.GetHybridNackFecStatus(channel_num));
EXPECT_FALSE(vie_.GetNackStatus(channel_num));
// TODO: Check RTCP, PLI, TMMBR.
}
// Test that we constrain send codecs properly.
TEST_F(WebRtcVideoEngineTestFake, ConstrainSendCodecs) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
// Set max settings of 640x400x30.
EXPECT_TRUE(engine_.SetDefaultEncoderConfig(
cricket::VideoEncoderConfig(kVP8Codec)));
// Send codec format bigger than max setting.
cricket::VideoCodec codec(kVP8Codec);
codec.width = 1280;
codec.height = 800;
codec.framerate = 60;
std::vector<cricket::VideoCodec> codec_list;
codec_list.push_back(codec);
// Set send codec and verify codec has been constrained.
EXPECT_TRUE(channel_->SetSendCodecs(codec_list));
webrtc::VideoCodec gcodec;
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.id, gcodec.plType);
EXPECT_EQ(kVP8Codec.width, gcodec.width);
EXPECT_EQ(kVP8Codec.height, gcodec.height);
EXPECT_EQ(kVP8Codec.framerate, gcodec.maxFramerate);
EXPECT_STREQ(kVP8Codec.name.c_str(), gcodec.plName);
}
// Test that SetSendCodecs rejects bad format.
TEST_F(WebRtcVideoEngineTestFake, SetSendCodecsRejectBadFormat) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
// Set w = 0.
cricket::VideoCodec codec(kVP8Codec);
codec.width = 0;
std::vector<cricket::VideoCodec> codec_list;
codec_list.push_back(codec);
// Verify SetSendCodecs failed and send codec is not changed on engine.
EXPECT_FALSE(channel_->SetSendCodecs(codec_list));
webrtc::VideoCodec gcodec;
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(0, gcodec.plType);
// Set h = 0.
codec_list[0].width = 640;
codec_list[0].height = 0;
// Verify SetSendCodecs failed and send codec is not changed on engine.
EXPECT_FALSE(channel_->SetSendCodecs(codec_list));
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(0, gcodec.plType);
}
// Test that SetSendCodecs rejects bad codec.
TEST_F(WebRtcVideoEngineTestFake, SetSendCodecsRejectBadCodec) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
// Set bad codec name.
cricket::VideoCodec codec(kVP8Codec);
codec.name = "bad";
std::vector<cricket::VideoCodec> codec_list;
codec_list.push_back(codec);
// Verify SetSendCodecs failed and send codec is not changed on engine.
EXPECT_FALSE(channel_->SetSendCodecs(codec_list));
webrtc::VideoCodec gcodec;
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(0, gcodec.plType);
}
// Test that vie send codec is reset on new video frame size.
TEST_F(WebRtcVideoEngineTestFake, ResetVieSendCodecOnNewFrameSize) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
// Set send codec.
std::vector<cricket::VideoCodec> codec_list;
codec_list.push_back(kVP8Codec);
EXPECT_TRUE(channel_->SetSendCodecs(codec_list));
EXPECT_TRUE(channel_->AddSendStream(
cricket::StreamParams::CreateLegacy(123)));
EXPECT_TRUE(channel_->SetSend(true));
// Capture a smaller frame and verify vie send codec has been reset to
// the new size.
SendI420Frame(kVP8Codec.width / 2, kVP8Codec.height / 2);
webrtc::VideoCodec gcodec;
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.width / 2, gcodec.width);
EXPECT_EQ(kVP8Codec.height / 2, gcodec.height);
// Capture a frame bigger than send_codec_ and verify vie send codec has been
// reset (and clipped) to send_codec_.
SendI420Frame(kVP8Codec.width * 2, kVP8Codec.height * 2);
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.width, gcodec.width);
EXPECT_EQ(kVP8Codec.height, gcodec.height);
}
// Test that we set our inbound codecs properly.
TEST_F(WebRtcVideoEngineTestFake, SetRecvCodecs) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
std::vector<cricket::VideoCodec> codecs;
codecs.push_back(kVP8Codec);
EXPECT_TRUE(channel_->SetRecvCodecs(codecs));
webrtc::VideoCodec wcodec;
EXPECT_TRUE(engine_.ConvertFromCricketVideoCodec(kVP8Codec, &wcodec));
EXPECT_TRUE(vie_.ReceiveCodecRegistered(channel_num, wcodec));
}
// Test that channel connects and disconnects external capturer correctly.
TEST_F(WebRtcVideoEngineTestFake, HasExternalCapturer) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_EQ(1, vie_.GetNumCapturers());
int capture_id = vie_.GetCaptureId(channel_num);
EXPECT_EQ(channel_num, vie_.GetCaptureChannelId(capture_id));
// Delete the channel should disconnect the capturer.
delete channel_;
channel_ = NULL;
EXPECT_EQ(0, vie_.GetNumCapturers());
}
// Test that channel adds and removes renderer correctly.
TEST_F(WebRtcVideoEngineTestFake, HasRenderer) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(vie_.GetHasRenderer(channel_num));
EXPECT_FALSE(vie_.GetRenderStarted(channel_num));
}
// Test that rtcp is enabled on the channel.
TEST_F(WebRtcVideoEngineTestFake, RtcpEnabled) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_EQ(webrtc::kRtcpCompound_RFC4585, vie_.GetRtcpStatus(channel_num));
}
// Test that key frame request method is set on the channel.
TEST_F(WebRtcVideoEngineTestFake, KeyFrameRequestEnabled) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_EQ(webrtc::kViEKeyFrameRequestPliRtcp,
vie_.GetKeyFrameRequestMethod(channel_num));
}
// Test that remb is enabled on the default channel.
TEST_F(WebRtcVideoEngineTestFake, RembEnabled) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->AddSendStream(
cricket::StreamParams::CreateLegacy(1)));
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_TRUE(vie_.GetRembStatus(channel_num));
EXPECT_FALSE(vie_.GetRembStatusSend(channel_num));
EXPECT_TRUE(channel_->SetSend(true));
EXPECT_TRUE(vie_.GetRembStatus(channel_num));
EXPECT_TRUE(vie_.GetRembStatusSend(channel_num));
}
// Test that remb is enabled on a receive channel but it uses the default
// channel for sending remb packets.
TEST_F(WebRtcVideoEngineTestFake, RembEnabledOnReceiveChannels) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->SetOptions(cricket::OPT_CONFERENCE));
EXPECT_TRUE(channel_->AddSendStream(
cricket::StreamParams::CreateLegacy(1)));
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_TRUE(vie_.GetRembStatus(channel_num));
EXPECT_FALSE(vie_.GetRembStatusSend(channel_num));
EXPECT_TRUE(channel_->SetSend(true));
EXPECT_TRUE(channel_->AddRecvStream(cricket::StreamParams::CreateLegacy(1)));
int new_channel_num = vie_.GetLastChannel();
EXPECT_NE(channel_num, new_channel_num);
EXPECT_TRUE(vie_.GetRembStatus(channel_num));
EXPECT_TRUE(vie_.GetRembStatusSend(channel_num));
EXPECT_TRUE(vie_.GetRembStatus(new_channel_num));
EXPECT_FALSE(vie_.GetRembStatusSend(new_channel_num));
}
// Test that AddRecvStream doesn't create new channel for 1:1 call.
TEST_F(WebRtcVideoEngineTestFake, AddRecvStream1On1) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->AddRecvStream(cricket::StreamParams::CreateLegacy(1)));
EXPECT_EQ(channel_num, vie_.GetLastChannel());
}
// Test that AddRecvStream doesn't change remb for 1:1 call.
TEST_F(WebRtcVideoEngineTestFake, NoRembChangeAfterAddRecvStream) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->AddSendStream(
cricket::StreamParams::CreateLegacy(1)));
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_TRUE(vie_.GetRembStatus(channel_num));
EXPECT_FALSE(vie_.GetRembStatusSend(channel_num));
EXPECT_TRUE(channel_->SetSend(true));
EXPECT_TRUE(channel_->AddRecvStream(cricket::StreamParams::CreateLegacy(1)));
EXPECT_TRUE(vie_.GetRembStatus(channel_num));
EXPECT_TRUE(vie_.GetRembStatusSend(channel_num));
}
// Test remb sending is on after StartSending and off after StopSending.
TEST_F(WebRtcVideoEngineTestFake, RembOnOff) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
// Verify remb sending is off before StartSending.
EXPECT_TRUE(vie_.GetRembStatus(channel_num));
EXPECT_FALSE(vie_.GetRembStatusSend(channel_num));
// Verify remb sending is on after StartSending.
EXPECT_TRUE(channel_->AddSendStream(
cricket::StreamParams::CreateLegacy(1)));
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_TRUE(channel_->SetSend(true));
EXPECT_TRUE(vie_.GetRembStatus(channel_num));
EXPECT_TRUE(vie_.GetRembStatusSend(channel_num));
// Verify remb sending is off after StopSending.
EXPECT_TRUE(channel_->SetSend(false));
EXPECT_TRUE(vie_.GetRembStatus(channel_num));
EXPECT_FALSE(vie_.GetRembStatusSend(channel_num));
}
// Test that nack is enabled on the channel if we don't offer red/fec.
TEST_F(WebRtcVideoEngineTestFake, NackEnabled) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
std::vector<cricket::VideoCodec> codecs(engine_.codecs());
codecs.resize(1); // toss out red and ulpfec
EXPECT_TRUE(channel_->SetSendCodecs(codecs));
EXPECT_TRUE(vie_.GetNackStatus(channel_num));
}
// Test that we enable hybrid NACK FEC mode.
TEST_F(WebRtcVideoEngineTestFake, HybridNackFec) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->SetRecvCodecs(engine_.codecs()));
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_TRUE(vie_.GetHybridNackFecStatus(channel_num));
EXPECT_FALSE(vie_.GetNackStatus(channel_num));
}
// Test that we enable hybrid NACK FEC mode when calling SetSendCodecs and
// SetReceiveCodecs in reversed order.
TEST_F(WebRtcVideoEngineTestFake, HybridNackFecReversedOrder) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_TRUE(channel_->SetRecvCodecs(engine_.codecs()));
EXPECT_TRUE(vie_.GetHybridNackFecStatus(channel_num));
EXPECT_FALSE(vie_.GetNackStatus(channel_num));
}
// Test NACK vs Hybrid NACK/FEC interop call setup, i.e. only use NACK even if
// red/fec is offered as receive codec.
TEST_F(WebRtcVideoEngineTestFake, VideoProtectionInterop) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
std::vector<cricket::VideoCodec> recv_codecs(engine_.codecs());
std::vector<cricket::VideoCodec> send_codecs(engine_.codecs());
// Only add VP8 as send codec.
send_codecs.resize(1);
EXPECT_TRUE(channel_->SetRecvCodecs(recv_codecs));
EXPECT_TRUE(channel_->SetSendCodecs(send_codecs));
EXPECT_FALSE(vie_.GetHybridNackFecStatus(channel_num));
EXPECT_TRUE(vie_.GetNackStatus(channel_num));
}
// Test NACK vs Hybrid NACK/FEC interop call setup, i.e. only use NACK even if
// red/fec is offered as receive codec. Call order reversed compared to
// VideoProtectionInterop.
TEST_F(WebRtcVideoEngineTestFake, VideoProtectionInteropReversed) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
std::vector<cricket::VideoCodec> recv_codecs(engine_.codecs());
std::vector<cricket::VideoCodec> send_codecs(engine_.codecs());
// Only add VP8 as send codec.
send_codecs.resize(1);
EXPECT_TRUE(channel_->SetSendCodecs(send_codecs));
EXPECT_TRUE(channel_->SetRecvCodecs(recv_codecs));
EXPECT_FALSE(vie_.GetHybridNackFecStatus(channel_num));
EXPECT_TRUE(vie_.GetNackStatus(channel_num));
}
// Test that NACK, not hybrid mode, is enabled in conference mode.
TEST_F(WebRtcVideoEngineTestFake, HybridNackFecConference) {
EXPECT_TRUE(SetupEngine());
// Setup the send channel.
int send_channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->SetOptions(cricket::OPT_CONFERENCE));
EXPECT_TRUE(channel_->SetRecvCodecs(engine_.codecs()));
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_FALSE(vie_.GetHybridNackFecStatus(send_channel_num));
EXPECT_TRUE(vie_.GetNackStatus(send_channel_num));
// Add a receive stream.
EXPECT_TRUE(channel_->AddRecvStream(cricket::StreamParams::CreateLegacy(1)));
int receive_channel_num = vie_.GetLastChannel();
EXPECT_FALSE(vie_.GetHybridNackFecStatus(receive_channel_num));
EXPECT_TRUE(vie_.GetNackStatus(receive_channel_num));
}
// Test that we can create a channel and start/stop rendering out on it.
TEST_F(WebRtcVideoEngineTestFake, SetRender) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
// Verify we can start/stop/start/stop rendering.
EXPECT_TRUE(channel_->SetRender(true));
EXPECT_TRUE(vie_.GetRenderStarted(channel_num));
EXPECT_TRUE(channel_->SetRender(false));
EXPECT_FALSE(vie_.GetRenderStarted(channel_num));
EXPECT_TRUE(channel_->SetRender(true));
EXPECT_TRUE(vie_.GetRenderStarted(channel_num));
EXPECT_TRUE(channel_->SetRender(false));
EXPECT_FALSE(vie_.GetRenderStarted(channel_num));
}
// Test that we can create a channel and start/stop sending out on it.
TEST_F(WebRtcVideoEngineTestFake, SetSend) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
// Set send codecs on the channel.
std::vector<cricket::VideoCodec> codecs;
codecs.push_back(kVP8Codec);
EXPECT_TRUE(channel_->SetSendCodecs(codecs));
EXPECT_TRUE(channel_->AddSendStream(
cricket::StreamParams::CreateLegacy(123)));
// Verify we can start/stop/start/stop sending.
EXPECT_TRUE(channel_->SetSend(true));
EXPECT_TRUE(vie_.GetSend(channel_num));
EXPECT_TRUE(channel_->SetSend(false));
EXPECT_FALSE(vie_.GetSend(channel_num));
EXPECT_TRUE(channel_->SetSend(true));
EXPECT_TRUE(vie_.GetSend(channel_num));
EXPECT_TRUE(channel_->SetSend(false));
EXPECT_FALSE(vie_.GetSend(channel_num));
}
// Test that we set bandwidth properly when using full auto bandwidth mode.
TEST_F(WebRtcVideoEngineTestFake, SetBandwidthAuto) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_TRUE(channel_->SetSendBandwidth(true, cricket::kAutoBandwidth));
webrtc::VideoCodec gcodec;
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.id, gcodec.plType);
EXPECT_STREQ(kVP8Codec.name.c_str(), gcodec.plName);
EXPECT_EQ(kMinBandwidthKbps, gcodec.minBitrate);
EXPECT_EQ(kStartBandwidthKbps, gcodec.startBitrate);
EXPECT_EQ(kMaxBandwidthKbps, gcodec.maxBitrate);
}
// Test that we set bandwidth properly when using auto with upper bound.
TEST_F(WebRtcVideoEngineTestFake, SetBandwidthAutoCapped) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_TRUE(channel_->SetSendBandwidth(true, 768000));
webrtc::VideoCodec gcodec;
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.id, gcodec.plType);
EXPECT_STREQ(kVP8Codec.name.c_str(), gcodec.plName);
EXPECT_EQ(kMinBandwidthKbps, gcodec.minBitrate);
EXPECT_EQ(kStartBandwidthKbps, gcodec.startBitrate);
EXPECT_EQ(768U, gcodec.maxBitrate);
}
// Test that we set bandwidth properly when using a fixed bandwidth.
TEST_F(WebRtcVideoEngineTestFake, SetBandwidthFixed) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
EXPECT_TRUE(channel_->SetSendBandwidth(false, 768000));
webrtc::VideoCodec gcodec;
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.id, gcodec.plType);
EXPECT_STREQ(kVP8Codec.name.c_str(), gcodec.plName);
EXPECT_EQ(768U, gcodec.minBitrate);
EXPECT_EQ(768U, gcodec.startBitrate);
EXPECT_EQ(768U, gcodec.maxBitrate);
}
// Test SetSendSsrc.
TEST_F(WebRtcVideoEngineTestFake, SetSendSsrcAndCname) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
cricket::StreamParams stream;
stream.ssrcs.push_back(1234);
stream.cname = "cname";
channel_->AddSendStream(stream);
unsigned int ssrc = 0;
EXPECT_EQ(0, vie_.GetLocalSSRC(channel_num, ssrc));
EXPECT_EQ(1234U, ssrc);
char rtcp_cname[256];
EXPECT_EQ(0, vie_.GetRTCPCName(channel_num, rtcp_cname));
EXPECT_STREQ("cname", rtcp_cname);
}
// Test that the local SSRC is the same on sending and receiving channels if the
// receive channel is created before the send channel.
TEST_F(WebRtcVideoEngineTestFake, SetSendSsrcAfterCreatingReceiveChannel) {
EXPECT_TRUE(SetupEngine());
EXPECT_TRUE(channel_->AddRecvStream(cricket::StreamParams::CreateLegacy(1)));
int receive_channel_num = vie_.GetLastChannel();
cricket::StreamParams stream = cricket::StreamParams::CreateLegacy(1234);
EXPECT_TRUE(channel_->AddSendStream(stream));
int send_channel_num = vie_.GetLastChannel();
unsigned int ssrc = 0;
EXPECT_EQ(0, vie_.GetLocalSSRC(send_channel_num, ssrc));
EXPECT_EQ(1234U, ssrc);
ssrc = 0;
EXPECT_EQ(0, vie_.GetLocalSSRC(receive_channel_num, ssrc));
EXPECT_EQ(1234U, ssrc);
}
// Test SetOptions with OPT_CONFERENCE flag.
TEST_F(WebRtcVideoEngineTestFake, SetOptionsWithConferenceMode) {
EXPECT_TRUE(SetupEngine());
int channel_num = vie_.GetLastChannel();
EXPECT_TRUE(channel_->SetSendCodecs(engine_.codecs()));
// Verify default send codec and bitrate.
webrtc::VideoCodec gcodec;
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.id, gcodec.plType);
EXPECT_STREQ(kVP8Codec.name.c_str(), gcodec.plName);
EXPECT_EQ(kMinBandwidthKbps, gcodec.minBitrate);
EXPECT_EQ(kStartBandwidthKbps, gcodec.startBitrate);
EXPECT_EQ(kMaxBandwidthKbps, gcodec.maxBitrate);
// Set options with OPT_CONFERENCE flag.
EXPECT_TRUE(channel_->SetOptions(cricket::OPT_CONFERENCE));
// Verify channel now has new max bitrate.
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.id, gcodec.plType);
EXPECT_STREQ(kVP8Codec.name.c_str(), gcodec.plName);
EXPECT_EQ(kMinBandwidthKbps, gcodec.minBitrate);
EXPECT_EQ(kStartBandwidthKbps, gcodec.startBitrate);
EXPECT_EQ(kConferenceModeMaxBandwidthKbps, gcodec.maxBitrate);
// Set options back to zero.
EXPECT_TRUE(channel_->SetOptions(0));
// Verify channel now has default max bitrate again.
EXPECT_EQ(0, vie_.GetSendCodec(channel_num, gcodec));
EXPECT_EQ(kVP8Codec.id, gcodec.plType);
EXPECT_STREQ(kVP8Codec.name.c_str(), gcodec.plName);
EXPECT_EQ(kMinBandwidthKbps, gcodec.minBitrate);
EXPECT_EQ(kStartBandwidthKbps, gcodec.startBitrate);
EXPECT_EQ(kMaxBandwidthKbps, gcodec.maxBitrate);
}
/////////////////////////
// Tests with real ViE //
/////////////////////////
// Tests that we can find codecs by name or id.
TEST_F(WebRtcVideoEngineTest, FindCodec) {
// We should not need to init engine in order to get codecs.
const std::vector<cricket::VideoCodec>& c = engine_.codecs();
EXPECT_EQ(3U, c.size());
cricket::VideoCodec vp8(104, "VP8", 320, 200, 30, 0);
EXPECT_TRUE(engine_.FindCodec(vp8));
cricket::VideoCodec vp8_ci(104, "vp8", 320, 200, 30, 0);
EXPECT_TRUE(engine_.FindCodec(vp8));
cricket::VideoCodec vp8_diff_fr_diff_pref(104, "VP8", 320, 200, 50, 50);
EXPECT_TRUE(engine_.FindCodec(vp8_diff_fr_diff_pref));
cricket::VideoCodec vp8_diff_id(95, "VP8", 320, 200, 30, 0);
EXPECT_FALSE(engine_.FindCodec(vp8_diff_id));
vp8_diff_id.id = 97;
EXPECT_TRUE(engine_.FindCodec(vp8_diff_id));
cricket::VideoCodec vp8_diff_res(104, "VP8", 320, 111, 30, 0);
EXPECT_FALSE(engine_.FindCodec(vp8_diff_res));
// PeerConnection doesn't negotiate the resolution at this point.
// Test that FindCodec can handle the case when width/height is 0.
cricket::VideoCodec vp8_zero_res(104, "VP8", 0, 0, 30, 0);
EXPECT_TRUE(engine_.FindCodec(vp8_zero_res));
cricket::VideoCodec red(101, "RED", 0, 0, 30, 0);
EXPECT_TRUE(engine_.FindCodec(red));
cricket::VideoCodec red_ci(101, "red", 0, 0, 30, 0);
EXPECT_TRUE(engine_.FindCodec(red));
cricket::VideoCodec fec(102, "ULPFEC", 0, 0, 30, 0);
EXPECT_TRUE(engine_.FindCodec(fec));
cricket::VideoCodec fec_ci(102, "ulpfec", 0, 0, 30, 0);
EXPECT_TRUE(engine_.FindCodec(fec));
}
TEST_F(WebRtcVideoEngineTest, StartupShutdown) {
EXPECT_TRUE(engine_.Init());
engine_.Terminate();
}
TEST_PRE_VIDEOENGINE_INIT(WebRtcVideoEngineTest, ConstrainNewCodec)
TEST_POST_VIDEOENGINE_INIT(WebRtcVideoEngineTest, ConstrainNewCodec)
TEST_PRE_VIDEOENGINE_INIT(WebRtcVideoEngineTest, ConstrainRunningCodec)
TEST_POST_VIDEOENGINE_INIT(WebRtcVideoEngineTest, ConstrainRunningCodec)
// TODO: Figure out why ViE is munging the COM refcount.
#ifdef WIN32
TEST_F(WebRtcVideoEngineTest, DISABLED_CheckCoInitialize) {
Base::CheckCoInitialize();
}
#endif
TEST_F(WebRtcVideoEngineTest, CreateChannel) {
EXPECT_TRUE(engine_.Init());
cricket::VideoMediaChannel* channel = engine_.CreateChannel(NULL);
EXPECT_TRUE(channel != NULL);
delete channel;
}
TEST_F(WebRtcVideoEngineTest, SetCaptureDevice) {
cricket::Device device;
EXPECT_TRUE(engine_.Init());
EXPECT_TRUE(engine_.SetCaptureDevice(&device));
EXPECT_FALSE(engine_.IsCapturing());
// FakeVideoCapturer returns CR_SUCCESS.
EXPECT_EQ(cricket::CR_SUCCESS, engine_.SetCapture(true));
EXPECT_TRUE(engine_.IsCapturing());
EXPECT_TRUE(engine_.SetCaptureDevice(NULL));
EXPECT_FALSE(engine_.IsCapturing());
}
TEST_F(WebRtcVideoEngineTest, SetCaptureModule) {
// Use 123 to verify there's no assumption to the module id
FakeWebRtcVideoCaptureModule* vcm =
new FakeWebRtcVideoCaptureModule(NULL, 123);
EXPECT_TRUE(engine_.Init());
// The ownership of the vcm is transferred to the engine.
// Technically we should call vcm->AddRef since we are using the vcm below,
// however the FakeWebRtcVideoCaptureModule didn't implemented the refcount.
// So for testing, this should be fine.
// The SetCaptureModule call always starts the capture.
EXPECT_TRUE(engine_.SetCaptureModule(vcm));
EXPECT_TRUE(engine_.IsCapturing());
EXPECT_EQ(engine_.default_codec_format().width, vcm->cap().width);
EXPECT_EQ(engine_.default_codec_format().height, vcm->cap().height);
EXPECT_EQ(cricket::VideoFormat::IntervalToFps(
engine_.default_codec_format().interval),
vcm->cap().maxFPS);
EXPECT_EQ(webrtc::kVideoI420, vcm->cap().rawType);
EXPECT_EQ(webrtc::kVideoCodecUnknown, vcm->cap().codecType);
EXPECT_TRUE(engine_.SetCaptureModule(NULL));
EXPECT_FALSE(engine_.IsCapturing());
}
TEST_F(WebRtcVideoEngineTest, SetVideoCapturer) {
// Use 123 to verify there's no assumption to the module id
FakeWebRtcVideoCaptureModule* vcm =
new FakeWebRtcVideoCaptureModule(NULL, 123);
talk_base::scoped_ptr<cricket::WebRtcVideoCapturer> capturer(
new cricket::WebRtcVideoCapturer);
EXPECT_TRUE(capturer->Init(vcm));
EXPECT_TRUE(engine_.Init());
const uint32 ssrc_dummy = 0;
EXPECT_TRUE(engine_.SetVideoCapturer(capturer.get(), ssrc_dummy));
EXPECT_FALSE(engine_.IsCapturing());
EXPECT_EQ(cricket::CR_PENDING, engine_.SetCapture(true));
EXPECT_TRUE(engine_.IsCapturing());
EXPECT_EQ(engine_.default_codec_format().width, vcm->cap().width);
EXPECT_EQ(engine_.default_codec_format().height, vcm->cap().height);
EXPECT_EQ(cricket::VideoFormat::IntervalToFps(
engine_.default_codec_format().interval),
vcm->cap().maxFPS);
EXPECT_EQ(webrtc::kVideoI420, vcm->cap().rawType);
EXPECT_EQ(webrtc::kVideoCodecUnknown, vcm->cap().codecType);
EXPECT_TRUE(engine_.SetVideoCapturer(NULL, ssrc_dummy));
EXPECT_FALSE(engine_.IsCapturing());
}
TEST_F(WebRtcVideoEngineTest, TestRegisterVideoProcessor) {
cricket::FakeMediaProcessor vp;
EXPECT_TRUE(engine_.Init());
EXPECT_TRUE(engine_.RegisterProcessor(&vp));
engine_.TriggerMediaFrame(0, NULL);
EXPECT_EQ(1, vp.video_frame_count());
EXPECT_TRUE(engine_.UnregisterProcessor(&vp));
engine_.TriggerMediaFrame(0, NULL);
EXPECT_EQ(1, vp.video_frame_count());
engine_.Terminate();
}
TEST_F(WebRtcVideoMediaChannelTest, SetRecvCodecs) {
std::vector<cricket::VideoCodec> codecs;
codecs.push_back(kVP8Codec);
EXPECT_TRUE(channel_->SetRecvCodecs(codecs));
}
TEST_F(WebRtcVideoMediaChannelTest, SetRecvCodecsWrongPayloadType) {
std::vector<cricket::VideoCodec> codecs;
codecs.push_back(kVP8Codec);
codecs[0].id = 99;
EXPECT_TRUE(channel_->SetRecvCodecs(codecs));
}
TEST_F(WebRtcVideoMediaChannelTest, SetRecvCodecsUnsupportedCodec) {
std::vector<cricket::VideoCodec> codecs;
codecs.push_back(kVP8Codec);
codecs.push_back(cricket::VideoCodec(101, "VP1", 640, 400, 30, 0));
EXPECT_FALSE(channel_->SetRecvCodecs(codecs));
}
TEST_F(WebRtcVideoMediaChannelTest, SetSend) {
Base::SetSend();
}
TEST_F(WebRtcVideoMediaChannelTest, SetSendWithoutCodecs) {
Base::SetSendWithoutCodecs();
}
TEST_F(WebRtcVideoMediaChannelTest, SetSendSetsTransportBufferSizes) {
Base::SetSendSetsTransportBufferSizes();
}
TEST_F(WebRtcVideoMediaChannelTest, SendAndReceiveVp8Vga) {
SendAndReceive(cricket::VideoCodec(100, "VP8", 640, 400, 30, 0));
}
TEST_F(WebRtcVideoMediaChannelTest, SendAndReceiveVp8Qvga) {
SendAndReceive(cricket::VideoCodec(100, "VP8", 320, 200, 30, 0));
}
TEST_F(WebRtcVideoMediaChannelTest, SendAndReceiveH264SvcQqvga) {
SendAndReceive(cricket::VideoCodec(100, "VP8", 160, 100, 30, 0));
}
// TODO: Figure out why this test doesn't work.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_SendManyResizeOnce) {
SendManyResizeOnce();
}
// TODO: Fix this test to tolerate missing stats.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_GetStats) {
Base::GetStats();
}
// TODO: Fix this test to tolerate missing stats.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_GetStatsMultipleRecvStreams) {
Base::GetStatsMultipleRecvStreams();
}
// TODO: Restore this test once we support multiple send streams.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_GetStatsMultipleSendStreams) {
Base::GetStatsMultipleSendStreams();
}
TEST_F(WebRtcVideoMediaChannelTest, SetSendBandwidth) {
Base::SetSendBandwidth();
}
TEST_F(WebRtcVideoMediaChannelTest, SetSendSsrc) {
Base::SetSendSsrc();
}
TEST_F(WebRtcVideoMediaChannelTest, SetSendSsrcAfterSetCodecs) {
Base::SetSendSsrcAfterSetCodecs();
}
TEST_F(WebRtcVideoMediaChannelTest, SetRenderer) {
Base::SetRenderer();
}
TEST_F(WebRtcVideoMediaChannelTest, AddRemoveRecvStreams) {
Base::AddRemoveRecvStreams();
}
TEST_F(WebRtcVideoMediaChannelTest, AddRemoveRecvStreamsNoConference) {
Base::AddRemoveRecvStreamsNoConference();
}
TEST_F(WebRtcVideoMediaChannelTest, AddRemoveSendStreams) {
Base::AddRemoveSendStreams();
}
TEST_F(WebRtcVideoMediaChannelTest, SimulateConference) {
Base::SimulateConference();
}
TEST_F(WebRtcVideoMediaChannelTest, SetOptionsFailsWhenSending) {
EXPECT_TRUE(channel_->SetOptions(cricket::OPT_CONFERENCE));
// Verify SetOptions returns true on a different options.
EXPECT_TRUE(channel_->SetOptions(cricket::OPT_CPU_ADAPTATION));
// Set send codecs on the channel and start sending.
std::vector<cricket::VideoCodec> codecs;
codecs.push_back(kVP8Codec);
EXPECT_TRUE(channel_->SetSendCodecs(codecs));
EXPECT_TRUE(channel_->SetSend(true));
// Verify SetOptions returns false if channel is already sending.
EXPECT_FALSE(channel_->SetOptions(cricket::OPT_CONFERENCE));
// Verify SetOptions returns true with the old options.
EXPECT_TRUE(channel_->SetOptions(cricket::OPT_CPU_ADAPTATION));
}
// TODO: Investigate why this test is flaky.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_AdaptResolution16x10) {
Base::AdaptResolution16x10();
}
// TODO: Investigate why this test is flaky.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_AdaptResolution4x3) {
Base::AdaptResolution4x3();
}
// TODO: Restore this test once we support sending 0 fps.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_AdaptDropAllFrames) {
Base::AdaptDropAllFrames();
}
// TODO: Understand why we get decode errors on this test.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_AdaptFramerate) {
Base::AdaptFramerate();
}
// TODO: Understand why format is set but the encoded frames do not
// change.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_SetSendStreamFormat) {
Base::SetSendStreamFormat();
}
// TODO: Understand why we receive a not-quite-black frame.
TEST_F(WebRtcVideoMediaChannelTest, DISABLED_Mute) {
Base::Mute();
}