blob: 5d30f45dc5b8bf9b4c33a04a837168e2fa486f6e [file] [log] [blame]
solenbergc7a8b082015-10-16 14:35:07 -07001/*
2 * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved.
3 *
4 * 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.
9 */
10
Jonas Olssona4d87372019-07-05 19:08:33 +020011#include "call/call.h"
12
solenbergc7a8b082015-10-16 14:35:07 -070013#include <list>
kwiberg1c07c702017-03-27 07:15:49 -070014#include <map>
kwibergb25345e2016-03-12 06:10:44 -080015#include <memory>
zstein7cb69d52017-05-08 11:52:38 -070016#include <utility>
solenbergc7a8b082015-10-16 14:35:07 -070017
Karl Wiberg918f50c2018-07-05 11:40:33 +020018#include "absl/memory/memory.h"
Ali Tofigh641a1b12022-05-17 11:48:46 +020019#include "absl/strings/string_view.h"
Karl Wibergf3850f62017-11-02 13:04:41 +010020#include "api/audio_codecs/builtin_audio_decoder_factory.h"
Danil Chapovalov83bbe912019-08-07 12:24:53 +020021#include "api/rtc_event_log/rtc_event_log.h"
Danil Chapovalov53d45ba2019-07-03 14:56:33 +020022#include "api/task_queue/default_task_queue_factory.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020023#include "api/test/mock_audio_mixer.h"
Henrik Boströmf4a99912020-06-11 12:07:14 +020024#include "api/test/video/function_video_encoder_factory.h"
Erik Språng014dd3c2019-11-28 13:44:25 +010025#include "api/transport/field_trial_based_config.h"
Henrik Boströmf4a99912020-06-11 12:07:14 +020026#include "api/video/builtin_video_bitrate_allocator_factory.h"
Fredrik Solenberg8f5787a2018-01-11 13:52:30 +010027#include "audio/audio_receive_stream.h"
Yves Gerey665174f2018-06-19 15:03:05 +020028#include "audio/audio_send_stream.h"
Henrik Boströmf4a99912020-06-11 12:07:14 +020029#include "call/adaptation/test/fake_resource.h"
Henrik Boström29444c62020-07-01 15:48:46 +020030#include "call/adaptation/test/mock_resource_listener.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020031#include "call/audio_state.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020032#include "modules/audio_device/include/mock_audio_device.h"
Fredrik Solenberg2a877972017-12-15 16:42:15 +010033#include "modules/audio_processing/include/mock_audio_processing.h"
Niels Möller3e9cb2c2021-01-14 12:00:57 +010034#include "modules/include/module.h"
Tomas Gunnarssonf25761d2020-06-03 22:55:33 +020035#include "modules/rtp_rtcp/source/rtp_rtcp_interface.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020036#include "test/fake_encoder.h"
37#include "test/gtest.h"
38#include "test/mock_audio_decoder_factory.h"
39#include "test/mock_transport.h"
Tommi553c8692020-05-05 15:35:45 +020040#include "test/run_loop.h"
solenbergc7a8b082015-10-16 14:35:07 -070041
42namespace {
43
Henrik Boström29444c62020-07-01 15:48:46 +020044using ::testing::_;
Henrik Boströmf4a99912020-06-11 12:07:14 +020045using ::testing::Contains;
Alex Konradic20baf62020-12-03 11:30:34 -050046using ::testing::NiceMock;
Henrik Boström29444c62020-07-01 15:48:46 +020047using ::testing::StrictMock;
Henrik Boströmf4a99912020-06-11 12:07:14 +020048
solenbergc7a8b082015-10-16 14:35:07 -070049struct CallHelper {
Per Åhgrencc73ed32020-04-26 23:56:17 +020050 explicit CallHelper(bool use_null_audio_processing) {
Danil Chapovalov53d45ba2019-07-03 14:56:33 +020051 task_queue_factory_ = webrtc::CreateDefaultTaskQueueFactory();
solenberg566ef242015-11-06 15:34:49 -080052 webrtc::AudioState::Config audio_state_config;
Fredrik Solenberg2a877972017-12-15 16:42:15 +010053 audio_state_config.audio_mixer =
Tomas Gunnarssonc1d58912021-04-22 19:21:43 +020054 rtc::make_ref_counted<webrtc::test::MockAudioMixer>();
Fredrik Solenberg2a877972017-12-15 16:42:15 +010055 audio_state_config.audio_processing =
Per Åhgrencc73ed32020-04-26 23:56:17 +020056 use_null_audio_processing
57 ? nullptr
Tomas Gunnarssonc1d58912021-04-22 19:21:43 +020058 : rtc::make_ref_counted<
Alex Konradic20baf62020-12-03 11:30:34 -050059 NiceMock<webrtc::test::MockAudioProcessing>>();
Fredrik Solenberg2a877972017-12-15 16:42:15 +010060 audio_state_config.audio_device_module =
Tomas Gunnarssonc1d58912021-04-22 19:21:43 +020061 rtc::make_ref_counted<webrtc::test::MockAudioDeviceModule>();
skvlad11a9cbf2016-10-07 11:53:05 -070062 webrtc::Call::Config config(&event_log_);
solenberg566ef242015-11-06 15:34:49 -080063 config.audio_state = webrtc::AudioState::Create(audio_state_config);
Danil Chapovalov53d45ba2019-07-03 14:56:33 +020064 config.task_queue_factory = task_queue_factory_.get();
Erik Språng014dd3c2019-11-28 13:44:25 +010065 config.trials = &field_trials_;
solenbergc7a8b082015-10-16 14:35:07 -070066 call_.reset(webrtc::Call::Create(config));
67 }
68
69 webrtc::Call* operator->() { return call_.get(); }
70
71 private:
Tommi553c8692020-05-05 15:35:45 +020072 webrtc::test::RunLoop loop_;
Danil Chapovalov83bbe912019-08-07 12:24:53 +020073 webrtc::RtcEventLogNull event_log_;
Erik Språng014dd3c2019-11-28 13:44:25 +010074 webrtc::FieldTrialBasedConfig field_trials_;
Danil Chapovalov53d45ba2019-07-03 14:56:33 +020075 std::unique_ptr<webrtc::TaskQueueFactory> task_queue_factory_;
kwibergb25345e2016-03-12 06:10:44 -080076 std::unique_ptr<webrtc::Call> call_;
solenbergc7a8b082015-10-16 14:35:07 -070077};
78} // namespace
79
80namespace webrtc {
81
Henrik Boström29444c62020-07-01 15:48:46 +020082namespace {
83
84rtc::scoped_refptr<Resource> FindResourceWhoseNameContains(
85 const std::vector<rtc::scoped_refptr<Resource>>& resources,
Ali Tofigh641a1b12022-05-17 11:48:46 +020086 absl::string_view name_contains) {
Henrik Boström29444c62020-07-01 15:48:46 +020087 for (const auto& resource : resources) {
Ali Tofigh641a1b12022-05-17 11:48:46 +020088 if (resource->Name().find(std::string(name_contains)) != std::string::npos)
Henrik Boström29444c62020-07-01 15:48:46 +020089 return resource;
90 }
91 return nullptr;
92}
93
94} // namespace
95
solenbergc7a8b082015-10-16 14:35:07 -070096TEST(CallTest, ConstructDestruct) {
Per Åhgrencc73ed32020-04-26 23:56:17 +020097 for (bool use_null_audio_processing : {false, true}) {
98 CallHelper call(use_null_audio_processing);
99 }
solenbergc7a8b082015-10-16 14:35:07 -0700100}
101
102TEST(CallTest, CreateDestroy_AudioSendStream) {
Per Åhgrencc73ed32020-04-26 23:56:17 +0200103 for (bool use_null_audio_processing : {false, true}) {
104 CallHelper call(use_null_audio_processing);
105 MockTransport send_transport;
106 AudioSendStream::Config config(&send_transport);
107 config.rtp.ssrc = 42;
108 AudioSendStream* stream = call->CreateAudioSendStream(config);
109 EXPECT_NE(stream, nullptr);
110 call->DestroyAudioSendStream(stream);
111 }
solenbergc7a8b082015-10-16 14:35:07 -0700112}
113
114TEST(CallTest, CreateDestroy_AudioReceiveStream) {
Per Åhgrencc73ed32020-04-26 23:56:17 +0200115 for (bool use_null_audio_processing : {false, true}) {
116 CallHelper call(use_null_audio_processing);
Tommi3176ef72022-05-22 20:47:28 +0200117 AudioReceiveStreamInterface::Config config;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200118 MockTransport rtcp_send_transport;
119 config.rtp.remote_ssrc = 42;
120 config.rtcp_send_transport = &rtcp_send_transport;
121 config.decoder_factory =
Tomas Gunnarssonc1d58912021-04-22 19:21:43 +0200122 rtc::make_ref_counted<webrtc::MockAudioDecoderFactory>();
Tommi3176ef72022-05-22 20:47:28 +0200123 AudioReceiveStreamInterface* stream =
124 call->CreateAudioReceiveStream(config);
Per Åhgrencc73ed32020-04-26 23:56:17 +0200125 EXPECT_NE(stream, nullptr);
126 call->DestroyAudioReceiveStream(stream);
127 }
solenbergc7a8b082015-10-16 14:35:07 -0700128}
129
130TEST(CallTest, CreateDestroy_AudioSendStreams) {
Per Åhgrencc73ed32020-04-26 23:56:17 +0200131 for (bool use_null_audio_processing : {false, true}) {
132 CallHelper call(use_null_audio_processing);
133 MockTransport send_transport;
134 AudioSendStream::Config config(&send_transport);
135 std::list<AudioSendStream*> streams;
136 for (int i = 0; i < 2; ++i) {
137 for (uint32_t ssrc = 0; ssrc < 1234567; ssrc += 34567) {
138 config.rtp.ssrc = ssrc;
139 AudioSendStream* stream = call->CreateAudioSendStream(config);
140 EXPECT_NE(stream, nullptr);
141 if (ssrc & 1) {
142 streams.push_back(stream);
143 } else {
144 streams.push_front(stream);
145 }
solenbergc7a8b082015-10-16 14:35:07 -0700146 }
Per Åhgrencc73ed32020-04-26 23:56:17 +0200147 for (auto s : streams) {
148 call->DestroyAudioSendStream(s);
149 }
150 streams.clear();
solenbergc7a8b082015-10-16 14:35:07 -0700151 }
solenbergc7a8b082015-10-16 14:35:07 -0700152 }
153}
154
155TEST(CallTest, CreateDestroy_AudioReceiveStreams) {
Per Åhgrencc73ed32020-04-26 23:56:17 +0200156 for (bool use_null_audio_processing : {false, true}) {
157 CallHelper call(use_null_audio_processing);
Tommi3176ef72022-05-22 20:47:28 +0200158 AudioReceiveStreamInterface::Config config;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200159 MockTransport rtcp_send_transport;
160 config.rtcp_send_transport = &rtcp_send_transport;
161 config.decoder_factory =
Tomas Gunnarssonc1d58912021-04-22 19:21:43 +0200162 rtc::make_ref_counted<webrtc::MockAudioDecoderFactory>();
Tommi3176ef72022-05-22 20:47:28 +0200163 std::list<AudioReceiveStreamInterface*> streams;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200164 for (int i = 0; i < 2; ++i) {
165 for (uint32_t ssrc = 0; ssrc < 1234567; ssrc += 34567) {
166 config.rtp.remote_ssrc = ssrc;
Tommi3176ef72022-05-22 20:47:28 +0200167 AudioReceiveStreamInterface* stream =
168 call->CreateAudioReceiveStream(config);
Per Åhgrencc73ed32020-04-26 23:56:17 +0200169 EXPECT_NE(stream, nullptr);
170 if (ssrc & 1) {
171 streams.push_back(stream);
172 } else {
173 streams.push_front(stream);
174 }
solenbergc7a8b082015-10-16 14:35:07 -0700175 }
Per Åhgrencc73ed32020-04-26 23:56:17 +0200176 for (auto s : streams) {
177 call->DestroyAudioReceiveStream(s);
178 }
179 streams.clear();
solenbergc7a8b082015-10-16 14:35:07 -0700180 }
solenbergc7a8b082015-10-16 14:35:07 -0700181 }
182}
brandtr25445d32016-10-23 23:37:14 -0700183
solenberg7602aab2016-11-14 11:30:07 -0800184TEST(CallTest, CreateDestroy_AssociateAudioSendReceiveStreams_RecvFirst) {
Per Åhgrencc73ed32020-04-26 23:56:17 +0200185 for (bool use_null_audio_processing : {false, true}) {
186 CallHelper call(use_null_audio_processing);
Tommi3176ef72022-05-22 20:47:28 +0200187 AudioReceiveStreamInterface::Config recv_config;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200188 MockTransport rtcp_send_transport;
189 recv_config.rtp.remote_ssrc = 42;
190 recv_config.rtp.local_ssrc = 777;
191 recv_config.rtcp_send_transport = &rtcp_send_transport;
192 recv_config.decoder_factory =
Tomas Gunnarssonc1d58912021-04-22 19:21:43 +0200193 rtc::make_ref_counted<webrtc::MockAudioDecoderFactory>();
Tommi3176ef72022-05-22 20:47:28 +0200194 AudioReceiveStreamInterface* recv_stream =
Per Åhgrencc73ed32020-04-26 23:56:17 +0200195 call->CreateAudioReceiveStream(recv_config);
196 EXPECT_NE(recv_stream, nullptr);
solenberg7602aab2016-11-14 11:30:07 -0800197
Per Åhgrencc73ed32020-04-26 23:56:17 +0200198 MockTransport send_transport;
199 AudioSendStream::Config send_config(&send_transport);
200 send_config.rtp.ssrc = 777;
201 AudioSendStream* send_stream = call->CreateAudioSendStream(send_config);
202 EXPECT_NE(send_stream, nullptr);
solenberg7602aab2016-11-14 11:30:07 -0800203
Tommidddbbeb2022-05-20 15:21:33 +0200204 AudioReceiveStreamImpl* internal_recv_stream =
205 static_cast<AudioReceiveStreamImpl*>(recv_stream);
Per Åhgrencc73ed32020-04-26 23:56:17 +0200206 EXPECT_EQ(send_stream,
207 internal_recv_stream->GetAssociatedSendStreamForTesting());
solenberg7602aab2016-11-14 11:30:07 -0800208
Per Åhgrencc73ed32020-04-26 23:56:17 +0200209 call->DestroyAudioSendStream(send_stream);
210 EXPECT_EQ(nullptr,
211 internal_recv_stream->GetAssociatedSendStreamForTesting());
Fredrik Solenberg8f5787a2018-01-11 13:52:30 +0100212
Per Åhgrencc73ed32020-04-26 23:56:17 +0200213 call->DestroyAudioReceiveStream(recv_stream);
214 }
solenberg7602aab2016-11-14 11:30:07 -0800215}
216
217TEST(CallTest, CreateDestroy_AssociateAudioSendReceiveStreams_SendFirst) {
Per Åhgrencc73ed32020-04-26 23:56:17 +0200218 for (bool use_null_audio_processing : {false, true}) {
219 CallHelper call(use_null_audio_processing);
220 MockTransport send_transport;
221 AudioSendStream::Config send_config(&send_transport);
222 send_config.rtp.ssrc = 777;
223 AudioSendStream* send_stream = call->CreateAudioSendStream(send_config);
224 EXPECT_NE(send_stream, nullptr);
solenberg7602aab2016-11-14 11:30:07 -0800225
Tommi3176ef72022-05-22 20:47:28 +0200226 AudioReceiveStreamInterface::Config recv_config;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200227 MockTransport rtcp_send_transport;
228 recv_config.rtp.remote_ssrc = 42;
229 recv_config.rtp.local_ssrc = 777;
230 recv_config.rtcp_send_transport = &rtcp_send_transport;
231 recv_config.decoder_factory =
Tomas Gunnarssonc1d58912021-04-22 19:21:43 +0200232 rtc::make_ref_counted<webrtc::MockAudioDecoderFactory>();
Tommi3176ef72022-05-22 20:47:28 +0200233 AudioReceiveStreamInterface* recv_stream =
Per Åhgrencc73ed32020-04-26 23:56:17 +0200234 call->CreateAudioReceiveStream(recv_config);
235 EXPECT_NE(recv_stream, nullptr);
solenberg7602aab2016-11-14 11:30:07 -0800236
Tommidddbbeb2022-05-20 15:21:33 +0200237 AudioReceiveStreamImpl* internal_recv_stream =
238 static_cast<AudioReceiveStreamImpl*>(recv_stream);
Per Åhgrencc73ed32020-04-26 23:56:17 +0200239 EXPECT_EQ(send_stream,
240 internal_recv_stream->GetAssociatedSendStreamForTesting());
Fredrik Solenberg8f5787a2018-01-11 13:52:30 +0100241
Per Åhgrencc73ed32020-04-26 23:56:17 +0200242 call->DestroyAudioReceiveStream(recv_stream);
solenberg7602aab2016-11-14 11:30:07 -0800243
Per Åhgrencc73ed32020-04-26 23:56:17 +0200244 call->DestroyAudioSendStream(send_stream);
245 }
solenberg7602aab2016-11-14 11:30:07 -0800246}
247
brandtr25445d32016-10-23 23:37:14 -0700248TEST(CallTest, CreateDestroy_FlexfecReceiveStream) {
Per Åhgrencc73ed32020-04-26 23:56:17 +0200249 for (bool use_null_audio_processing : {false, true}) {
250 CallHelper call(use_null_audio_processing);
251 MockTransport rtcp_send_transport;
252 FlexfecReceiveStream::Config config(&rtcp_send_transport);
253 config.payload_type = 118;
Tommi1c1f5402021-06-14 10:54:20 +0200254 config.rtp.remote_ssrc = 38837212;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200255 config.protected_media_ssrcs = {27273};
brandtr25445d32016-10-23 23:37:14 -0700256
Per Åhgrencc73ed32020-04-26 23:56:17 +0200257 FlexfecReceiveStream* stream = call->CreateFlexfecReceiveStream(config);
258 EXPECT_NE(stream, nullptr);
259 call->DestroyFlexfecReceiveStream(stream);
260 }
brandtr25445d32016-10-23 23:37:14 -0700261}
262
263TEST(CallTest, CreateDestroy_FlexfecReceiveStreams) {
Per Åhgrencc73ed32020-04-26 23:56:17 +0200264 for (bool use_null_audio_processing : {false, true}) {
265 CallHelper call(use_null_audio_processing);
266 MockTransport rtcp_send_transport;
267 FlexfecReceiveStream::Config config(&rtcp_send_transport);
268 config.payload_type = 118;
269 std::list<FlexfecReceiveStream*> streams;
brandtr25445d32016-10-23 23:37:14 -0700270
Per Åhgrencc73ed32020-04-26 23:56:17 +0200271 for (int i = 0; i < 2; ++i) {
272 for (uint32_t ssrc = 0; ssrc < 1234567; ssrc += 34567) {
Tommi1c1f5402021-06-14 10:54:20 +0200273 config.rtp.remote_ssrc = ssrc;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200274 config.protected_media_ssrcs = {ssrc + 1};
275 FlexfecReceiveStream* stream = call->CreateFlexfecReceiveStream(config);
276 EXPECT_NE(stream, nullptr);
277 if (ssrc & 1) {
278 streams.push_back(stream);
279 } else {
280 streams.push_front(stream);
281 }
brandtr25445d32016-10-23 23:37:14 -0700282 }
Per Åhgrencc73ed32020-04-26 23:56:17 +0200283 for (auto s : streams) {
284 call->DestroyFlexfecReceiveStream(s);
285 }
286 streams.clear();
brandtr25445d32016-10-23 23:37:14 -0700287 }
brandtr25445d32016-10-23 23:37:14 -0700288 }
289}
290
291TEST(CallTest, MultipleFlexfecReceiveStreamsProtectingSingleVideoStream) {
Per Åhgrencc73ed32020-04-26 23:56:17 +0200292 for (bool use_null_audio_processing : {false, true}) {
293 CallHelper call(use_null_audio_processing);
294 MockTransport rtcp_send_transport;
295 FlexfecReceiveStream::Config config(&rtcp_send_transport);
296 config.payload_type = 118;
297 config.protected_media_ssrcs = {1324234};
298 FlexfecReceiveStream* stream;
299 std::list<FlexfecReceiveStream*> streams;
brandtr25445d32016-10-23 23:37:14 -0700300
Tommi1c1f5402021-06-14 10:54:20 +0200301 config.rtp.remote_ssrc = 838383;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200302 stream = call->CreateFlexfecReceiveStream(config);
303 EXPECT_NE(stream, nullptr);
304 streams.push_back(stream);
brandtr25445d32016-10-23 23:37:14 -0700305
Tommi1c1f5402021-06-14 10:54:20 +0200306 config.rtp.remote_ssrc = 424993;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200307 stream = call->CreateFlexfecReceiveStream(config);
308 EXPECT_NE(stream, nullptr);
309 streams.push_back(stream);
brandtr25445d32016-10-23 23:37:14 -0700310
Tommi1c1f5402021-06-14 10:54:20 +0200311 config.rtp.remote_ssrc = 99383;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200312 stream = call->CreateFlexfecReceiveStream(config);
313 EXPECT_NE(stream, nullptr);
314 streams.push_back(stream);
brandtr25445d32016-10-23 23:37:14 -0700315
Tommi1c1f5402021-06-14 10:54:20 +0200316 config.rtp.remote_ssrc = 5548;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200317 stream = call->CreateFlexfecReceiveStream(config);
318 EXPECT_NE(stream, nullptr);
319 streams.push_back(stream);
brandtr25445d32016-10-23 23:37:14 -0700320
Per Åhgrencc73ed32020-04-26 23:56:17 +0200321 for (auto s : streams) {
322 call->DestroyFlexfecReceiveStream(s);
323 }
brandtr25445d32016-10-23 23:37:14 -0700324 }
325}
326
ossuc3d4b482017-05-23 06:07:11 -0700327TEST(CallTest, RecreatingAudioStreamWithSameSsrcReusesRtpState) {
328 constexpr uint32_t kSSRC = 12345;
Per Åhgrencc73ed32020-04-26 23:56:17 +0200329 for (bool use_null_audio_processing : {false, true}) {
330 CallHelper call(use_null_audio_processing);
ossuc3d4b482017-05-23 06:07:11 -0700331
Per Åhgrencc73ed32020-04-26 23:56:17 +0200332 auto create_stream_and_get_rtp_state = [&](uint32_t ssrc) {
333 MockTransport send_transport;
334 AudioSendStream::Config config(&send_transport);
335 config.rtp.ssrc = ssrc;
336 AudioSendStream* stream = call->CreateAudioSendStream(config);
337 const RtpState rtp_state =
338 static_cast<internal::AudioSendStream*>(stream)->GetRtpState();
339 call->DestroyAudioSendStream(stream);
340 return rtp_state;
341 };
ossuc3d4b482017-05-23 06:07:11 -0700342
Per Åhgrencc73ed32020-04-26 23:56:17 +0200343 const RtpState rtp_state1 = create_stream_and_get_rtp_state(kSSRC);
344 const RtpState rtp_state2 = create_stream_and_get_rtp_state(kSSRC);
ossuc3d4b482017-05-23 06:07:11 -0700345
Per Åhgrencc73ed32020-04-26 23:56:17 +0200346 EXPECT_EQ(rtp_state1.sequence_number, rtp_state2.sequence_number);
347 EXPECT_EQ(rtp_state1.start_timestamp, rtp_state2.start_timestamp);
348 EXPECT_EQ(rtp_state1.timestamp, rtp_state2.timestamp);
349 EXPECT_EQ(rtp_state1.capture_time_ms, rtp_state2.capture_time_ms);
350 EXPECT_EQ(rtp_state1.last_timestamp_time_ms,
351 rtp_state2.last_timestamp_time_ms);
Per Åhgrencc73ed32020-04-26 23:56:17 +0200352 }
ossuc3d4b482017-05-23 06:07:11 -0700353}
Fredrik Solenberg8f5787a2018-01-11 13:52:30 +0100354
Henrik Boströmf4a99912020-06-11 12:07:14 +0200355TEST(CallTest, AddAdaptationResourceAfterCreatingVideoSendStream) {
356 CallHelper call(true);
357 // Create a VideoSendStream.
358 test::FunctionVideoEncoderFactory fake_encoder_factory([]() {
359 return std::make_unique<test::FakeEncoder>(Clock::GetRealTimeClock());
360 });
361 auto bitrate_allocator_factory = CreateBuiltinVideoBitrateAllocatorFactory();
362 MockTransport send_transport;
363 VideoSendStream::Config config(&send_transport);
364 config.rtp.payload_type = 110;
365 config.rtp.ssrcs = {42};
366 config.encoder_settings.encoder_factory = &fake_encoder_factory;
367 config.encoder_settings.bitrate_allocator_factory =
368 bitrate_allocator_factory.get();
369 VideoEncoderConfig encoder_config;
370 encoder_config.max_bitrate_bps = 1337;
Henrik Boström29444c62020-07-01 15:48:46 +0200371 VideoSendStream* stream1 =
372 call->CreateVideoSendStream(config.Copy(), encoder_config.Copy());
373 EXPECT_NE(stream1, nullptr);
374 config.rtp.ssrcs = {43};
375 VideoSendStream* stream2 =
376 call->CreateVideoSendStream(config.Copy(), encoder_config.Copy());
377 EXPECT_NE(stream2, nullptr);
378 // Add a fake resource.
Henrik Boströmf4a99912020-06-11 12:07:14 +0200379 auto fake_resource = FakeResource::Create("FakeResource");
380 call->AddAdaptationResource(fake_resource);
Artem Titovea240272021-07-26 12:40:21 +0200381 // An adapter resource mirroring the `fake_resource` should now be present on
Henrik Boström29444c62020-07-01 15:48:46 +0200382 // both streams.
383 auto injected_resource1 = FindResourceWhoseNameContains(
384 stream1->GetAdaptationResources(), fake_resource->Name());
385 EXPECT_TRUE(injected_resource1);
386 auto injected_resource2 = FindResourceWhoseNameContains(
387 stream2->GetAdaptationResources(), fake_resource->Name());
388 EXPECT_TRUE(injected_resource2);
389 // Overwrite the real resource listeners with mock ones to verify the signal
390 // gets through.
391 injected_resource1->SetResourceListener(nullptr);
392 StrictMock<MockResourceListener> resource_listener1;
393 EXPECT_CALL(resource_listener1, OnResourceUsageStateMeasured(_, _))
394 .Times(1)
395 .WillOnce([injected_resource1](rtc::scoped_refptr<Resource> resource,
396 ResourceUsageState usage_state) {
397 EXPECT_EQ(injected_resource1, resource);
398 EXPECT_EQ(ResourceUsageState::kOveruse, usage_state);
399 });
400 injected_resource1->SetResourceListener(&resource_listener1);
401 injected_resource2->SetResourceListener(nullptr);
402 StrictMock<MockResourceListener> resource_listener2;
403 EXPECT_CALL(resource_listener2, OnResourceUsageStateMeasured(_, _))
404 .Times(1)
405 .WillOnce([injected_resource2](rtc::scoped_refptr<Resource> resource,
406 ResourceUsageState usage_state) {
407 EXPECT_EQ(injected_resource2, resource);
408 EXPECT_EQ(ResourceUsageState::kOveruse, usage_state);
409 });
410 injected_resource2->SetResourceListener(&resource_listener2);
411 // The kOveruse signal should get to our resource listeners.
412 fake_resource->SetUsageState(ResourceUsageState::kOveruse);
413 call->DestroyVideoSendStream(stream1);
414 call->DestroyVideoSendStream(stream2);
Henrik Boströmf4a99912020-06-11 12:07:14 +0200415}
416
417TEST(CallTest, AddAdaptationResourceBeforeCreatingVideoSendStream) {
418 CallHelper call(true);
419 // Add a fake resource.
420 auto fake_resource = FakeResource::Create("FakeResource");
421 call->AddAdaptationResource(fake_resource);
422 // Create a VideoSendStream.
423 test::FunctionVideoEncoderFactory fake_encoder_factory([]() {
424 return std::make_unique<test::FakeEncoder>(Clock::GetRealTimeClock());
425 });
426 auto bitrate_allocator_factory = CreateBuiltinVideoBitrateAllocatorFactory();
427 MockTransport send_transport;
428 VideoSendStream::Config config(&send_transport);
429 config.rtp.payload_type = 110;
430 config.rtp.ssrcs = {42};
431 config.encoder_settings.encoder_factory = &fake_encoder_factory;
432 config.encoder_settings.bitrate_allocator_factory =
433 bitrate_allocator_factory.get();
434 VideoEncoderConfig encoder_config;
435 encoder_config.max_bitrate_bps = 1337;
Henrik Boström29444c62020-07-01 15:48:46 +0200436 VideoSendStream* stream1 =
437 call->CreateVideoSendStream(config.Copy(), encoder_config.Copy());
438 EXPECT_NE(stream1, nullptr);
439 config.rtp.ssrcs = {43};
440 VideoSendStream* stream2 =
441 call->CreateVideoSendStream(config.Copy(), encoder_config.Copy());
442 EXPECT_NE(stream2, nullptr);
Artem Titovea240272021-07-26 12:40:21 +0200443 // An adapter resource mirroring the `fake_resource` should be present on both
Henrik Boström29444c62020-07-01 15:48:46 +0200444 // streams.
445 auto injected_resource1 = FindResourceWhoseNameContains(
446 stream1->GetAdaptationResources(), fake_resource->Name());
447 EXPECT_TRUE(injected_resource1);
448 auto injected_resource2 = FindResourceWhoseNameContains(
449 stream2->GetAdaptationResources(), fake_resource->Name());
450 EXPECT_TRUE(injected_resource2);
451 // Overwrite the real resource listeners with mock ones to verify the signal
452 // gets through.
453 injected_resource1->SetResourceListener(nullptr);
454 StrictMock<MockResourceListener> resource_listener1;
455 EXPECT_CALL(resource_listener1, OnResourceUsageStateMeasured(_, _))
456 .Times(1)
457 .WillOnce([injected_resource1](rtc::scoped_refptr<Resource> resource,
458 ResourceUsageState usage_state) {
459 EXPECT_EQ(injected_resource1, resource);
460 EXPECT_EQ(ResourceUsageState::kUnderuse, usage_state);
461 });
462 injected_resource1->SetResourceListener(&resource_listener1);
463 injected_resource2->SetResourceListener(nullptr);
464 StrictMock<MockResourceListener> resource_listener2;
465 EXPECT_CALL(resource_listener2, OnResourceUsageStateMeasured(_, _))
466 .Times(1)
467 .WillOnce([injected_resource2](rtc::scoped_refptr<Resource> resource,
468 ResourceUsageState usage_state) {
469 EXPECT_EQ(injected_resource2, resource);
470 EXPECT_EQ(ResourceUsageState::kUnderuse, usage_state);
471 });
472 injected_resource2->SetResourceListener(&resource_listener2);
473 // The kUnderuse signal should get to our resource listeners.
474 fake_resource->SetUsageState(ResourceUsageState::kUnderuse);
475 call->DestroyVideoSendStream(stream1);
476 call->DestroyVideoSendStream(stream2);
Henrik Boströmf4a99912020-06-11 12:07:14 +0200477}
478
Tommi25c77c12020-05-25 17:44:55 +0200479TEST(CallTest, SharedModuleThread) {
480 class SharedModuleThreadUser : public Module {
481 public:
482 SharedModuleThreadUser(ProcessThread* expected_thread,
483 rtc::scoped_refptr<SharedModuleThread> thread)
484 : expected_thread_(expected_thread), thread_(std::move(thread)) {
485 thread_->EnsureStarted();
486 thread_->process_thread()->RegisterModule(this, RTC_FROM_HERE);
487 }
488
489 ~SharedModuleThreadUser() override {
490 thread_->process_thread()->DeRegisterModule(this);
491 EXPECT_TRUE(thread_was_checked_);
492 }
493
494 private:
495 int64_t TimeUntilNextProcess() override { return 1000; }
496 void Process() override {}
497 void ProcessThreadAttached(ProcessThread* process_thread) override {
498 if (!process_thread) {
499 // Being detached.
500 return;
501 }
502 EXPECT_EQ(process_thread, expected_thread_);
503 thread_was_checked_ = true;
504 }
505
506 bool thread_was_checked_ = false;
507 ProcessThread* const expected_thread_;
508 rtc::scoped_refptr<SharedModuleThread> thread_;
509 };
510
511 // Create our test instance and pass a lambda to it that gets executed when
Artem Titovea240272021-07-26 12:40:21 +0200512 // the reference count goes back to 1 - meaning `shared` again is the only
Tommi25c77c12020-05-25 17:44:55 +0200513 // reference, which means we can free the variable and deallocate the thread.
514 rtc::scoped_refptr<SharedModuleThread> shared;
Per Kjellander4c50e702020-06-30 14:39:43 +0200515 shared =
516 SharedModuleThread::Create(ProcessThread::Create("MySharedProcessThread"),
517 [&shared]() { shared = nullptr; });
Tommi25c77c12020-05-25 17:44:55 +0200518 ProcessThread* process_thread = shared->process_thread();
519
520 ASSERT_TRUE(shared.get());
521
522 {
523 // Create a couple of users of the thread.
524 // These instances are in a separate scope to trigger the callback to our
525 // lambda, which will run when these go out of scope.
526 SharedModuleThreadUser user1(process_thread, shared);
527 SharedModuleThreadUser user2(process_thread, shared);
528 }
529
530 // The thread should now have been stopped and freed.
531 EXPECT_FALSE(shared);
532}
533
solenbergc7a8b082015-10-16 14:35:07 -0700534} // namespace webrtc