blob: d318ac5b202157d9efa63f03b0d8814e7bf56ce1 [file] [log] [blame]
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:13 +00001/*
kjellander65c7f672016-02-12 00:05:01 -08002 * Copyright 2008 The WebRTC project authors. All Rights Reserved.
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:13 +00003 *
kjellander65c7f672016-02-12 00:05:01 -08004 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:13 +00009 */
henrike@webrtc.org28e20752013-07-10 00:45:36 +000010
deadbeef112b2e92017-02-10 20:13:37 -080011#include <memory>
Zhi Huang2dfc42d2017-12-04 13:38:48 -080012#include <utility>
deadbeef112b2e92017-02-10 20:13:37 -080013
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020014#include "media/base/fakemediaengine.h"
15#include "media/base/fakevideocapturer.h"
16#include "media/base/testutils.h"
17#include "media/engine/fakewebrtccall.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020018#include "pc/channelmanager.h"
Zhi Huangb5261582017-09-29 10:51:43 -070019#include "pc/test/faketransportcontroller.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020020#include "rtc_base/gunit.h"
21#include "rtc_base/logging.h"
22#include "rtc_base/thread.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000023
deadbeef1a2183d2017-02-10 23:44:49 -080024namespace {
deadbeef7af91dd2016-12-13 11:29:11 -080025const bool kDefaultSrtpRequired = true;
26}
27
28namespace cricket {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000029
30static const AudioCodec kAudioCodecs[] = {
deadbeef67cf2c12016-04-13 10:07:16 -070031 AudioCodec(97, "voice", 1, 2, 3), AudioCodec(111, "OPUS", 48000, 32000, 2),
henrike@webrtc.org28e20752013-07-10 00:45:36 +000032};
33
34static const VideoCodec kVideoCodecs[] = {
perkj26752742016-10-24 01:21:16 -070035 VideoCodec(99, "H264"), VideoCodec(100, "VP8"), VideoCodec(96, "rtx"),
henrike@webrtc.org28e20752013-07-10 00:45:36 +000036};
37
38class ChannelManagerTest : public testing::Test {
39 protected:
deadbeefcbecd352015-09-23 11:50:27 -070040 ChannelManagerTest()
tommie7251592017-07-14 14:44:46 -070041 : network_(rtc::Thread::CreateWithSocketServer()),
42 worker_(rtc::Thread::Create()),
43 fme_(new cricket::FakeMediaEngine()),
stefanc1aeaf02015-10-15 07:26:07 -070044 fdme_(new cricket::FakeDataEngine()),
deadbeef112b2e92017-02-10 20:13:37 -080045 cm_(new cricket::ChannelManager(
46 std::unique_ptr<MediaEngineInterface>(fme_),
47 std::unique_ptr<DataEngineInterface>(fdme_),
Steve Antonc9e15602017-11-06 15:40:09 -080048 rtc::Thread::Current(),
deadbeef112b2e92017-02-10 20:13:37 -080049 rtc::Thread::Current())),
Sebastian Jansson8f83b422018-02-21 13:07:13 +010050 fake_call_(),
stefanc1aeaf02015-10-15 07:26:07 -070051 transport_controller_(
deadbeef112b2e92017-02-10 20:13:37 -080052 new cricket::FakeTransportController(ICEROLE_CONTROLLING)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000053 fme_->SetAudioCodecs(MAKE_VECTOR(kAudioCodecs));
54 fme_->SetVideoCodecs(MAKE_VECTOR(kVideoCodecs));
henrike@webrtc.org28e20752013-07-10 00:45:36 +000055 }
56
tommie7251592017-07-14 14:44:46 -070057 std::unique_ptr<rtc::Thread> network_;
58 std::unique_ptr<rtc::Thread> worker_;
deadbeef112b2e92017-02-10 20:13:37 -080059 // |fme_| and |fdme_| are actually owned by |cm_|.
henrike@webrtc.org28e20752013-07-10 00:45:36 +000060 cricket::FakeMediaEngine* fme_;
61 cricket::FakeDataEngine* fdme_;
deadbeef112b2e92017-02-10 20:13:37 -080062 std::unique_ptr<cricket::ChannelManager> cm_;
stefanc1aeaf02015-10-15 07:26:07 -070063 cricket::FakeCall fake_call_;
deadbeef112b2e92017-02-10 20:13:37 -080064 std::unique_ptr<cricket::FakeTransportController> transport_controller_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000065};
66
67// Test that we startup/shutdown properly.
68TEST_F(ChannelManagerTest, StartupShutdown) {
69 EXPECT_FALSE(cm_->initialized());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000070 EXPECT_EQ(rtc::Thread::Current(), cm_->worker_thread());
henrike@webrtc.org28e20752013-07-10 00:45:36 +000071 EXPECT_TRUE(cm_->Init());
72 EXPECT_TRUE(cm_->initialized());
73 cm_->Terminate();
74 EXPECT_FALSE(cm_->initialized());
75}
76
77// Test that we startup/shutdown properly with a worker thread.
78TEST_F(ChannelManagerTest, StartupShutdownOnThread) {
tommie7251592017-07-14 14:44:46 -070079 network_->Start();
80 worker_->Start();
henrike@webrtc.org28e20752013-07-10 00:45:36 +000081 EXPECT_FALSE(cm_->initialized());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000082 EXPECT_EQ(rtc::Thread::Current(), cm_->worker_thread());
tommie7251592017-07-14 14:44:46 -070083 EXPECT_TRUE(cm_->set_network_thread(network_.get()));
84 EXPECT_EQ(network_.get(), cm_->network_thread());
85 EXPECT_TRUE(cm_->set_worker_thread(worker_.get()));
86 EXPECT_EQ(worker_.get(), cm_->worker_thread());
henrike@webrtc.org28e20752013-07-10 00:45:36 +000087 EXPECT_TRUE(cm_->Init());
88 EXPECT_TRUE(cm_->initialized());
Danil Chapovalov33b01f22016-05-11 19:55:27 +020089 // Setting the network or worker thread while initialized should fail.
90 EXPECT_FALSE(cm_->set_network_thread(rtc::Thread::Current()));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000091 EXPECT_FALSE(cm_->set_worker_thread(rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +000092 cm_->Terminate();
93 EXPECT_FALSE(cm_->initialized());
94}
95
henrike@webrtc.org28e20752013-07-10 00:45:36 +000096// Test that we can create and destroy a voice and video channel.
97TEST_F(ChannelManagerTest, CreateDestroyChannels) {
98 EXPECT_TRUE(cm_->Init());
zhihuangb2cdd932017-01-19 16:54:25 -080099 cricket::DtlsTransportInternal* rtp_transport =
100 transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -0800101 cricket::CN_AUDIO, cricket::ICE_CANDIDATE_COMPONENT_RTP);
skvlad6c87a672016-05-17 17:49:52 -0700102 cricket::VoiceChannel* voice_channel = cm_->CreateVoiceChannel(
nisseeaabdf62017-05-05 02:23:02 -0700103 &fake_call_, cricket::MediaConfig(),
104 rtp_transport, nullptr /*rtcp_transport*/,
deadbeef1a2183d2017-02-10 23:44:49 -0800105 rtc::Thread::Current(), cricket::CN_AUDIO, kDefaultSrtpRequired,
106 AudioOptions());
Jelena Marusicc28a8962015-05-29 15:05:44 +0200107 EXPECT_TRUE(voice_channel != nullptr);
skvlad6c87a672016-05-17 17:49:52 -0700108 cricket::VideoChannel* video_channel = cm_->CreateVideoChannel(
nisseeaabdf62017-05-05 02:23:02 -0700109 &fake_call_, cricket::MediaConfig(),
110 rtp_transport, nullptr /*rtcp_transport*/,
deadbeef1a2183d2017-02-10 23:44:49 -0800111 rtc::Thread::Current(), cricket::CN_VIDEO, kDefaultSrtpRequired,
112 VideoOptions());
Jelena Marusicc28a8962015-05-29 15:05:44 +0200113 EXPECT_TRUE(video_channel != nullptr);
deadbeef953c2ce2017-01-09 14:53:41 -0800114 cricket::RtpDataChannel* rtp_data_channel = cm_->CreateRtpDataChannel(
nisseeaabdf62017-05-05 02:23:02 -0700115 cricket::MediaConfig(), rtp_transport, nullptr /*rtcp_transport*/,
deadbeef1a2183d2017-02-10 23:44:49 -0800116 rtc::Thread::Current(), cricket::CN_DATA, kDefaultSrtpRequired);
deadbeef953c2ce2017-01-09 14:53:41 -0800117 EXPECT_TRUE(rtp_data_channel != nullptr);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000118 cm_->DestroyVideoChannel(video_channel);
Fredrik Solenberg709ed672015-09-15 12:26:33 +0200119 cm_->DestroyVoiceChannel(voice_channel);
deadbeef953c2ce2017-01-09 14:53:41 -0800120 cm_->DestroyRtpDataChannel(rtp_data_channel);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000121 cm_->Terminate();
122}
123
124// Test that we can create and destroy a voice and video channel with a worker.
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +0000125TEST_F(ChannelManagerTest, CreateDestroyChannelsOnThread) {
tommie7251592017-07-14 14:44:46 -0700126 network_->Start();
127 worker_->Start();
128 EXPECT_TRUE(cm_->set_worker_thread(worker_.get()));
129 EXPECT_TRUE(cm_->set_network_thread(network_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000130 EXPECT_TRUE(cm_->Init());
tommie7251592017-07-14 14:44:46 -0700131 transport_controller_.reset(new cricket::FakeTransportController(
132 network_.get(), ICEROLE_CONTROLLING));
zhihuangb2cdd932017-01-19 16:54:25 -0800133 cricket::DtlsTransportInternal* rtp_transport =
134 transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -0800135 cricket::CN_AUDIO, cricket::ICE_CANDIDATE_COMPONENT_RTP);
skvlad6c87a672016-05-17 17:49:52 -0700136 cricket::VoiceChannel* voice_channel = cm_->CreateVoiceChannel(
nisseeaabdf62017-05-05 02:23:02 -0700137 &fake_call_, cricket::MediaConfig(),
138 rtp_transport, nullptr /*rtcp_transport*/,
deadbeef1a2183d2017-02-10 23:44:49 -0800139 rtc::Thread::Current(), cricket::CN_AUDIO, kDefaultSrtpRequired,
140 AudioOptions());
Jelena Marusicc28a8962015-05-29 15:05:44 +0200141 EXPECT_TRUE(voice_channel != nullptr);
skvlad6c87a672016-05-17 17:49:52 -0700142 cricket::VideoChannel* video_channel = cm_->CreateVideoChannel(
nisseeaabdf62017-05-05 02:23:02 -0700143 &fake_call_, cricket::MediaConfig(),
144 rtp_transport, nullptr /*rtcp_transport*/,
deadbeef1a2183d2017-02-10 23:44:49 -0800145 rtc::Thread::Current(), cricket::CN_VIDEO, kDefaultSrtpRequired,
146 VideoOptions());
Jelena Marusicc28a8962015-05-29 15:05:44 +0200147 EXPECT_TRUE(video_channel != nullptr);
deadbeef953c2ce2017-01-09 14:53:41 -0800148 cricket::RtpDataChannel* rtp_data_channel = cm_->CreateRtpDataChannel(
nisseeaabdf62017-05-05 02:23:02 -0700149 cricket::MediaConfig(), rtp_transport, nullptr /*rtcp_transport*/,
deadbeef1a2183d2017-02-10 23:44:49 -0800150 rtc::Thread::Current(), cricket::CN_DATA, kDefaultSrtpRequired);
deadbeef953c2ce2017-01-09 14:53:41 -0800151 EXPECT_TRUE(rtp_data_channel != nullptr);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000152 cm_->DestroyVideoChannel(video_channel);
Fredrik Solenberg709ed672015-09-15 12:26:33 +0200153 cm_->DestroyVoiceChannel(voice_channel);
deadbeef953c2ce2017-01-09 14:53:41 -0800154 cm_->DestroyRtpDataChannel(rtp_data_channel);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000155 cm_->Terminate();
156}
157
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000158TEST_F(ChannelManagerTest, SetVideoRtxEnabled) {
159 std::vector<VideoCodec> codecs;
perkj26752742016-10-24 01:21:16 -0700160 const VideoCodec rtx_codec(96, "rtx");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000161
162 // By default RTX is disabled.
magjed3cf8ece2016-11-10 03:36:53 -0800163 cm_->GetSupportedVideoCodecs(&codecs);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000164 EXPECT_FALSE(ContainsMatchingCodec(codecs, rtx_codec));
165
166 // Enable and check.
167 EXPECT_TRUE(cm_->SetVideoRtxEnabled(true));
magjed3cf8ece2016-11-10 03:36:53 -0800168 cm_->GetSupportedVideoCodecs(&codecs);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000169 EXPECT_TRUE(ContainsMatchingCodec(codecs, rtx_codec));
170
171 // Disable and check.
172 EXPECT_TRUE(cm_->SetVideoRtxEnabled(false));
magjed3cf8ece2016-11-10 03:36:53 -0800173 cm_->GetSupportedVideoCodecs(&codecs);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000174 EXPECT_FALSE(ContainsMatchingCodec(codecs, rtx_codec));
175
176 // Cannot toggle rtx after initialization.
177 EXPECT_TRUE(cm_->Init());
178 EXPECT_FALSE(cm_->SetVideoRtxEnabled(true));
179 EXPECT_FALSE(cm_->SetVideoRtxEnabled(false));
180
181 // Can set again after terminate.
182 cm_->Terminate();
183 EXPECT_TRUE(cm_->SetVideoRtxEnabled(true));
magjed3cf8ece2016-11-10 03:36:53 -0800184 cm_->GetSupportedVideoCodecs(&codecs);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000185 EXPECT_TRUE(ContainsMatchingCodec(codecs, rtx_codec));
186}
187
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800188enum class RTPTransportType { kRtp, kSrtp, kDtlsSrtp };
189
190class ChannelManagerTestWithRtpTransport
191 : public ChannelManagerTest,
192 public ::testing::WithParamInterface<RTPTransportType> {
193 public:
194 std::unique_ptr<webrtc::RtpTransportInternal> CreateRtpTransport() {
195 RTPTransportType type = GetParam();
196 switch (type) {
197 case RTPTransportType::kRtp:
198 return CreatePlainRtpTransport();
199 case RTPTransportType::kSrtp:
200 return CreateSrtpTransport();
201 case RTPTransportType::kDtlsSrtp:
202 return CreateDtlsSrtpTransport();
203 }
204 return nullptr;
205 }
206
207 void TestCreateDestroyChannels(webrtc::RtpTransportInternal* rtp_transport) {
208 cricket::VoiceChannel* voice_channel = cm_->CreateVoiceChannel(
209 &fake_call_, cricket::MediaConfig(), rtp_transport,
210 rtc::Thread::Current(), cricket::CN_AUDIO, kDefaultSrtpRequired,
211 AudioOptions());
212 EXPECT_TRUE(voice_channel != nullptr);
213 cricket::VideoChannel* video_channel = cm_->CreateVideoChannel(
214 &fake_call_, cricket::MediaConfig(), rtp_transport,
215 rtc::Thread::Current(), cricket::CN_VIDEO, kDefaultSrtpRequired,
216 VideoOptions());
217 EXPECT_TRUE(video_channel != nullptr);
218 cricket::RtpDataChannel* rtp_data_channel = cm_->CreateRtpDataChannel(
219 cricket::MediaConfig(), rtp_transport, rtc::Thread::Current(),
220 cricket::CN_DATA, kDefaultSrtpRequired);
221 EXPECT_TRUE(rtp_data_channel != nullptr);
222 cm_->DestroyVideoChannel(video_channel);
223 cm_->DestroyVoiceChannel(voice_channel);
224 cm_->DestroyRtpDataChannel(rtp_data_channel);
225 cm_->Terminate();
226 }
227
228 private:
229 std::unique_ptr<webrtc::RtpTransportInternal> CreatePlainRtpTransport() {
230 return rtc::MakeUnique<webrtc::RtpTransport>(/*rtcp_mux_required=*/true);
231 }
232
233 std::unique_ptr<webrtc::RtpTransportInternal> CreateSrtpTransport() {
234 auto rtp_transport =
235 rtc::MakeUnique<webrtc::RtpTransport>(/*rtcp_mux_required=*/true);
236 auto srtp_transport =
237 rtc::MakeUnique<webrtc::SrtpTransport>(std::move(rtp_transport));
238 return srtp_transport;
239 }
240
241 std::unique_ptr<webrtc::RtpTransportInternal> CreateDtlsSrtpTransport() {
242 auto rtp_transport =
243 rtc::MakeUnique<webrtc::RtpTransport>(/*rtcp_mux_required=*/true);
244 auto srtp_transport =
245 rtc::MakeUnique<webrtc::SrtpTransport>(std::move(rtp_transport));
246 auto dtls_srtp_transport_ =
247 rtc::MakeUnique<webrtc::DtlsSrtpTransport>(std::move(srtp_transport));
248 return dtls_srtp_transport_;
249 }
250};
251
252TEST_P(ChannelManagerTestWithRtpTransport, CreateDestroyChannels) {
253 EXPECT_TRUE(cm_->Init());
254 auto rtp_transport = CreateRtpTransport();
255 TestCreateDestroyChannels(rtp_transport.get());
256}
257
258TEST_P(ChannelManagerTestWithRtpTransport, CreateDestroyChannelsOnThread) {
259 network_->Start();
260 worker_->Start();
261 EXPECT_TRUE(cm_->set_worker_thread(worker_.get()));
262 EXPECT_TRUE(cm_->set_network_thread(network_.get()));
263 EXPECT_TRUE(cm_->Init());
264 auto rtp_transport = CreateRtpTransport();
265 TestCreateDestroyChannels(rtp_transport.get());
266}
267
268INSTANTIATE_TEST_CASE_P(ChannelManagerTest,
269 ChannelManagerTestWithRtpTransport,
270 ::testing::Values(RTPTransportType::kRtp,
271 RTPTransportType::kSrtp,
272 RTPTransportType::kDtlsSrtp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000273} // namespace cricket