henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 1 | /* |
kjellander | 1afca73 | 2016-02-07 20:46:45 -0800 | [diff] [blame] | 2 | * Copyright (c) 2004 The WebRTC project authors. All Rights Reserved. |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 3 | * |
kjellander | 1afca73 | 2016-02-07 20:46:45 -0800 | [diff] [blame] | 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. |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 9 | */ |
| 10 | |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 11 | #ifndef MEDIA_ENGINE_WEBRTC_VOICE_ENGINE_H_ |
| 12 | #define MEDIA_ENGINE_WEBRTC_VOICE_ENGINE_H_ |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 13 | |
| 14 | #include <map> |
kwiberg | 686a8ef | 2016-02-26 03:00:35 -0800 | [diff] [blame] | 15 | #include <memory> |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 16 | #include <string> |
| 17 | #include <vector> |
| 18 | |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 19 | #include "api/audio_codecs/audio_encoder_factory.h" |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 20 | #include "api/rtp_receiver_interface.h" |
Mirko Bonadei | d970807 | 2019-01-25 20:26:48 +0100 | [diff] [blame] | 21 | #include "api/scoped_refptr.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 22 | #include "call/audio_state.h" |
| 23 | #include "call/call.h" |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 24 | #include "media/base/rtp_utils.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 25 | #include "media/engine/apm_helpers.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 26 | #include "modules/audio_processing/include/audio_processing.h" |
| 27 | #include "pc/channel.h" |
| 28 | #include "rtc_base/buffer.h" |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 29 | #include "rtc_base/constructor_magic.h" |
Sebastian Jansson | 470a5ea | 2019-01-23 12:37:49 +0100 | [diff] [blame] | 30 | #include "rtc_base/experiments/audio_allocation_settings.h" |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 31 | #include "rtc_base/network_route.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 32 | #include "rtc_base/task_queue.h" |
| 33 | #include "rtc_base/thread_checker.h" |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 34 | |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 35 | namespace cricket { |
| 36 | |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 37 | class AudioDeviceModule; |
gyzhou | 95aa964 | 2016-12-13 14:06:26 -0800 | [diff] [blame] | 38 | class AudioMixer; |
Taylor Brandstetter | 1a018dc | 2016-03-08 12:37:39 -0800 | [diff] [blame] | 39 | class AudioSource; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 40 | class WebRtcVoiceMediaChannel; |
| 41 | |
| 42 | // WebRtcVoiceEngine is a class to be used with CompositeMediaEngine. |
| 43 | // It uses the WebRtc VoiceEngine library for audio handling. |
Sebastian Jansson | 84848f2 | 2018-11-16 10:40:36 +0100 | [diff] [blame] | 44 | class WebRtcVoiceEngine final : public VoiceEngineInterface { |
Jelena Marusic | c28a896 | 2015-05-29 15:05:44 +0200 | [diff] [blame] | 45 | friend class WebRtcVoiceMediaChannel; |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 46 | |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 47 | public: |
ossu | 29b1a8d | 2016-06-13 07:34:51 -0700 | [diff] [blame] | 48 | WebRtcVoiceEngine( |
| 49 | webrtc::AudioDeviceModule* adm, |
ossu | eb1fde4 | 2017-05-02 06:46:30 -0700 | [diff] [blame] | 50 | const rtc::scoped_refptr<webrtc::AudioEncoderFactory>& encoder_factory, |
gyzhou | 95aa964 | 2016-12-13 14:06:26 -0800 | [diff] [blame] | 51 | const rtc::scoped_refptr<webrtc::AudioDecoderFactory>& decoder_factory, |
peah | a9cc40b | 2017-06-29 08:32:09 -0700 | [diff] [blame] | 52 | rtc::scoped_refptr<webrtc::AudioMixer> audio_mixer, |
| 53 | rtc::scoped_refptr<webrtc::AudioProcessing> audio_processing); |
Sebastian Jansson | 84848f2 | 2018-11-16 10:40:36 +0100 | [diff] [blame] | 54 | ~WebRtcVoiceEngine() override; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 55 | |
deadbeef | eb02c03 | 2017-06-15 08:29:25 -0700 | [diff] [blame] | 56 | // Does initialization that needs to occur on the worker thread. |
Sebastian Jansson | 84848f2 | 2018-11-16 10:40:36 +0100 | [diff] [blame] | 57 | void Init() override; |
deadbeef | eb02c03 | 2017-06-15 08:29:25 -0700 | [diff] [blame] | 58 | |
Sebastian Jansson | 84848f2 | 2018-11-16 10:40:36 +0100 | [diff] [blame] | 59 | rtc::scoped_refptr<webrtc::AudioState> GetAudioState() const override; |
| 60 | VoiceMediaChannel* CreateMediaChannel( |
| 61 | webrtc::Call* call, |
| 62 | const MediaConfig& config, |
| 63 | const AudioOptions& options, |
| 64 | const webrtc::CryptoOptions& crypto_options) override; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 65 | |
Sebastian Jansson | 84848f2 | 2018-11-16 10:40:36 +0100 | [diff] [blame] | 66 | const std::vector<AudioCodec>& send_codecs() const override; |
| 67 | const std::vector<AudioCodec>& recv_codecs() const override; |
| 68 | RtpCapabilities GetCapabilities() const override; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 69 | |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 70 | // For tracking WebRtc channels. Needed because we have to pause them |
| 71 | // all when switching devices. |
| 72 | // May only be called by WebRtcVoiceMediaChannel. |
solenberg | 63b3454 | 2015-09-29 06:06:31 -0700 | [diff] [blame] | 73 | void RegisterChannel(WebRtcVoiceMediaChannel* channel); |
| 74 | void UnregisterChannel(WebRtcVoiceMediaChannel* channel); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 75 | |
ivoc | d66b44d | 2016-01-15 03:06:36 -0800 | [diff] [blame] | 76 | // Starts AEC dump using an existing file. A maximum file size in bytes can be |
| 77 | // specified. When the maximum file size is reached, logging is stopped and |
| 78 | // the file is closed. If max_size_bytes is set to <= 0, no limit will be |
| 79 | // used. |
Sebastian Jansson | 84848f2 | 2018-11-16 10:40:36 +0100 | [diff] [blame] | 80 | bool StartAecDump(rtc::PlatformFile file, int64_t max_size_bytes) override; |
wu@webrtc.org | a989080 | 2013-12-13 00:21:03 +0000 | [diff] [blame] | 81 | |
ivoc | 797ef12 | 2015-10-22 03:25:41 -0700 | [diff] [blame] | 82 | // Stops AEC dump. |
Sebastian Jansson | 84848f2 | 2018-11-16 10:40:36 +0100 | [diff] [blame] | 83 | void StopAecDump() override; |
ivoc | 797ef12 | 2015-10-22 03:25:41 -0700 | [diff] [blame] | 84 | |
peah | b1c9d1d | 2017-07-25 15:45:24 -0700 | [diff] [blame] | 85 | const webrtc::AudioProcessing::Config GetApmConfigForTest() const { |
| 86 | return apm()->GetConfig(); |
peah | 8271d04 | 2016-11-22 07:24:52 -0800 | [diff] [blame] | 87 | } |
| 88 | |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 89 | private: |
solenberg | 63b3454 | 2015-09-29 06:06:31 -0700 | [diff] [blame] | 90 | // Every option that is "set" will be applied. Every option not "set" will be |
| 91 | // ignored. This allows us to selectively turn on and off different options |
| 92 | // easily at any time. |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 93 | bool ApplyOptions(const AudioOptions& options); |
xians@webrtc.org | 3cefbc9 | 2014-10-10 09:42:53 +0000 | [diff] [blame] | 94 | |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 95 | void StartAecDump(const std::string& filename); |
solenberg | 0a617e2 | 2015-10-20 15:49:38 -0700 | [diff] [blame] | 96 | int CreateVoEChannel(); |
aleloi | 048cbdd | 2017-05-29 02:56:27 -0700 | [diff] [blame] | 97 | |
deadbeef | eb02c03 | 2017-06-15 08:29:25 -0700 | [diff] [blame] | 98 | std::unique_ptr<rtc::TaskQueue> low_priority_worker_queue_; |
aleloi | 048cbdd | 2017-05-29 02:56:27 -0700 | [diff] [blame] | 99 | |
solenberg | 5b5129a | 2016-04-08 05:35:48 -0700 | [diff] [blame] | 100 | webrtc::AudioDeviceModule* adm(); |
peah | b1c9d1d | 2017-07-25 15:45:24 -0700 | [diff] [blame] | 101 | webrtc::AudioProcessing* apm() const; |
Fredrik Solenberg | 2a87797 | 2017-12-15 16:42:15 +0100 | [diff] [blame] | 102 | webrtc::AudioState* audio_state(); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 103 | |
ossu | 20a4b3f | 2017-04-27 02:08:52 -0700 | [diff] [blame] | 104 | AudioCodecs CollectCodecs( |
| 105 | const std::vector<webrtc::AudioCodecSpec>& specs) const; |
ossu | c54071d | 2016-08-17 02:45:41 -0700 | [diff] [blame] | 106 | |
solenberg | 566ef24 | 2015-11-06 15:34:49 -0800 | [diff] [blame] | 107 | rtc::ThreadChecker signal_thread_checker_; |
| 108 | rtc::ThreadChecker worker_thread_checker_; |
| 109 | |
Sebastian Jansson | 470a5ea | 2019-01-23 12:37:49 +0100 | [diff] [blame] | 110 | const webrtc::AudioAllocationSettings allocation_settings_; |
| 111 | |
Fredrik Solenberg | 2a87797 | 2017-12-15 16:42:15 +0100 | [diff] [blame] | 112 | // The audio device module. |
solenberg | ff97631 | 2016-03-30 23:28:51 -0700 | [diff] [blame] | 113 | rtc::scoped_refptr<webrtc::AudioDeviceModule> adm_; |
ossu | 20a4b3f | 2017-04-27 02:08:52 -0700 | [diff] [blame] | 114 | rtc::scoped_refptr<webrtc::AudioEncoderFactory> encoder_factory_; |
ossu | 29b1a8d | 2016-06-13 07:34:51 -0700 | [diff] [blame] | 115 | rtc::scoped_refptr<webrtc::AudioDecoderFactory> decoder_factory_; |
deadbeef | eb02c03 | 2017-06-15 08:29:25 -0700 | [diff] [blame] | 116 | rtc::scoped_refptr<webrtc::AudioMixer> audio_mixer_; |
Fredrik Solenberg | 2a87797 | 2017-12-15 16:42:15 +0100 | [diff] [blame] | 117 | // The audio processing module. |
peah | a9cc40b | 2017-06-29 08:32:09 -0700 | [diff] [blame] | 118 | rtc::scoped_refptr<webrtc::AudioProcessing> apm_; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 119 | // The primary instance of WebRtc VoiceEngine. |
solenberg | 566ef24 | 2015-11-06 15:34:49 -0800 | [diff] [blame] | 120 | rtc::scoped_refptr<webrtc::AudioState> audio_state_; |
ossu | c54071d | 2016-08-17 02:45:41 -0700 | [diff] [blame] | 121 | std::vector<AudioCodec> send_codecs_; |
| 122 | std::vector<AudioCodec> recv_codecs_; |
solenberg | 63b3454 | 2015-09-29 06:06:31 -0700 | [diff] [blame] | 123 | std::vector<WebRtcVoiceMediaChannel*> channels_; |
solenberg | 246b817 | 2015-12-08 09:50:23 -0800 | [diff] [blame] | 124 | bool is_dumping_aec_ = false; |
deadbeef | eb02c03 | 2017-06-15 08:29:25 -0700 | [diff] [blame] | 125 | bool initialized_ = false; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 126 | |
solenberg | 246b817 | 2015-12-08 09:50:23 -0800 | [diff] [blame] | 127 | webrtc::AgcConfig default_agc_config_; |
Alessio Bazzica | cc22f51 | 2018-08-30 13:01:34 +0200 | [diff] [blame] | 128 | // Cache received extended_filter_aec, delay_agnostic_aec and experimental_ns |
| 129 | // values, and apply them in case they are missing in the audio options. |
| 130 | // We need to do this because SetExtraOptions() will revert to defaults for |
| 131 | // options which are not provided. |
Danil Chapovalov | 00c7183 | 2018-06-15 15:58:38 +0200 | [diff] [blame] | 132 | absl::optional<bool> extended_filter_aec_; |
| 133 | absl::optional<bool> delay_agnostic_aec_; |
| 134 | absl::optional<bool> experimental_ns_; |
Fredrik Solenberg | 8f5787a | 2018-01-11 13:52:30 +0100 | [diff] [blame] | 135 | // Jitter buffer settings for new streams. |
| 136 | size_t audio_jitter_buffer_max_packets_ = 50; |
| 137 | bool audio_jitter_buffer_fast_accelerate_ = false; |
Jakob Ivarsson | 10403ae | 2018-11-27 15:45:20 +0100 | [diff] [blame] | 138 | int audio_jitter_buffer_min_delay_ms_ = 0; |
Jakob Ivarsson | 53eae87 | 2019-01-10 15:58:36 +0100 | [diff] [blame] | 139 | bool audio_jitter_buffer_enable_rtx_handling_ = false; |
solenberg | c96df77 | 2015-10-21 13:01:53 -0700 | [diff] [blame] | 140 | |
solenberg | ff97631 | 2016-03-30 23:28:51 -0700 | [diff] [blame] | 141 | RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(WebRtcVoiceEngine); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 142 | }; |
| 143 | |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 144 | // WebRtcVoiceMediaChannel is an implementation of VoiceMediaChannel that uses |
| 145 | // WebRtc Voice Engine. |
solenberg | 566ef24 | 2015-11-06 15:34:49 -0800 | [diff] [blame] | 146 | class WebRtcVoiceMediaChannel final : public VoiceMediaChannel, |
| 147 | public webrtc::Transport { |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 148 | public: |
Fredrik Solenberg | b071a19 | 2015-09-17 16:42:56 +0200 | [diff] [blame] | 149 | WebRtcVoiceMediaChannel(WebRtcVoiceEngine* engine, |
nisse | 51542be | 2016-02-12 02:27:06 -0800 | [diff] [blame] | 150 | const MediaConfig& config, |
Fredrik Solenberg | b071a19 | 2015-09-17 16:42:56 +0200 | [diff] [blame] | 151 | const AudioOptions& options, |
Benjamin Wright | bfb444c | 2018-10-15 10:20:24 -0700 | [diff] [blame] | 152 | const webrtc::CryptoOptions& crypto_options, |
Fredrik Solenberg | b071a19 | 2015-09-17 16:42:56 +0200 | [diff] [blame] | 153 | webrtc::Call* call); |
Fredrik Solenberg | aaf8ff2 | 2015-05-07 16:05:53 +0200 | [diff] [blame] | 154 | ~WebRtcVoiceMediaChannel() override; |
Fredrik Solenberg | e444a3d | 2015-05-07 16:42:08 +0200 | [diff] [blame] | 155 | |
solenberg | 66f4339 | 2015-09-09 01:36:22 -0700 | [diff] [blame] | 156 | const AudioOptions& options() const { return options_; } |
Fredrik Solenberg | e444a3d | 2015-05-07 16:42:08 +0200 | [diff] [blame] | 157 | |
Peter Thatcher | c2ee2c8 | 2015-08-07 16:05:34 -0700 | [diff] [blame] | 158 | bool SetSendParameters(const AudioSendParameters& params) override; |
| 159 | bool SetRecvParameters(const AudioRecvParameters& params) override; |
Taylor Brandstetter | db0cd9e | 2016-05-16 11:40:30 -0700 | [diff] [blame] | 160 | webrtc::RtpParameters GetRtpSendParameters(uint32_t ssrc) const override; |
Zach Stein | ba37b4b | 2018-01-23 15:02:36 -0800 | [diff] [blame] | 161 | webrtc::RTCError SetRtpSendParameters( |
| 162 | uint32_t ssrc, |
| 163 | const webrtc::RtpParameters& parameters) override; |
Taylor Brandstetter | db0cd9e | 2016-05-16 11:40:30 -0700 | [diff] [blame] | 164 | webrtc::RtpParameters GetRtpReceiveParameters(uint32_t ssrc) const override; |
| 165 | bool SetRtpReceiveParameters( |
| 166 | uint32_t ssrc, |
| 167 | const webrtc::RtpParameters& parameters) override; |
skvlad | e0d4637 | 2016-04-07 22:59:22 -0700 | [diff] [blame] | 168 | |
aleloi | 84ef615 | 2016-08-04 05:28:21 -0700 | [diff] [blame] | 169 | void SetPlayout(bool playout) override; |
Taylor Brandstetter | 1a018dc | 2016-03-08 12:37:39 -0800 | [diff] [blame] | 170 | void SetSend(bool send) override; |
Peter Boström | 0c4e06b | 2015-10-07 12:23:21 +0200 | [diff] [blame] | 171 | bool SetAudioSend(uint32_t ssrc, |
| 172 | bool enable, |
| 173 | const AudioOptions* options, |
Taylor Brandstetter | 1a018dc | 2016-03-08 12:37:39 -0800 | [diff] [blame] | 174 | AudioSource* source) override; |
Fredrik Solenberg | aaf8ff2 | 2015-05-07 16:05:53 +0200 | [diff] [blame] | 175 | bool AddSendStream(const StreamParams& sp) override; |
Peter Boström | 0c4e06b | 2015-10-07 12:23:21 +0200 | [diff] [blame] | 176 | bool RemoveSendStream(uint32_t ssrc) override; |
Fredrik Solenberg | aaf8ff2 | 2015-05-07 16:05:53 +0200 | [diff] [blame] | 177 | bool AddRecvStream(const StreamParams& sp) override; |
Peter Boström | 0c4e06b | 2015-10-07 12:23:21 +0200 | [diff] [blame] | 178 | bool RemoveRecvStream(uint32_t ssrc) override; |
Benjamin Wright | 84583f6 | 2018-10-04 14:22:34 -0700 | [diff] [blame] | 179 | |
| 180 | // E2EE Frame API |
| 181 | // Set a frame decryptor to a particular ssrc that will intercept all |
| 182 | // incoming audio payloads and attempt to decrypt them before forwarding the |
| 183 | // result. |
| 184 | void SetFrameDecryptor(uint32_t ssrc, |
| 185 | rtc::scoped_refptr<webrtc::FrameDecryptorInterface> |
| 186 | frame_decryptor) override; |
| 187 | // Set a frame encryptor to a particular ssrc that will intercept all |
| 188 | // outgoing audio payloads frames and attempt to encrypt them and forward the |
| 189 | // result to the packetizer. |
| 190 | void SetFrameEncryptor(uint32_t ssrc, |
| 191 | rtc::scoped_refptr<webrtc::FrameEncryptorInterface> |
| 192 | frame_encryptor) override; |
| 193 | |
solenberg | 2100c0b | 2017-03-01 11:29:29 -0800 | [diff] [blame] | 194 | // SSRC=0 will apply the new volume to current and future unsignaled streams. |
solenberg | 4bac9c5 | 2015-10-09 02:32:53 -0700 | [diff] [blame] | 195 | bool SetOutputVolume(uint32_t ssrc, double volume) override; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 196 | |
Ruslan Burakov | 7ea4605 | 2019-02-16 02:07:05 +0100 | [diff] [blame] | 197 | bool SetBaseMinimumPlayoutDelayMs(uint32_t ssrc, int delay_ms) override; |
| 198 | absl::optional<int> GetBaseMinimumPlayoutDelayMs( |
| 199 | uint32_t ssrc) const override; |
| 200 | |
Fredrik Solenberg | aaf8ff2 | 2015-05-07 16:05:53 +0200 | [diff] [blame] | 201 | bool CanInsertDtmf() override; |
solenberg | 1d63dd0 | 2015-12-02 12:35:09 -0800 | [diff] [blame] | 202 | bool InsertDtmf(uint32_t ssrc, int event, int duration) override; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 203 | |
jbauch | eec21bd | 2016-03-20 06:15:43 -0700 | [diff] [blame] | 204 | void OnPacketReceived(rtc::CopyOnWriteBuffer* packet, |
Niels Möller | e693381 | 2018-11-05 13:01:41 +0100 | [diff] [blame] | 205 | int64_t packet_time_us) override; |
jbauch | eec21bd | 2016-03-20 06:15:43 -0700 | [diff] [blame] | 206 | void OnRtcpReceived(rtc::CopyOnWriteBuffer* packet, |
Niels Möller | e693381 | 2018-11-05 13:01:41 +0100 | [diff] [blame] | 207 | int64_t packet_time_us) override; |
Honghai Zhang | cc411c0 | 2016-03-29 17:27:21 -0700 | [diff] [blame] | 208 | void OnNetworkRouteChanged(const std::string& transport_name, |
Honghai Zhang | 0e533ef | 2016-04-19 15:41:36 -0700 | [diff] [blame] | 209 | const rtc::NetworkRoute& network_route) override; |
skvlad | 7a43d25 | 2016-03-22 15:32:27 -0700 | [diff] [blame] | 210 | void OnReadyToSend(bool ready) override; |
Fredrik Solenberg | aaf8ff2 | 2015-05-07 16:05:53 +0200 | [diff] [blame] | 211 | bool GetStats(VoiceMediaInfo* info) override; |
Fredrik Solenberg | e444a3d | 2015-05-07 16:42:08 +0200 | [diff] [blame] | 212 | |
solenberg | 2100c0b | 2017-03-01 11:29:29 -0800 | [diff] [blame] | 213 | // SSRC=0 will set the audio sink on the latest unsignaled stream, future or |
| 214 | // current. Only one stream at a time will use the sink. |
Tommi | f888bb5 | 2015-12-12 01:37:01 +0100 | [diff] [blame] | 215 | void SetRawAudioSink( |
| 216 | uint32_t ssrc, |
kwiberg | 686a8ef | 2016-02-26 03:00:35 -0800 | [diff] [blame] | 217 | std::unique_ptr<webrtc::AudioSinkInterface> sink) override; |
Tommi | f888bb5 | 2015-12-12 01:37:01 +0100 | [diff] [blame] | 218 | |
zhihuang | 38ede13 | 2017-06-15 12:52:32 -0700 | [diff] [blame] | 219 | std::vector<webrtc::RtpSource> GetSources(uint32_t ssrc) const override; |
hbos | 8d609f6 | 2017-04-10 07:39:05 -0700 | [diff] [blame] | 220 | |
Fredrik Solenberg | e444a3d | 2015-05-07 16:42:08 +0200 | [diff] [blame] | 221 | // implements Transport interface |
stefan | 1d8a506 | 2015-10-02 03:39:33 -0700 | [diff] [blame] | 222 | bool SendRtp(const uint8_t* data, |
| 223 | size_t len, |
| 224 | const webrtc::PacketOptions& options) override { |
jbauch | eec21bd | 2016-03-20 06:15:43 -0700 | [diff] [blame] | 225 | rtc::CopyOnWriteBuffer packet(data, len, kMaxRtpPacketLen); |
stefan | c1aeaf0 | 2015-10-15 07:26:07 -0700 | [diff] [blame] | 226 | rtc::PacketOptions rtc_options; |
| 227 | rtc_options.packet_id = options.packet_id; |
Tim Haloun | 6ca9836 | 2018-09-17 17:06:08 -0700 | [diff] [blame] | 228 | if (DscpEnabled()) { |
| 229 | rtc_options.dscp = PreferredDscp(); |
| 230 | } |
Sebastian Jansson | 0378997 | 2018-10-09 18:27:57 +0200 | [diff] [blame] | 231 | rtc_options.info_signaled_after_sent.included_in_feedback = |
| 232 | options.included_in_feedback; |
| 233 | rtc_options.info_signaled_after_sent.included_in_allocation = |
| 234 | options.included_in_allocation; |
stefan | c1aeaf0 | 2015-10-15 07:26:07 -0700 | [diff] [blame] | 235 | return VoiceMediaChannel::SendPacket(&packet, rtc_options); |
Fredrik Solenberg | e444a3d | 2015-05-07 16:42:08 +0200 | [diff] [blame] | 236 | } |
| 237 | |
pbos | 2d56668 | 2015-09-28 09:59:31 -0700 | [diff] [blame] | 238 | bool SendRtcp(const uint8_t* data, size_t len) override { |
jbauch | eec21bd | 2016-03-20 06:15:43 -0700 | [diff] [blame] | 239 | rtc::CopyOnWriteBuffer packet(data, len, kMaxRtpPacketLen); |
Qingsi Wang | 6e641e6 | 2018-04-11 20:14:17 -0700 | [diff] [blame] | 240 | rtc::PacketOptions rtc_options; |
Tim Haloun | 6ca9836 | 2018-09-17 17:06:08 -0700 | [diff] [blame] | 241 | if (DscpEnabled()) { |
| 242 | rtc_options.dscp = PreferredDscp(); |
| 243 | } |
Tim Haloun | 648d28a | 2018-10-18 16:52:22 -0700 | [diff] [blame] | 244 | |
Qingsi Wang | 6e641e6 | 2018-04-11 20:14:17 -0700 | [diff] [blame] | 245 | return VoiceMediaChannel::SendRtcp(&packet, rtc_options); |
Fredrik Solenberg | e444a3d | 2015-05-07 16:42:08 +0200 | [diff] [blame] | 246 | } |
| 247 | |
Fredrik Solenberg | e444a3d | 2015-05-07 16:42:08 +0200 | [diff] [blame] | 248 | private: |
Fredrik Solenberg | b071a19 | 2015-09-17 16:42:56 +0200 | [diff] [blame] | 249 | bool SetOptions(const AudioOptions& options); |
Fredrik Solenberg | b071a19 | 2015-09-17 16:42:56 +0200 | [diff] [blame] | 250 | bool SetRecvCodecs(const std::vector<AudioCodec>& codecs); |
solenberg | 72e29d2 | 2016-03-08 06:35:16 -0800 | [diff] [blame] | 251 | bool SetSendCodecs(const std::vector<AudioCodec>& codecs); |
Taylor Brandstetter | 1a018dc | 2016-03-08 12:37:39 -0800 | [diff] [blame] | 252 | bool SetLocalSource(uint32_t ssrc, AudioSource* source); |
Peter Boström | 0c4e06b | 2015-10-07 12:23:21 +0200 | [diff] [blame] | 253 | bool MuteStream(uint32_t ssrc, bool mute); |
Fredrik Solenberg | b071a19 | 2015-09-17 16:42:56 +0200 | [diff] [blame] | 254 | |
Fredrik Solenberg | e444a3d | 2015-05-07 16:42:08 +0200 | [diff] [blame] | 255 | WebRtcVoiceEngine* engine() { return engine_; } |
kwiberg | 37b8b11 | 2016-11-03 02:46:53 -0700 | [diff] [blame] | 256 | void ChangePlayout(bool playout); |
solenberg | 0a617e2 | 2015-10-20 15:49:38 -0700 | [diff] [blame] | 257 | int CreateVoEChannel(); |
solenberg | 7add058 | 2015-11-20 09:59:34 -0800 | [diff] [blame] | 258 | bool DeleteVoEChannel(int channel); |
deadbeef | 8034614 | 2016-04-27 14:17:10 -0700 | [diff] [blame] | 259 | bool SetMaxSendBitrate(int bps); |
solenberg | d53a3f9 | 2016-04-14 13:56:37 -0700 | [diff] [blame] | 260 | void SetupRecording(); |
solenberg | 2100c0b | 2017-03-01 11:29:29 -0800 | [diff] [blame] | 261 | // Check if 'ssrc' is an unsignaled stream, and if so mark it as not being |
| 262 | // unsignaled anymore (i.e. it is now removed, or signaled), and return true. |
| 263 | bool MaybeDeregisterUnsignaledRecvStream(uint32_t ssrc); |
Fredrik Solenberg | 4b60c73 | 2015-05-07 14:07:48 +0200 | [diff] [blame] | 264 | |
solenberg | 566ef24 | 2015-11-06 15:34:49 -0800 | [diff] [blame] | 265 | rtc::ThreadChecker worker_thread_checker_; |
Fredrik Solenberg | 4b60c73 | 2015-05-07 14:07:48 +0200 | [diff] [blame] | 266 | |
solenberg | 566ef24 | 2015-11-06 15:34:49 -0800 | [diff] [blame] | 267 | WebRtcVoiceEngine* const engine_ = nullptr; |
Taylor Brandstetter | 0cd086b | 2016-04-20 16:23:10 -0700 | [diff] [blame] | 268 | std::vector<AudioCodec> send_codecs_; |
kwiberg | 1c07c70 | 2017-03-27 07:15:49 -0700 | [diff] [blame] | 269 | |
| 270 | // TODO(kwiberg): decoder_map_ and recv_codecs_ store the exact same |
| 271 | // information, in slightly different formats. Eliminate recv_codecs_. |
| 272 | std::map<int, webrtc::SdpAudioFormat> decoder_map_; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 273 | std::vector<AudioCodec> recv_codecs_; |
kwiberg | 1c07c70 | 2017-03-27 07:15:49 -0700 | [diff] [blame] | 274 | |
deadbeef | 8034614 | 2016-04-27 14:17:10 -0700 | [diff] [blame] | 275 | int max_send_bitrate_bps_ = 0; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 276 | AudioOptions options_; |
Danil Chapovalov | 00c7183 | 2018-06-15 15:58:38 +0200 | [diff] [blame] | 277 | absl::optional<int> dtmf_payload_type_; |
solenberg | ffbbcac | 2016-11-17 05:25:37 -0800 | [diff] [blame] | 278 | int dtmf_payload_freq_ = -1; |
solenberg | 72e29d2 | 2016-03-08 06:35:16 -0800 | [diff] [blame] | 279 | bool recv_transport_cc_enabled_ = false; |
solenberg | 8189b02 | 2016-06-14 12:13:00 -0700 | [diff] [blame] | 280 | bool recv_nack_enabled_ = false; |
solenberg | ffbbcac | 2016-11-17 05:25:37 -0800 | [diff] [blame] | 281 | bool desired_playout_ = false; |
solenberg | 566ef24 | 2015-11-06 15:34:49 -0800 | [diff] [blame] | 282 | bool playout_ = false; |
Taylor Brandstetter | 1a018dc | 2016-03-08 12:37:39 -0800 | [diff] [blame] | 283 | bool send_ = false; |
solenberg | 566ef24 | 2015-11-06 15:34:49 -0800 | [diff] [blame] | 284 | webrtc::Call* const call_ = nullptr; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 285 | |
Jiawei Ou | 5571812 | 2018-11-09 13:17:39 -0800 | [diff] [blame] | 286 | const MediaConfig::Audio audio_config_; |
| 287 | |
solenberg | 2100c0b | 2017-03-01 11:29:29 -0800 | [diff] [blame] | 288 | // Queue of unsignaled SSRCs; oldest at the beginning. |
| 289 | std::vector<uint32_t> unsignaled_recv_ssrcs_; |
| 290 | |
Seth Hampson | 5897a6e | 2018-04-03 11:16:33 -0700 | [diff] [blame] | 291 | // This is a stream param that comes from the remote description, but wasn't |
| 292 | // signaled with any a=ssrc lines. It holds the information that was signaled |
| 293 | // before the unsignaled receive stream is created when the first packet is |
| 294 | // received. |
| 295 | StreamParams unsignaled_stream_params_; |
| 296 | |
solenberg | 2100c0b | 2017-03-01 11:29:29 -0800 | [diff] [blame] | 297 | // Volume for unsignaled streams, which may be set before the stream exists. |
solenberg | 1ac5614 | 2015-10-13 03:58:19 -0700 | [diff] [blame] | 298 | double default_recv_volume_ = 1.0; |
Ruslan Burakov | 7ea4605 | 2019-02-16 02:07:05 +0100 | [diff] [blame] | 299 | |
| 300 | // Delay for unsignaled streams, which may be set before the stream exists. |
| 301 | int default_recv_base_minimum_delay_ms_ = 0; |
| 302 | |
solenberg | 2100c0b | 2017-03-01 11:29:29 -0800 | [diff] [blame] | 303 | // Sink for latest unsignaled stream - may be set before the stream exists. |
kwiberg | 686a8ef | 2016-02-26 03:00:35 -0800 | [diff] [blame] | 304 | std::unique_ptr<webrtc::AudioSinkInterface> default_sink_; |
solenberg | 8093d54 | 2015-11-12 06:02:30 -0800 | [diff] [blame] | 305 | // Default SSRC to use for RTCP receiver reports in case of no signaled |
solenberg | 0a617e2 | 2015-10-20 15:49:38 -0700 | [diff] [blame] | 306 | // send streams. See: https://code.google.com/p/webrtc/issues/detail?id=4740 |
solenberg | 8093d54 | 2015-11-12 06:02:30 -0800 | [diff] [blame] | 307 | // and https://code.google.com/p/chromium/issues/detail?id=547661 |
| 308 | uint32_t receiver_reports_ssrc_ = 0xFA17FA17u; |
solenberg | 1ac5614 | 2015-10-13 03:58:19 -0700 | [diff] [blame] | 309 | |
solenberg | c96df77 | 2015-10-21 13:01:53 -0700 | [diff] [blame] | 310 | class WebRtcAudioSendStream; |
| 311 | std::map<uint32_t, WebRtcAudioSendStream*> send_streams_; |
solenberg | 3a94154 | 2015-11-16 07:34:50 -0800 | [diff] [blame] | 312 | std::vector<webrtc::RtpExtension> send_rtp_extensions_; |
Steve Anton | bb50ce5 | 2018-03-26 10:24:32 -0700 | [diff] [blame] | 313 | std::string mid_; |
solenberg | c96df77 | 2015-10-21 13:01:53 -0700 | [diff] [blame] | 314 | |
| 315 | class WebRtcAudioReceiveStream; |
solenberg | 7add058 | 2015-11-20 09:59:34 -0800 | [diff] [blame] | 316 | std::map<uint32_t, WebRtcAudioReceiveStream*> recv_streams_; |
Fredrik Solenberg | 4b60c73 | 2015-05-07 14:07:48 +0200 | [diff] [blame] | 317 | std::vector<webrtc::RtpExtension> recv_rtp_extensions_; |
solenberg | c96df77 | 2015-10-21 13:01:53 -0700 | [diff] [blame] | 318 | |
Danil Chapovalov | 00c7183 | 2018-06-15 15:58:38 +0200 | [diff] [blame] | 319 | absl::optional<webrtc::AudioSendStream::Config::SendCodecSpec> |
ossu | 20a4b3f | 2017-04-27 02:08:52 -0700 | [diff] [blame] | 320 | send_codec_spec_; |
solenberg | 72e29d2 | 2016-03-08 06:35:16 -0800 | [diff] [blame] | 321 | |
Karl Wiberg | 0812634 | 2018-03-20 19:18:55 +0100 | [diff] [blame] | 322 | // TODO(kwiberg): Per-SSRC codec pair IDs? |
| 323 | const webrtc::AudioCodecPairId codec_pair_id_ = |
| 324 | webrtc::AudioCodecPairId::Create(); |
| 325 | |
Benjamin Wright | bfb444c | 2018-10-15 10:20:24 -0700 | [diff] [blame] | 326 | // Per peer connection crypto options that last for the lifetime of the peer |
| 327 | // connection. |
| 328 | const webrtc::CryptoOptions crypto_options_; |
Benjamin Wright | 84583f6 | 2018-10-04 14:22:34 -0700 | [diff] [blame] | 329 | // Unsignaled streams have an option to have a frame decryptor set on them. |
| 330 | rtc::scoped_refptr<webrtc::FrameDecryptorInterface> |
| 331 | unsignaled_frame_decryptor_; |
| 332 | |
solenberg | c96df77 | 2015-10-21 13:01:53 -0700 | [diff] [blame] | 333 | RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(WebRtcVoiceMediaChannel); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 334 | }; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 335 | } // namespace cricket |
| 336 | |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 337 | #endif // MEDIA_ENGINE_WEBRTC_VOICE_ENGINE_H_ |