blob: 63f78c8f7145359c023e632cc8ec73aec5b13cb1 [file] [log] [blame]
Sebastian Jansson98b07e92018-09-27 13:47:01 +02001/*
2 * Copyright 2018 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#include "test/scenario/audio_stream.h"
11
Steve Anton40d55332019-01-07 10:21:47 -080012#include "absl/memory/memory.h"
Sebastian Jansson98b07e92018-09-27 13:47:01 +020013#include "test/call_test.h"
14
Sebastian Janssonb9972fa2018-10-17 16:27:55 +020015#if WEBRTC_ENABLE_PROTOBUF
16RTC_PUSH_IGNORING_WUNDEF()
17#ifdef WEBRTC_ANDROID_PLATFORM_BUILD
18#include "external/webrtc/webrtc/modules/audio_coding/audio_network_adaptor/config.pb.h"
19#else
20#include "modules/audio_coding/audio_network_adaptor/config.pb.h"
21#endif
22RTC_POP_IGNORING_WUNDEF()
23#endif
24
Sebastian Jansson98b07e92018-09-27 13:47:01 +020025namespace webrtc {
26namespace test {
Sebastian Janssonb9972fa2018-10-17 16:27:55 +020027namespace {
Sebastian Janssone112bb82019-06-13 17:36:01 +020028enum : int { // The first valid value is 1.
29 kTransportSequenceNumberExtensionId = 1,
30 kAbsSendTimeExtensionId
31};
32
Sebastian Janssonb9972fa2018-10-17 16:27:55 +020033absl::optional<std::string> CreateAdaptationString(
34 AudioStreamConfig::NetworkAdaptation config) {
35#if WEBRTC_ENABLE_PROTOBUF
36
37 audio_network_adaptor::config::ControllerManager cont_conf;
38 if (config.frame.max_rate_for_60_ms.IsFinite()) {
39 auto controller =
40 cont_conf.add_controllers()->mutable_frame_length_controller();
41 controller->set_fl_decreasing_packet_loss_fraction(
42 config.frame.min_packet_loss_for_decrease);
43 controller->set_fl_increasing_packet_loss_fraction(
44 config.frame.max_packet_loss_for_increase);
45
46 controller->set_fl_20ms_to_60ms_bandwidth_bps(
47 config.frame.min_rate_for_20_ms.bps<int32_t>());
48 controller->set_fl_60ms_to_20ms_bandwidth_bps(
49 config.frame.max_rate_for_60_ms.bps<int32_t>());
50
51 if (config.frame.max_rate_for_120_ms.IsFinite()) {
52 controller->set_fl_60ms_to_120ms_bandwidth_bps(
53 config.frame.min_rate_for_60_ms.bps<int32_t>());
54 controller->set_fl_120ms_to_60ms_bandwidth_bps(
55 config.frame.max_rate_for_120_ms.bps<int32_t>());
56 }
57 }
58 cont_conf.add_controllers()->mutable_bitrate_controller();
59 std::string config_string = cont_conf.SerializeAsString();
60 return config_string;
61#else
62 RTC_LOG(LS_ERROR) << "audio_network_adaptation is enabled"
63 " but WEBRTC_ENABLE_PROTOBUF is false.\n"
64 "Ignoring settings.";
65 return absl::nullopt;
66#endif // WEBRTC_ENABLE_PROTOBUF
67}
68} // namespace
Sebastian Jansson98b07e92018-09-27 13:47:01 +020069
70SendAudioStream::SendAudioStream(
71 CallClient* sender,
72 AudioStreamConfig config,
73 rtc::scoped_refptr<AudioEncoderFactory> encoder_factory,
74 Transport* send_transport)
75 : sender_(sender), config_(config) {
Bjorn A Mellem7a9a0922019-11-26 09:19:40 -080076 AudioSendStream::Config send_config(send_transport);
Sebastian Jansson98b07e92018-09-27 13:47:01 +020077 ssrc_ = sender->GetNextAudioSsrc();
78 send_config.rtp.ssrc = ssrc_;
79 SdpAudioFormat::Parameters sdp_params;
80 if (config.source.channels == 2)
81 sdp_params["stereo"] = "1";
Danil Chapovalov0c626af2020-02-10 11:16:00 +010082 if (config.encoder.initial_frame_length != TimeDelta::Millis(20))
Sebastian Jansson98b07e92018-09-27 13:47:01 +020083 sdp_params["ptime"] =
84 std::to_string(config.encoder.initial_frame_length.ms());
Sebastian Janssonad871942019-01-16 17:21:28 +010085 if (config.encoder.enable_dtx)
86 sdp_params["usedtx"] = "1";
Sebastian Jansson98b07e92018-09-27 13:47:01 +020087
88 // SdpAudioFormat::num_channels indicates that the encoder is capable of
89 // stereo, but the actual channel count used is based on the "stereo"
90 // parameter.
91 send_config.send_codec_spec = AudioSendStream::Config::SendCodecSpec(
92 CallTest::kAudioSendPayloadType, {"opus", 48000, 2, sdp_params});
93 RTC_DCHECK_LE(config.source.channels, 2);
94 send_config.encoder_factory = encoder_factory;
95
96 if (config.encoder.fixed_rate)
97 send_config.send_codec_spec->target_bitrate_bps =
98 config.encoder.fixed_rate->bps();
Sebastian Jansson7c1ac762020-02-27 12:04:20 +010099 if (!config.adapt.binary_proto.empty()) {
100 send_config.audio_network_adaptor_config = config.adapt.binary_proto;
101 } else if (config.network_adaptation) {
Sebastian Janssonb9972fa2018-10-17 16:27:55 +0200102 send_config.audio_network_adaptor_config =
103 CreateAdaptationString(config.adapt);
104 }
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200105 if (config.encoder.allocate_bitrate ||
106 config.stream.in_bandwidth_estimation) {
107 DataRate min_rate = DataRate::Infinity();
108 DataRate max_rate = DataRate::Infinity();
109 if (config.encoder.fixed_rate) {
110 min_rate = *config.encoder.fixed_rate;
111 max_rate = *config.encoder.fixed_rate;
112 } else {
113 min_rate = *config.encoder.min_rate;
114 max_rate = *config.encoder.max_rate;
115 }
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200116 send_config.min_bitrate_bps = min_rate.bps();
117 send_config.max_bitrate_bps = max_rate.bps();
118 }
119
120 if (config.stream.in_bandwidth_estimation) {
121 send_config.send_codec_spec->transport_cc_enabled = true;
Sebastian Janssone112bb82019-06-13 17:36:01 +0200122 send_config.rtp.extensions = {{RtpExtension::kTransportSequenceNumberUri,
123 kTransportSequenceNumberExtensionId}};
124 }
125 if (config.stream.abs_send_time) {
126 send_config.rtp.extensions.push_back(
127 {RtpExtension::kAbsSendTimeUri, kAbsSendTimeExtensionId});
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200128 }
129
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200130 sender_->SendTask([&] {
131 send_stream_ = sender_->call_->CreateAudioSendStream(send_config);
132 if (field_trial::IsEnabled("WebRTC-SendSideBwe-WithOverhead")) {
133 sender->call_->OnAudioTransportOverheadChanged(
134 sender_->transport_->packet_overhead().bytes());
135 }
136 });
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200137}
138
139SendAudioStream::~SendAudioStream() {
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200140 sender_->SendTask(
141 [this] { sender_->call_->DestroyAudioSendStream(send_stream_); });
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200142}
143
144void SendAudioStream::Start() {
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200145 sender_->SendTask([this] {
146 send_stream_->Start();
147 sender_->call_->SignalChannelNetworkState(MediaType::AUDIO, kNetworkUp);
148 });
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200149}
150
Sebastian Janssonbdfadd62019-02-08 13:34:57 +0100151void SendAudioStream::Stop() {
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200152 sender_->SendTask([this] { send_stream_->Stop(); });
Sebastian Janssonbdfadd62019-02-08 13:34:57 +0100153}
154
Sebastian Janssonad871942019-01-16 17:21:28 +0100155void SendAudioStream::SetMuted(bool mute) {
Sebastian Jansson9f215a72020-02-27 11:51:53 +0100156 sender_->SendTask([this, mute] { send_stream_->SetMuted(mute); });
Sebastian Janssonad871942019-01-16 17:21:28 +0100157}
158
Sebastian Jansson359d60a2018-10-25 16:22:02 +0200159ColumnPrinter SendAudioStream::StatsPrinter() {
160 return ColumnPrinter::Lambda(
161 "audio_target_rate",
162 [this](rtc::SimpleStringBuilder& sb) {
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200163 sender_->SendTask([this, &sb] {
164 AudioSendStream::Stats stats = send_stream_->GetStats();
165 sb.AppendFormat("%.0lf", stats.target_bitrate_bps / 8.0);
166 });
Sebastian Jansson359d60a2018-10-25 16:22:02 +0200167 },
168 64);
169}
170
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200171ReceiveAudioStream::ReceiveAudioStream(
172 CallClient* receiver,
173 AudioStreamConfig config,
174 SendAudioStream* send_stream,
175 rtc::scoped_refptr<AudioDecoderFactory> decoder_factory,
176 Transport* feedback_transport)
177 : receiver_(receiver), config_(config) {
178 AudioReceiveStream::Config recv_config;
Sebastian Jansson5fbebd52019-02-20 11:16:19 +0100179 recv_config.rtp.local_ssrc = receiver_->GetNextAudioLocalSsrc();
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200180 recv_config.rtcp_send_transport = feedback_transport;
181 recv_config.rtp.remote_ssrc = send_stream->ssrc_;
Sebastian Jansson800e1212018-10-22 11:49:03 +0200182 receiver->ssrc_media_types_[recv_config.rtp.remote_ssrc] = MediaType::AUDIO;
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200183 if (config.stream.in_bandwidth_estimation) {
184 recv_config.rtp.transport_cc = true;
Sebastian Jansson63c38e22019-08-06 17:17:43 +0200185 recv_config.rtp.extensions = {{RtpExtension::kTransportSequenceNumberUri,
186 kTransportSequenceNumberExtensionId}};
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200187 }
188 recv_config.decoder_factory = decoder_factory;
189 recv_config.decoder_map = {
190 {CallTest::kAudioSendPayloadType, {"opus", 48000, 2}}};
191 recv_config.sync_group = config.render.sync_group;
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200192 receiver_->SendTask([&] {
193 receive_stream_ = receiver_->call_->CreateAudioReceiveStream(recv_config);
194 });
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200195}
196ReceiveAudioStream::~ReceiveAudioStream() {
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200197 receiver_->SendTask(
198 [&] { receiver_->call_->DestroyAudioReceiveStream(receive_stream_); });
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200199}
200
Sebastian Jansson49a78432018-11-20 16:15:29 +0100201void ReceiveAudioStream::Start() {
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200202 receiver_->SendTask([&] {
203 receive_stream_->Start();
204 receiver_->call_->SignalChannelNetworkState(MediaType::AUDIO, kNetworkUp);
205 });
Sebastian Jansson49a78432018-11-20 16:15:29 +0100206}
207
Sebastian Janssonbdfadd62019-02-08 13:34:57 +0100208void ReceiveAudioStream::Stop() {
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200209 receiver_->SendTask([&] { receive_stream_->Stop(); });
Sebastian Janssonbdfadd62019-02-08 13:34:57 +0100210}
211
Sebastian Janssonf4481c82019-04-09 12:48:34 +0200212AudioReceiveStream::Stats ReceiveAudioStream::GetStats() const {
213 AudioReceiveStream::Stats result;
Niels Möller6b4d9622020-09-14 10:47:50 +0200214 receiver_->SendTask([&] {
215 result = receive_stream_->GetStats(/*get_and_clear_legacy_stats=*/true);
216 });
Sebastian Janssonf4481c82019-04-09 12:48:34 +0200217 return result;
218}
219
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200220AudioStreamPair::~AudioStreamPair() = default;
221
222AudioStreamPair::AudioStreamPair(
223 CallClient* sender,
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200224 rtc::scoped_refptr<AudioEncoderFactory> encoder_factory,
225 CallClient* receiver,
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200226 rtc::scoped_refptr<AudioDecoderFactory> decoder_factory,
227 AudioStreamConfig config)
228 : config_(config),
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200229 send_stream_(sender, config, encoder_factory, sender->transport_.get()),
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200230 receive_stream_(receiver,
231 config,
232 &send_stream_,
233 decoder_factory,
Sebastian Jansson105a10a2019-04-01 09:18:14 +0200234 receiver->transport_.get()) {}
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200235
236} // namespace test
237} // namespace webrtc