OLD | NEW |
1 /* | 1 /* |
2 * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved. | 2 * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license | 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 | 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 | 6 * tree. An additional intellectual property rights grant can be found |
7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
9 */ | 9 */ |
10 | 10 |
11 #include "webrtc/modules/audio_coding/codecs/opus/audio_encoder_opus.h" | 11 #include "webrtc/modules/audio_coding/codecs/opus/audio_encoder_opus.h" |
12 | 12 |
13 #include <algorithm> | 13 #include <algorithm> |
14 #include <iterator> | 14 #include <iterator> |
| 15 #include <utility> |
15 | 16 |
| 17 #include "webrtc/base/arraysize.h" |
16 #include "webrtc/base/checks.h" | 18 #include "webrtc/base/checks.h" |
17 #include "webrtc/base/logging.h" | 19 #include "webrtc/base/logging.h" |
18 #include "webrtc/base/numerics/exp_filter.h" | 20 #include "webrtc/base/numerics/exp_filter.h" |
19 #include "webrtc/base/safe_conversions.h" | 21 #include "webrtc/base/safe_conversions.h" |
| 22 #include "webrtc/base/string_to_number.h" |
20 #include "webrtc/base/timeutils.h" | 23 #include "webrtc/base/timeutils.h" |
21 #include "webrtc/common_types.h" | 24 #include "webrtc/common_types.h" |
22 #include "webrtc/modules/audio_coding/audio_network_adaptor/audio_network_adapto
r_impl.h" | 25 #include "webrtc/modules/audio_coding/audio_network_adaptor/audio_network_adapto
r_impl.h" |
23 #include "webrtc/modules/audio_coding/audio_network_adaptor/controller_manager.h
" | 26 #include "webrtc/modules/audio_coding/audio_network_adaptor/controller_manager.h
" |
24 #include "webrtc/modules/audio_coding/codecs/opus/opus_interface.h" | 27 #include "webrtc/modules/audio_coding/codecs/opus/opus_interface.h" |
25 #include "webrtc/system_wrappers/include/field_trial.h" | 28 #include "webrtc/system_wrappers/include/field_trial.h" |
26 | 29 |
27 namespace webrtc { | 30 namespace webrtc { |
28 | 31 |
29 namespace { | 32 namespace { |
30 | 33 |
31 constexpr int kSampleRateHz = 48000; | 34 // Codec parameters for Opus. |
| 35 // draft-spittka-payload-rtp-opus-03 |
| 36 |
| 37 // Recommended bitrates: |
| 38 // 8-12 kb/s for NB speech, |
| 39 // 16-20 kb/s for WB speech, |
| 40 // 28-40 kb/s for FB speech, |
| 41 // 48-64 kb/s for FB mono music, and |
| 42 // 64-128 kb/s for FB stereo music. |
| 43 // The current implementation applies the following values to mono signals, |
| 44 // and multiplies them by 2 for stereo. |
| 45 constexpr int kOpusBitrateNbBps = 12000; |
| 46 constexpr int kOpusBitrateWbBps = 20000; |
| 47 constexpr int kOpusBitrateFbBps = 32000; |
32 | 48 |
33 // Opus API allows a min bitrate of 500bps, but Opus documentation suggests | 49 // Opus API allows a min bitrate of 500bps, but Opus documentation suggests |
34 // a minimum bitrate of 6kbps. | 50 // bitrate should be in the range of 6000 to 510000, inclusive. |
35 constexpr int kMinBitrateBps = 6000; | 51 constexpr int kOpusMinBitrateBps = 6000; |
| 52 constexpr int kOpusMaxBitrateBps = 510000; |
36 | 53 |
37 constexpr int kMaxBitrateBps = 512000; | 54 constexpr int kSampleRateHz = 48000; |
| 55 constexpr int kDefaultMaxPlaybackRate = 48000; |
38 | 56 |
| 57 // These two lists must be sorted from low to high |
39 #if WEBRTC_OPUS_SUPPORT_120MS_PTIME | 58 #if WEBRTC_OPUS_SUPPORT_120MS_PTIME |
40 constexpr int kSupportedFrameLengths[] = {20, 60, 120}; | 59 constexpr int kANASupportedFrameLengths[] = {20, 60, 120}; |
| 60 constexpr int kOpusSupportedFrameLengths[] = {10, 20, 40, 60, 120}; |
41 #else | 61 #else |
42 constexpr int kSupportedFrameLengths[] = {20, 60}; | 62 constexpr int kANASupportedFrameLengths[] = {20, 60}; |
| 63 constexpr int kOpusSupportedFrameLengths[] = {10, 20, 40, 60}; |
43 #endif | 64 #endif |
44 | 65 |
45 // PacketLossFractionSmoother uses an exponential filter with a time constant | 66 // PacketLossFractionSmoother uses an exponential filter with a time constant |
46 // of -1.0 / ln(0.9999) = 10000 ms. | 67 // of -1.0 / ln(0.9999) = 10000 ms. |
47 constexpr float kAlphaForPacketLossFractionSmoother = 0.9999f; | 68 constexpr float kAlphaForPacketLossFractionSmoother = 0.9999f; |
48 | 69 |
49 AudioEncoderOpus::Config CreateConfig(const CodecInst& codec_inst) { | |
50 AudioEncoderOpus::Config config; | |
51 config.frame_size_ms = rtc::CheckedDivExact(codec_inst.pacsize, 48); | |
52 config.num_channels = codec_inst.channels; | |
53 config.bitrate_bps = rtc::Optional<int>(codec_inst.rate); | |
54 config.payload_type = codec_inst.pltype; | |
55 config.application = config.num_channels == 1 ? AudioEncoderOpus::kVoip | |
56 : AudioEncoderOpus::kAudio; | |
57 config.supported_frame_lengths_ms.push_back(config.frame_size_ms); | |
58 #if WEBRTC_OPUS_VARIABLE_COMPLEXITY | |
59 config.low_rate_complexity = 9; | |
60 #endif | |
61 return config; | |
62 } | |
63 | |
64 // Optimize the loss rate to configure Opus. Basically, optimized loss rate is | 70 // Optimize the loss rate to configure Opus. Basically, optimized loss rate is |
65 // the input loss rate rounded down to various levels, because a robustly good | 71 // the input loss rate rounded down to various levels, because a robustly good |
66 // audio quality is achieved by lowering the packet loss down. | 72 // audio quality is achieved by lowering the packet loss down. |
67 // Additionally, to prevent toggling, margins are used, i.e., when jumping to | 73 // Additionally, to prevent toggling, margins are used, i.e., when jumping to |
68 // a loss rate from below, a higher threshold is used than jumping to the same | 74 // a loss rate from below, a higher threshold is used than jumping to the same |
69 // level from above. | 75 // level from above. |
70 float OptimizePacketLossRate(float new_loss_rate, float old_loss_rate) { | 76 float OptimizePacketLossRate(float new_loss_rate, float old_loss_rate) { |
71 RTC_DCHECK_GE(new_loss_rate, 0.0f); | 77 RTC_DCHECK_GE(new_loss_rate, 0.0f); |
72 RTC_DCHECK_LE(new_loss_rate, 1.0f); | 78 RTC_DCHECK_LE(new_loss_rate, 1.0f); |
73 RTC_DCHECK_GE(old_loss_rate, 0.0f); | 79 RTC_DCHECK_GE(old_loss_rate, 0.0f); |
(...skipping 20 matching lines...) Expand all Loading... |
94 kLossRate5Margin * | 100 kLossRate5Margin * |
95 (kPacketLossRate5 - old_loss_rate > 0 ? 1 : -1)) { | 101 (kPacketLossRate5 - old_loss_rate > 0 ? 1 : -1)) { |
96 return kPacketLossRate5; | 102 return kPacketLossRate5; |
97 } else if (new_loss_rate >= kPacketLossRate1) { | 103 } else if (new_loss_rate >= kPacketLossRate1) { |
98 return kPacketLossRate1; | 104 return kPacketLossRate1; |
99 } else { | 105 } else { |
100 return 0.0f; | 106 return 0.0f; |
101 } | 107 } |
102 } | 108 } |
103 | 109 |
| 110 rtc::Optional<std::string> GetFormatParameter(const SdpAudioFormat& format, |
| 111 const std::string& param) { |
| 112 auto it = format.parameters.find(param); |
| 113 return (it == format.parameters.end()) |
| 114 ? rtc::Optional<std::string>() |
| 115 : rtc::Optional<std::string>(it->second); |
| 116 } |
| 117 |
| 118 template <typename T> |
| 119 rtc::Optional<T> GetFormatParameter(const SdpAudioFormat& format, |
| 120 const std::string& param) { |
| 121 return rtc::StringToNumber<T>(GetFormatParameter(format, param).value_or("")); |
| 122 } |
| 123 |
| 124 int CalculateDefaultBitrate(int max_playback_rate, size_t num_channels) { |
| 125 const int bitrate = [&] { |
| 126 if (max_playback_rate <= 8000) { |
| 127 return kOpusBitrateNbBps * rtc::dchecked_cast<int>(num_channels); |
| 128 } else if (max_playback_rate <= 16000) { |
| 129 return kOpusBitrateWbBps * rtc::dchecked_cast<int>(num_channels); |
| 130 } else { |
| 131 return kOpusBitrateFbBps * rtc::dchecked_cast<int>(num_channels); |
| 132 } |
| 133 }(); |
| 134 RTC_DCHECK_GE(bitrate, kOpusMinBitrateBps); |
| 135 RTC_DCHECK_LE(bitrate, kOpusMaxBitrateBps); |
| 136 return bitrate; |
| 137 } |
| 138 |
| 139 // Get the maxaveragebitrate parameter in string-form, so we can properly figure |
| 140 // out how invalid it is and accurately log invalid values. |
| 141 int CalculateBitrate(int max_playback_rate_hz, |
| 142 size_t num_channels, |
| 143 rtc::Optional<std::string> bitrate_param) { |
| 144 const int default_bitrate = |
| 145 CalculateDefaultBitrate(max_playback_rate_hz, num_channels); |
| 146 |
| 147 if (bitrate_param) { |
| 148 const auto bitrate = rtc::StringToNumber<int>(*bitrate_param); |
| 149 if (bitrate) { |
| 150 const int chosen_bitrate = |
| 151 std::max(kOpusMinBitrateBps, std::min(*bitrate, kOpusMaxBitrateBps)); |
| 152 if (bitrate != chosen_bitrate) { |
| 153 LOG(LS_WARNING) << "Invalid maxaveragebitrate " << *bitrate |
| 154 << " clamped to " << chosen_bitrate; |
| 155 } |
| 156 return chosen_bitrate; |
| 157 } |
| 158 LOG(LS_WARNING) << "Invalid maxaveragebitrate \"" << *bitrate_param |
| 159 << "\" replaced by default bitrate " << default_bitrate; |
| 160 } |
| 161 |
| 162 return default_bitrate; |
| 163 } |
| 164 |
| 165 int GetChannelCount(const SdpAudioFormat& format) { |
| 166 const auto param = GetFormatParameter(format, "stereo"); |
| 167 if (param == "1") { |
| 168 return 2; |
| 169 } else { |
| 170 return 1; |
| 171 } |
| 172 } |
| 173 |
| 174 int GetMaxPlaybackRate(const SdpAudioFormat& format) { |
| 175 const auto param = GetFormatParameter<int>(format, "maxplaybackrate"); |
| 176 if (param && *param >= 8000) { |
| 177 return std::min(*param, kDefaultMaxPlaybackRate); |
| 178 } |
| 179 return kDefaultMaxPlaybackRate; |
| 180 } |
| 181 |
| 182 int GetFrameSizeMs(const SdpAudioFormat& format) { |
| 183 const auto ptime = GetFormatParameter<int>(format, "ptime"); |
| 184 if (ptime) { |
| 185 // Pick the next highest supported frame length from |
| 186 // kOpusSupportedFrameLengths. |
| 187 for (const int supported_frame_length : kOpusSupportedFrameLengths) { |
| 188 if (supported_frame_length >= *ptime) { |
| 189 return supported_frame_length; |
| 190 } |
| 191 } |
| 192 // If none was found, return the largest supported frame length. |
| 193 return *(std::end(kOpusSupportedFrameLengths) - 1); |
| 194 } |
| 195 |
| 196 return AudioEncoderOpus::Config::kDefaultFrameSizeMs; |
| 197 } |
| 198 |
| 199 void FindSupportedFrameLengths(int min_frame_length_ms, |
| 200 int max_frame_length_ms, |
| 201 std::vector<int>* out) { |
| 202 out->clear(); |
| 203 std::copy_if(std::begin(kANASupportedFrameLengths), |
| 204 std::end(kANASupportedFrameLengths), std::back_inserter(*out), |
| 205 [&](int frame_length_ms) { |
| 206 return frame_length_ms >= min_frame_length_ms && |
| 207 frame_length_ms <= max_frame_length_ms; |
| 208 }); |
| 209 RTC_DCHECK(std::is_sorted(out->begin(), out->end())); |
| 210 } |
| 211 |
104 } // namespace | 212 } // namespace |
105 | 213 |
| 214 rtc::Optional<AudioCodecInfo> AudioEncoderOpus::QueryAudioEncoder( |
| 215 const SdpAudioFormat& format) { |
| 216 if (STR_CASE_CMP(format.name.c_str(), GetPayloadName()) == 0 && |
| 217 format.clockrate_hz == 48000 && format.num_channels == 2) { |
| 218 const size_t num_channels = GetChannelCount(format); |
| 219 const int bitrate = |
| 220 CalculateBitrate(GetMaxPlaybackRate(format), num_channels, |
| 221 GetFormatParameter(format, "maxaveragebitrate")); |
| 222 AudioCodecInfo info(48000, num_channels, bitrate, kOpusMinBitrateBps, |
| 223 kOpusMaxBitrateBps); |
| 224 info.allow_comfort_noise = false; |
| 225 info.supports_network_adaption = true; |
| 226 |
| 227 return rtc::Optional<AudioCodecInfo>(info); |
| 228 } |
| 229 return rtc::Optional<AudioCodecInfo>(); |
| 230 } |
| 231 |
| 232 AudioEncoderOpus::Config AudioEncoderOpus::CreateConfig( |
| 233 const CodecInst& codec_inst) { |
| 234 AudioEncoderOpus::Config config; |
| 235 config.frame_size_ms = rtc::CheckedDivExact(codec_inst.pacsize, 48); |
| 236 config.num_channels = codec_inst.channels; |
| 237 config.bitrate_bps = rtc::Optional<int>(codec_inst.rate); |
| 238 config.payload_type = codec_inst.pltype; |
| 239 config.application = config.num_channels == 1 ? AudioEncoderOpus::kVoip |
| 240 : AudioEncoderOpus::kAudio; |
| 241 config.supported_frame_lengths_ms.push_back(config.frame_size_ms); |
| 242 #if WEBRTC_OPUS_VARIABLE_COMPLEXITY |
| 243 config.low_rate_complexity = 9; |
| 244 #endif |
| 245 return config; |
| 246 } |
| 247 |
| 248 AudioEncoderOpus::Config AudioEncoderOpus::CreateConfig( |
| 249 int payload_type, |
| 250 const SdpAudioFormat& format) { |
| 251 AudioEncoderOpus::Config config; |
| 252 |
| 253 config.num_channels = GetChannelCount(format); |
| 254 config.frame_size_ms = GetFrameSizeMs(format); |
| 255 config.max_playback_rate_hz = GetMaxPlaybackRate(format); |
| 256 config.fec_enabled = (GetFormatParameter(format, "useinbandfec") == "1"); |
| 257 config.dtx_enabled = (GetFormatParameter(format, "usedtx") == "1"); |
| 258 config.bitrate_bps = rtc::Optional<int>( |
| 259 CalculateBitrate(config.max_playback_rate_hz, config.num_channels, |
| 260 GetFormatParameter(format, "maxaveragebitrate"))); |
| 261 config.payload_type = payload_type; |
| 262 config.application = config.num_channels == 1 ? AudioEncoderOpus::kVoip |
| 263 : AudioEncoderOpus::kAudio; |
| 264 #if WEBRTC_OPUS_VARIABLE_COMPLEXITY |
| 265 config.low_rate_complexity = 9; |
| 266 #endif |
| 267 |
| 268 constexpr int kMinANAFrameLength = kANASupportedFrameLengths[0]; |
| 269 constexpr int kMaxANAFrameLength = |
| 270 kANASupportedFrameLengths[arraysize(kANASupportedFrameLengths) - 1]; |
| 271 // For now, minptime and maxptime are only used with ANA. If ptime is outside |
| 272 // of this range, it will get adjusted once ANA takes hold. Ideally, we'd know |
| 273 // if ANA was to be used when setting up the config, and adjust accordingly. |
| 274 const int min_frame_length_ms = |
| 275 GetFormatParameter<int>(format, "minptime").value_or(kMinANAFrameLength); |
| 276 const int max_frame_length_ms = |
| 277 GetFormatParameter<int>(format, "maxptime").value_or(kMaxANAFrameLength); |
| 278 |
| 279 FindSupportedFrameLengths(min_frame_length_ms, max_frame_length_ms, |
| 280 &config.supported_frame_lengths_ms); |
| 281 return config; |
| 282 } |
| 283 |
106 class AudioEncoderOpus::PacketLossFractionSmoother { | 284 class AudioEncoderOpus::PacketLossFractionSmoother { |
107 public: | 285 public: |
108 explicit PacketLossFractionSmoother(const Clock* clock) | 286 explicit PacketLossFractionSmoother(const Clock* clock) |
109 : clock_(clock), | 287 : clock_(clock), |
110 last_sample_time_ms_(clock_->TimeInMilliseconds()), | 288 last_sample_time_ms_(clock_->TimeInMilliseconds()), |
111 smoother_(kAlphaForPacketLossFractionSmoother) {} | 289 smoother_(kAlphaForPacketLossFractionSmoother) {} |
112 | 290 |
113 // Gets the smoothed packet loss fraction. | 291 // Gets the smoothed packet loss fraction. |
114 float GetAverage() const { | 292 float GetAverage() const { |
115 float value = smoother_.filtered(); | 293 float value = smoother_.filtered(); |
(...skipping 24 matching lines...) Expand all Loading... |
140 AudioEncoderOpus::Config::Config(const Config&) = default; | 318 AudioEncoderOpus::Config::Config(const Config&) = default; |
141 AudioEncoderOpus::Config::~Config() = default; | 319 AudioEncoderOpus::Config::~Config() = default; |
142 auto AudioEncoderOpus::Config::operator=(const Config&) -> Config& = default; | 320 auto AudioEncoderOpus::Config::operator=(const Config&) -> Config& = default; |
143 | 321 |
144 bool AudioEncoderOpus::Config::IsOk() const { | 322 bool AudioEncoderOpus::Config::IsOk() const { |
145 if (frame_size_ms <= 0 || frame_size_ms % 10 != 0) | 323 if (frame_size_ms <= 0 || frame_size_ms % 10 != 0) |
146 return false; | 324 return false; |
147 if (num_channels != 1 && num_channels != 2) | 325 if (num_channels != 1 && num_channels != 2) |
148 return false; | 326 return false; |
149 if (bitrate_bps && | 327 if (bitrate_bps && |
150 (*bitrate_bps < kMinBitrateBps || *bitrate_bps > kMaxBitrateBps)) | 328 (*bitrate_bps < kOpusMinBitrateBps || *bitrate_bps > kOpusMaxBitrateBps)) |
151 return false; | 329 return false; |
152 if (complexity < 0 || complexity > 10) | 330 if (complexity < 0 || complexity > 10) |
153 return false; | 331 return false; |
154 if (low_rate_complexity < 0 || low_rate_complexity > 10) | 332 if (low_rate_complexity < 0 || low_rate_complexity > 10) |
155 return false; | 333 return false; |
156 return true; | 334 return true; |
157 } | 335 } |
158 | 336 |
159 int AudioEncoderOpus::Config::GetBitrateBps() const { | 337 int AudioEncoderOpus::Config::GetBitrateBps() const { |
160 RTC_DCHECK(IsOk()); | 338 RTC_DCHECK(IsOk()); |
(...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
201 bitrate_smoother_(bitrate_smoother | 379 bitrate_smoother_(bitrate_smoother |
202 ? std::move(bitrate_smoother) : std::unique_ptr<SmoothingFilter>( | 380 ? std::move(bitrate_smoother) : std::unique_ptr<SmoothingFilter>( |
203 // We choose 5sec as initial time constant due to empirical data. | 381 // We choose 5sec as initial time constant due to empirical data. |
204 new SmoothingFilterImpl(5000, config.clock))) { | 382 new SmoothingFilterImpl(5000, config.clock))) { |
205 RTC_CHECK(RecreateEncoderInstance(config)); | 383 RTC_CHECK(RecreateEncoderInstance(config)); |
206 } | 384 } |
207 | 385 |
208 AudioEncoderOpus::AudioEncoderOpus(const CodecInst& codec_inst) | 386 AudioEncoderOpus::AudioEncoderOpus(const CodecInst& codec_inst) |
209 : AudioEncoderOpus(CreateConfig(codec_inst), nullptr) {} | 387 : AudioEncoderOpus(CreateConfig(codec_inst), nullptr) {} |
210 | 388 |
| 389 AudioEncoderOpus::AudioEncoderOpus(int payload_type, |
| 390 const SdpAudioFormat& format) |
| 391 : AudioEncoderOpus(CreateConfig(payload_type, format), nullptr) {} |
| 392 |
211 AudioEncoderOpus::~AudioEncoderOpus() { | 393 AudioEncoderOpus::~AudioEncoderOpus() { |
212 RTC_CHECK_EQ(0, WebRtcOpus_EncoderFree(inst_)); | 394 RTC_CHECK_EQ(0, WebRtcOpus_EncoderFree(inst_)); |
213 } | 395 } |
214 | 396 |
215 int AudioEncoderOpus::SampleRateHz() const { | 397 int AudioEncoderOpus::SampleRateHz() const { |
216 return kSampleRateHz; | 398 return kSampleRateHz; |
217 } | 399 } |
218 | 400 |
219 size_t AudioEncoderOpus::NumChannels() const { | 401 size_t AudioEncoderOpus::NumChannels() const { |
220 return config_.num_channels; | 402 return config_.num_channels; |
(...skipping 116 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
337 } else if (send_side_bwe_with_overhead_) { | 519 } else if (send_side_bwe_with_overhead_) { |
338 if (!overhead_bytes_per_packet_) { | 520 if (!overhead_bytes_per_packet_) { |
339 LOG(LS_INFO) | 521 LOG(LS_INFO) |
340 << "AudioEncoderOpus: Overhead unknown, target audio bitrate " | 522 << "AudioEncoderOpus: Overhead unknown, target audio bitrate " |
341 << target_audio_bitrate_bps << " bps is ignored."; | 523 << target_audio_bitrate_bps << " bps is ignored."; |
342 return; | 524 return; |
343 } | 525 } |
344 const int overhead_bps = static_cast<int>( | 526 const int overhead_bps = static_cast<int>( |
345 *overhead_bytes_per_packet_ * 8 * 100 / Num10MsFramesInNextPacket()); | 527 *overhead_bytes_per_packet_ * 8 * 100 / Num10MsFramesInNextPacket()); |
346 SetTargetBitrate(std::min( | 528 SetTargetBitrate(std::min( |
347 kMaxBitrateBps, | 529 kOpusMaxBitrateBps, |
348 std::max(kMinBitrateBps, target_audio_bitrate_bps - overhead_bps))); | 530 std::max(kOpusMinBitrateBps, target_audio_bitrate_bps - overhead_bps))); |
349 } else { | 531 } else { |
350 SetTargetBitrate(target_audio_bitrate_bps); | 532 SetTargetBitrate(target_audio_bitrate_bps); |
351 } | 533 } |
352 } | 534 } |
353 | 535 |
354 void AudioEncoderOpus::OnReceivedRtt(int rtt_ms) { | 536 void AudioEncoderOpus::OnReceivedRtt(int rtt_ms) { |
355 if (!audio_network_adaptor_) | 537 if (!audio_network_adaptor_) |
356 return; | 538 return; |
357 audio_network_adaptor_->SetRtt(rtt_ms); | 539 audio_network_adaptor_->SetRtt(rtt_ms); |
358 ApplyAudioNetworkAdaptor(); | 540 ApplyAudioNetworkAdaptor(); |
359 } | 541 } |
360 | 542 |
361 void AudioEncoderOpus::OnReceivedOverhead(size_t overhead_bytes_per_packet) { | 543 void AudioEncoderOpus::OnReceivedOverhead(size_t overhead_bytes_per_packet) { |
362 if (audio_network_adaptor_) { | 544 if (audio_network_adaptor_) { |
363 audio_network_adaptor_->SetOverhead(overhead_bytes_per_packet); | 545 audio_network_adaptor_->SetOverhead(overhead_bytes_per_packet); |
364 ApplyAudioNetworkAdaptor(); | 546 ApplyAudioNetworkAdaptor(); |
365 } else { | 547 } else { |
366 overhead_bytes_per_packet_ = | 548 overhead_bytes_per_packet_ = |
367 rtc::Optional<size_t>(overhead_bytes_per_packet); | 549 rtc::Optional<size_t>(overhead_bytes_per_packet); |
368 } | 550 } |
369 } | 551 } |
370 | 552 |
371 void AudioEncoderOpus::SetReceiverFrameLengthRange(int min_frame_length_ms, | 553 void AudioEncoderOpus::SetReceiverFrameLengthRange(int min_frame_length_ms, |
372 int max_frame_length_ms) { | 554 int max_frame_length_ms) { |
373 // Ensure that |SetReceiverFrameLengthRange| is called before | 555 // Ensure that |SetReceiverFrameLengthRange| is called before |
374 // |EnableAudioNetworkAdaptor|, otherwise we need to recreate | 556 // |EnableAudioNetworkAdaptor|, otherwise we need to recreate |
375 // |audio_network_adaptor_|, which is not a needed use case. | 557 // |audio_network_adaptor_|, which is not a needed use case. |
376 RTC_DCHECK(!audio_network_adaptor_); | 558 RTC_DCHECK(!audio_network_adaptor_); |
377 | 559 FindSupportedFrameLengths(min_frame_length_ms, max_frame_length_ms, |
378 config_.supported_frame_lengths_ms.clear(); | 560 &config_.supported_frame_lengths_ms); |
379 std::copy_if(std::begin(kSupportedFrameLengths), | |
380 std::end(kSupportedFrameLengths), | |
381 std::back_inserter(config_.supported_frame_lengths_ms), | |
382 [&](int frame_length_ms) { | |
383 return frame_length_ms >= min_frame_length_ms && | |
384 frame_length_ms <= max_frame_length_ms; | |
385 }); | |
386 RTC_DCHECK(std::is_sorted(config_.supported_frame_lengths_ms.begin(), | |
387 config_.supported_frame_lengths_ms.end())); | |
388 } | 561 } |
389 | 562 |
390 AudioEncoder::EncodedInfo AudioEncoderOpus::EncodeImpl( | 563 AudioEncoder::EncodedInfo AudioEncoderOpus::EncodeImpl( |
391 uint32_t rtp_timestamp, | 564 uint32_t rtp_timestamp, |
392 rtc::ArrayView<const int16_t> audio, | 565 rtc::ArrayView<const int16_t> audio, |
393 rtc::Buffer* encoded) { | 566 rtc::Buffer* encoded) { |
394 MaybeUpdateUplinkBandwidth(); | 567 MaybeUpdateUplinkBandwidth(); |
395 | 568 |
396 if (input_buffer_.empty()) | 569 if (input_buffer_.empty()) |
397 first_timestamp_in_buffer_ = rtp_timestamp; | 570 first_timestamp_in_buffer_ = rtp_timestamp; |
(...skipping 111 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
509 float opt_loss_rate = OptimizePacketLossRate(fraction, packet_loss_rate_); | 682 float opt_loss_rate = OptimizePacketLossRate(fraction, packet_loss_rate_); |
510 if (packet_loss_rate_ != opt_loss_rate) { | 683 if (packet_loss_rate_ != opt_loss_rate) { |
511 packet_loss_rate_ = opt_loss_rate; | 684 packet_loss_rate_ = opt_loss_rate; |
512 RTC_CHECK_EQ( | 685 RTC_CHECK_EQ( |
513 0, WebRtcOpus_SetPacketLossRate( | 686 0, WebRtcOpus_SetPacketLossRate( |
514 inst_, static_cast<int32_t>(packet_loss_rate_ * 100 + .5))); | 687 inst_, static_cast<int32_t>(packet_loss_rate_ * 100 + .5))); |
515 } | 688 } |
516 } | 689 } |
517 | 690 |
518 void AudioEncoderOpus::SetTargetBitrate(int bits_per_second) { | 691 void AudioEncoderOpus::SetTargetBitrate(int bits_per_second) { |
519 config_.bitrate_bps = rtc::Optional<int>( | 692 config_.bitrate_bps = rtc::Optional<int>(std::max( |
520 std::max(std::min(bits_per_second, kMaxBitrateBps), kMinBitrateBps)); | 693 std::min(bits_per_second, kOpusMaxBitrateBps), kOpusMinBitrateBps)); |
521 RTC_DCHECK(config_.IsOk()); | 694 RTC_DCHECK(config_.IsOk()); |
522 RTC_CHECK_EQ(0, WebRtcOpus_SetBitRate(inst_, config_.GetBitrateBps())); | 695 RTC_CHECK_EQ(0, WebRtcOpus_SetBitRate(inst_, config_.GetBitrateBps())); |
523 const auto new_complexity = config_.GetNewComplexity(); | 696 const auto new_complexity = config_.GetNewComplexity(); |
524 if (new_complexity && complexity_ != *new_complexity) { | 697 if (new_complexity && complexity_ != *new_complexity) { |
525 complexity_ = *new_complexity; | 698 complexity_ = *new_complexity; |
526 RTC_CHECK_EQ(0, WebRtcOpus_SetComplexity(inst_, complexity_)); | 699 RTC_CHECK_EQ(0, WebRtcOpus_SetComplexity(inst_, complexity_)); |
527 } | 700 } |
528 } | 701 } |
529 | 702 |
530 void AudioEncoderOpus::ApplyAudioNetworkAdaptor() { | 703 void AudioEncoderOpus::ApplyAudioNetworkAdaptor() { |
(...skipping 20 matching lines...) Expand all Loading... |
551 const std::string& config_string, | 724 const std::string& config_string, |
552 RtcEventLog* event_log, | 725 RtcEventLog* event_log, |
553 const Clock* clock) const { | 726 const Clock* clock) const { |
554 AudioNetworkAdaptorImpl::Config config; | 727 AudioNetworkAdaptorImpl::Config config; |
555 config.clock = clock; | 728 config.clock = clock; |
556 config.event_log = event_log; | 729 config.event_log = event_log; |
557 return std::unique_ptr<AudioNetworkAdaptor>(new AudioNetworkAdaptorImpl( | 730 return std::unique_ptr<AudioNetworkAdaptor>(new AudioNetworkAdaptorImpl( |
558 config, | 731 config, |
559 ControllerManagerImpl::Create( | 732 ControllerManagerImpl::Create( |
560 config_string, NumChannels(), supported_frame_lengths_ms(), | 733 config_string, NumChannels(), supported_frame_lengths_ms(), |
561 kMinBitrateBps, num_channels_to_encode_, next_frame_length_ms_, | 734 kOpusMinBitrateBps, num_channels_to_encode_, next_frame_length_ms_, |
562 GetTargetBitrate(), config_.fec_enabled, GetDtx(), clock))); | 735 GetTargetBitrate(), config_.fec_enabled, GetDtx(), clock))); |
563 } | 736 } |
564 | 737 |
565 void AudioEncoderOpus::MaybeUpdateUplinkBandwidth() { | 738 void AudioEncoderOpus::MaybeUpdateUplinkBandwidth() { |
566 if (audio_network_adaptor_) { | 739 if (audio_network_adaptor_) { |
567 int64_t now_ms = rtc::TimeMillis(); | 740 int64_t now_ms = rtc::TimeMillis(); |
568 if (!bitrate_smoother_last_update_time_ || | 741 if (!bitrate_smoother_last_update_time_ || |
569 now_ms - *bitrate_smoother_last_update_time_ >= | 742 now_ms - *bitrate_smoother_last_update_time_ >= |
570 config_.uplink_bandwidth_update_interval_ms) { | 743 config_.uplink_bandwidth_update_interval_ms) { |
571 rtc::Optional<float> smoothed_bitrate = bitrate_smoother_->GetAverage(); | 744 rtc::Optional<float> smoothed_bitrate = bitrate_smoother_->GetAverage(); |
572 if (smoothed_bitrate) | 745 if (smoothed_bitrate) |
573 audio_network_adaptor_->SetUplinkBandwidth(*smoothed_bitrate); | 746 audio_network_adaptor_->SetUplinkBandwidth(*smoothed_bitrate); |
574 bitrate_smoother_last_update_time_ = rtc::Optional<int64_t>(now_ms); | 747 bitrate_smoother_last_update_time_ = rtc::Optional<int64_t>(now_ms); |
575 } | 748 } |
576 } | 749 } |
577 } | 750 } |
578 | 751 |
579 } // namespace webrtc | 752 } // namespace webrtc |
OLD | NEW |