日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

WebRTC 的音频处理流水线

發布時間:2024/4/11 编程问答 51 豆豆
生活随笔 收集整理的這篇文章主要介紹了 WebRTC 的音频处理流水线 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

基于 RTC 場景下要解決的聲音的問題,WebRTC 有一個大體如下圖所示的音頻處理流水線:

WebRTC 的音頻處理流水線,不是一次性建立起來的,而是分階段分步驟建立的。整體而言,可以認為這個流水線分兩個階段建立,或者可以認為這個流水線分為兩部分:一部分可稱為靜態流水線,另一部分可稱為動態流水線,或者也可以稱為前端和后端。靜態流水線,在某個時間點建立一次,隨后在整個 WebRTC 通信過程中基本保持不變;動態流水線則在通信過程中,可能出現較為頻繁的變動,如本地打開或禁用錄制啟動發送或停止發送音頻數據,遠端發送者加入或退出頻道等,都會改變動態流水線。

如此,WebRTC 的音頻處理流水線大致如下圖所示:

WebRTC 音頻的靜態流水線,建立之后,其相關節點狀態由 AudioState 維護和管理。WebRTC 音頻的靜態流水線,主要包括 AudioDeviceModule,AudioProcessing,和 AudioMixer 等,其中 AudioDeviceModule 用于采集和播放音頻數據,AudioProcessing 主要用于對錄制的音頻數據做初始處理,如回聲消除,降噪等,AudioMixer 主要用于對遠端發送過來的音頻數據做混音。

WebRTC 音頻的靜態流水線在 WebRtcVoiceEngine 初始化時建立:

void WebRtcVoiceEngine::Init() {RTC_DCHECK(worker_thread_checker_.IsCurrent());RTC_LOG(LS_INFO) << "WebRtcVoiceEngine::Init";// TaskQueue expects to be created/destroyed on the same thread.low_priority_worker_queue_.reset(new rtc::TaskQueue(task_queue_factory_->CreateTaskQueue("rtc-low-prio", webrtc::TaskQueueFactory::Priority::LOW)));// Load our audio codec lists.RTC_LOG(LS_INFO) << "Supported send codecs in order of preference:";send_codecs_ = CollectCodecs(encoder_factory_->GetSupportedEncoders());for (const AudioCodec& codec : send_codecs_) {RTC_LOG(LS_INFO) << ToString(codec);}RTC_LOG(LS_INFO) << "Supported recv codecs in order of preference:";recv_codecs_ = CollectCodecs(decoder_factory_->GetSupportedDecoders());for (const AudioCodec& codec : recv_codecs_) {RTC_LOG(LS_INFO) << ToString(codec);}#if defined(WEBRTC_INCLUDE_INTERNAL_AUDIO_DEVICE)// No ADM supplied? Create a default one.if (!adm_) {adm_ = webrtc::AudioDeviceModule::Create(webrtc::AudioDeviceModule::kPlatformDefaultAudio, task_queue_factory_);} #endif // WEBRTC_INCLUDE_INTERNAL_AUDIO_DEVICERTC_CHECK(adm());webrtc::adm_helpers::Init(adm());webrtc::apm_helpers::Init(apm());// Set up AudioState.{webrtc::AudioState::Config config;if (audio_mixer_) {config.audio_mixer = audio_mixer_;} else {config.audio_mixer = webrtc::AudioMixerImpl::Create();}config.audio_processing = apm_;config.audio_device_module = adm_;audio_state_ = webrtc::AudioState::Create(config);}// Connect the ADM to our audio path.adm()->RegisterAudioCallback(audio_state()->audio_transport());// Set default engine options.{AudioOptions options;options.echo_cancellation = true;options.auto_gain_control = true;options.noise_suppression = true;options.highpass_filter = true;options.stereo_swapping = false;options.audio_jitter_buffer_max_packets = 200;options.audio_jitter_buffer_fast_accelerate = false;options.audio_jitter_buffer_min_delay_ms = 0;options.audio_jitter_buffer_enable_rtx_handling = false;options.typing_detection = true;options.experimental_agc = false;options.extended_filter_aec = false;options.delay_agnostic_aec = false;options.experimental_ns = false;options.residual_echo_detector = true;bool error = ApplyOptions(options);RTC_DCHECK(error);}initialized_ = true; }

WebRTC 音頻靜態流水線的幾個節點,由外部創建好注入進來,并通過 AudioState 創建的 AudioTransport 連接起來:WebRtcVoiceEngine::Init()
根據需要創建 AudioDeviceModule,然后創建 AudioState,AudioState 用注入的 AudioMixer 和 AudioProcessing 創建 AudioTransport,WebRtcVoiceEngine::Init() 隨后將 AudioState 創建的 AudioTransport 作為 AudioCallback 注冊給 AudioDeviceModule。WebRTC 音頻靜態流水線在 WebRtcVoiceEngine::Init() 中建立。動態流水線和靜態流水線在 AudioState 和 AudioTransport 處交匯。

AudioTransport

AudioDeviceModule 在錄到音頻數據時,將音頻數據送給 AudioTransport,在播放音頻時,也向 AudioTransport 拿要播放的音頻數據。AudioTransport 是錄制的音頻數據和播放的音頻數據的交匯點。

AudioTransport 接口的定義 (位于 webrtc/src/modules/audio_device/include/audio_device_defines.h) 如下:

class AudioTransport {public:virtual int32_t RecordedDataIsAvailable(const void* audioSamples,const size_t nSamples,const size_t nBytesPerSample,const size_t nChannels,const uint32_t samplesPerSec,const uint32_t totalDelayMS,const int32_t clockDrift,const uint32_t currentMicLevel,const bool keyPressed,uint32_t& newMicLevel) = 0; // NOLINT// Implementation has to setup safe values for all specified out parameters.virtual int32_t NeedMorePlayData(const size_t nSamples,const size_t nBytesPerSample,const size_t nChannels,const uint32_t samplesPerSec,void* audioSamples,size_t& nSamplesOut, // NOLINTint64_t* elapsed_time_ms,int64_t* ntp_time_ms) = 0; // NOLINT// Method to pull mixed render audio data from all active VoE channels.// The data will not be passed as reference for audio processing internally.virtual void PullRenderData(int bits_per_sample,int sample_rate,size_t number_of_channels,size_t number_of_frames,void* audio_data,int64_t* elapsed_time_ms,int64_t* ntp_time_ms) = 0;protected:virtual ~AudioTransport() {} };

WebRTC 中的 AudioTransport 接口實現 AudioTransportImpl 定義 (位于 webrtc/src/audio/audio_transport_impl.h) 如下:

namespace webrtc {class AudioSendStream;class AudioTransportImpl : public AudioTransport {public:AudioTransportImpl(AudioMixer* mixer, AudioProcessing* audio_processing);~AudioTransportImpl() override;int32_t RecordedDataIsAvailable(const void* audioSamples,const size_t nSamples,const size_t nBytesPerSample,const size_t nChannels,const uint32_t samplesPerSec,const uint32_t totalDelayMS,const int32_t clockDrift,const uint32_t currentMicLevel,const bool keyPressed,uint32_t& newMicLevel) override;int32_t NeedMorePlayData(const size_t nSamples,const size_t nBytesPerSample,const size_t nChannels,const uint32_t samplesPerSec,void* audioSamples,size_t& nSamplesOut,int64_t* elapsed_time_ms,int64_t* ntp_time_ms) override;void PullRenderData(int bits_per_sample,int sample_rate,size_t number_of_channels,size_t number_of_frames,void* audio_data,int64_t* elapsed_time_ms,int64_t* ntp_time_ms) override;void UpdateSendingStreams(std::vector<AudioSendStream*> streams,int send_sample_rate_hz,size_t send_num_channels);void SetStereoChannelSwapping(bool enable);bool typing_noise_detected() const;const voe::AudioLevel& audio_level() const { return audio_level_; }private:// Shared.AudioProcessing* audio_processing_ = nullptr;// Capture side.rtc::CriticalSection capture_lock_;std::vector<AudioSendStream*> sending_streams_ RTC_GUARDED_BY(capture_lock_);int send_sample_rate_hz_ RTC_GUARDED_BY(capture_lock_) = 8000;size_t send_num_channels_ RTC_GUARDED_BY(capture_lock_) = 1;bool typing_noise_detected_ RTC_GUARDED_BY(capture_lock_) = false;bool swap_stereo_channels_ RTC_GUARDED_BY(capture_lock_) = false;PushResampler<int16_t> capture_resampler_;voe::AudioLevel audio_level_;TypingDetection typing_detection_;// Render side.rtc::scoped_refptr<AudioMixer> mixer_;AudioFrame mixed_frame_;// Converts mixed audio to the audio device output rate.PushResampler<int16_t> render_resampler_;RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(AudioTransportImpl); }; } // namespace webrtc

AudioTransportImpl 實現的 AudioTransport 接口的要求的操作:將錄制的音頻數據送出去,及獲取播放的音頻數據。

AudioTransportImpl 對象創建時保存注入的 AudioMixer 和 AudioProcessing 對象指針:

AudioTransportImpl::AudioTransportImpl(AudioMixer* mixer,AudioProcessing* audio_processing): audio_processing_(audio_processing), mixer_(mixer) {RTC_DCHECK(mixer);RTC_DCHECK(audio_processing); }AudioTransportImpl::~AudioTransportImpl() {}

播放

AudioDeviceModule 可以通過 AudioTransportImpl 的 NeedMorePlayData() 和 PullRenderData() 獲取用于播放的音頻數據:

// Resample audio in |frame| to given sample rate preserving the // channel count and place the result in |destination|. int Resample(const AudioFrame& frame,const int destination_sample_rate,PushResampler<int16_t>* resampler,int16_t* destination) {const int number_of_channels = static_cast<int>(frame.num_channels_);const int target_number_of_samples_per_channel =destination_sample_rate / 100;resampler->InitializeIfNeeded(frame.sample_rate_hz_, destination_sample_rate,number_of_channels);// TODO(yujo): make resampler take an AudioFrame, and add special case// handling of muted frames.return resampler->Resample(frame.data(), frame.samples_per_channel_ * number_of_channels,destination, number_of_channels * target_number_of_samples_per_channel); } . . . . . . // Mix all received streams, feed the result to the AudioProcessing module, then // resample the result to the requested output rate. int32_t AudioTransportImpl::NeedMorePlayData(const size_t nSamples,const size_t nBytesPerSample,const size_t nChannels,const uint32_t samplesPerSec,void* audioSamples,size_t& nSamplesOut,int64_t* elapsed_time_ms,int64_t* ntp_time_ms) {RTC_DCHECK_EQ(sizeof(int16_t) * nChannels, nBytesPerSample);RTC_DCHECK_GE(nChannels, 1);RTC_DCHECK_LE(nChannels, 2);RTC_DCHECK_GE(samplesPerSec,static_cast<uint32_t>(AudioProcessing::NativeRate::kSampleRate8kHz));// 100 = 1 second / data duration (10 ms).RTC_DCHECK_EQ(nSamples * 100, samplesPerSec);RTC_DCHECK_LE(nBytesPerSample * nSamples * nChannels,AudioFrame::kMaxDataSizeBytes);mixer_->Mix(nChannels, &mixed_frame_);*elapsed_time_ms = mixed_frame_.elapsed_time_ms_;*ntp_time_ms = mixed_frame_.ntp_time_ms_;const auto error = audio_processing_->ProcessReverseStream(&mixed_frame_);RTC_DCHECK_EQ(error, AudioProcessing::kNoError);nSamplesOut = Resample(mixed_frame_, samplesPerSec, &render_resampler_,static_cast<int16_t*>(audioSamples));RTC_DCHECK_EQ(nSamplesOut, nChannels * nSamples);return 0; }// Used by Chromium - same as NeedMorePlayData() but because Chrome has its // own APM instance, does not call audio_processing_->ProcessReverseStream(). void AudioTransportImpl::PullRenderData(int bits_per_sample,int sample_rate,size_t number_of_channels,size_t number_of_frames,void* audio_data,int64_t* elapsed_time_ms,int64_t* ntp_time_ms) {RTC_DCHECK_EQ(bits_per_sample, 16);RTC_DCHECK_GE(number_of_channels, 1);RTC_DCHECK_GE(sample_rate, AudioProcessing::NativeRate::kSampleRate8kHz);// 100 = 1 second / data duration (10 ms).RTC_DCHECK_EQ(number_of_frames * 100, sample_rate);// 8 = bits per byte.RTC_DCHECK_LE(bits_per_sample / 8 * number_of_frames * number_of_channels,AudioFrame::kMaxDataSizeBytes);mixer_->Mix(number_of_channels, &mixed_frame_);*elapsed_time_ms = mixed_frame_.elapsed_time_ms_;*ntp_time_ms = mixed_frame_.ntp_time_ms_;auto output_samples = Resample(mixed_frame_, sample_rate, &render_resampler_,static_cast<int16_t*>(audio_data));RTC_DCHECK_EQ(output_samples, number_of_channels * number_of_frames); }

NeedMorePlayData() 執行如下操作:

  • 通過 AudioMixer 對遠端的音頻流做混音。AudioTransportImpl 沒有直接與 AudioReceiveStream 建立連接,但 AudioMixer 的 source 實際上是 AudioReceiveStream。
  • 把混音后的音頻數據送進 AudioProcessing。主要用于回聲消除。
  • 對混音后的音頻數據重采樣。如我們前面看到的 AudioMixerImpl 的實現,它的混音輸出采樣率有它自己的決定邏輯,可能與 AudioDeviceModule 設備需要的不一致,這里需要按照設備的需要,將混音后的數據重采樣到設備需要的采樣率。
  • 返回重采樣的數據。
  • PullRenderData() 的操作與 NeedMorePlayData() 僅有的區別在于,沒有上面的第 2 步。如注釋中所說,在某些情況下,混音之后的數據無需送進 AudioProcessing。此外,PullRenderData() 要求調用者只能請求單通道的音頻數據,NeedMorePlayData() 的調用者則可以請求單通道或立體聲的。

    重采樣和混音的更多細節,這里不再贅述。

    錄制

    AudioDeviceModule 可以通過 AudioTransport 的 RecordedDataIsAvailable() 把錄制的數據送出去:

    // We want to process at the lowest sample rate and channel count possible // without losing information. Choose the lowest native rate at least equal to // the minimum of input and codec rates, choose lowest channel count, and // configure the audio frame. void InitializeCaptureFrame(int input_sample_rate,int send_sample_rate_hz,size_t input_num_channels,size_t send_num_channels,AudioFrame* audio_frame) {RTC_DCHECK(audio_frame);int min_processing_rate_hz = std::min(input_sample_rate, send_sample_rate_hz);for (int native_rate_hz : AudioProcessing::kNativeSampleRatesHz) {audio_frame->sample_rate_hz_ = native_rate_hz;if (audio_frame->sample_rate_hz_ >= min_processing_rate_hz) {break;}}audio_frame->num_channels_ = std::min(input_num_channels, send_num_channels); }void ProcessCaptureFrame(uint32_t delay_ms,bool key_pressed,bool swap_stereo_channels,AudioProcessing* audio_processing,AudioFrame* audio_frame) {RTC_DCHECK(audio_processing);RTC_DCHECK(audio_frame);audio_processing->set_stream_delay_ms(delay_ms);audio_processing->set_stream_key_pressed(key_pressed);int error = audio_processing->ProcessStream(audio_frame);RTC_DCHECK_EQ(0, error) << "ProcessStream() error: " << error;if (swap_stereo_channels) {AudioFrameOperations::SwapStereoChannels(audio_frame);} } . . . . . . // Not used in Chromium. Process captured audio and distribute to all sending // streams, and try to do this at the lowest possible sample rate. int32_t AudioTransportImpl::RecordedDataIsAvailable(const void* audio_data,const size_t number_of_frames,const size_t bytes_per_sample,const size_t number_of_channels,const uint32_t sample_rate,const uint32_t audio_delay_milliseconds,const int32_t /*clock_drift*/,const uint32_t /*volume*/,const bool key_pressed,uint32_t& /*new_mic_volume*/) { // NOLINT: to avoid changing APIsRTC_DCHECK(audio_data);RTC_DCHECK_GE(number_of_channels, 1);RTC_DCHECK_LE(number_of_channels, 2);RTC_DCHECK_EQ(2 * number_of_channels, bytes_per_sample);RTC_DCHECK_GE(sample_rate, AudioProcessing::NativeRate::kSampleRate8kHz);// 100 = 1 second / data duration (10 ms).RTC_DCHECK_EQ(number_of_frames * 100, sample_rate);RTC_DCHECK_LE(bytes_per_sample * number_of_frames * number_of_channels,AudioFrame::kMaxDataSizeBytes);int send_sample_rate_hz = 0;size_t send_num_channels = 0;bool swap_stereo_channels = false;{rtc::CritScope lock(&capture_lock_);send_sample_rate_hz = send_sample_rate_hz_;send_num_channels = send_num_channels_;swap_stereo_channels = swap_stereo_channels_;}std::unique_ptr<AudioFrame> audio_frame(new AudioFrame());InitializeCaptureFrame(sample_rate, send_sample_rate_hz, number_of_channels,send_num_channels, audio_frame.get());voe::RemixAndResample(static_cast<const int16_t*>(audio_data),number_of_frames, number_of_channels, sample_rate,&capture_resampler_, audio_frame.get());ProcessCaptureFrame(audio_delay_milliseconds, key_pressed,swap_stereo_channels, audio_processing_,audio_frame.get());// Typing detection (utilizes the APM/VAD decision). We let the VAD determine// if we're using this feature or not.// TODO(solenberg): GetConfig() takes a lock. Work around that.bool typing_detected = false;if (audio_processing_->GetConfig().voice_detection.enabled) {if (audio_frame->vad_activity_ != AudioFrame::kVadUnknown) {bool vad_active = audio_frame->vad_activity_ == AudioFrame::kVadActive;typing_detected = typing_detection_.Process(key_pressed, vad_active);}}// Measure audio level of speech after all processing.double sample_duration = static_cast<double>(number_of_frames) / sample_rate;audio_level_.ComputeLevel(*audio_frame, sample_duration);// Copy frame and push to each sending stream. The copy is required since an// encoding task will be posted internally to each stream.{rtc::CritScope lock(&capture_lock_);typing_noise_detected_ = typing_detected;RTC_DCHECK_GT(audio_frame->samples_per_channel_, 0);if (!sending_streams_.empty()) {auto it = sending_streams_.begin();while (++it != sending_streams_.end()) {std::unique_ptr<AudioFrame> audio_frame_copy(new AudioFrame());audio_frame_copy->CopyFrom(*audio_frame);(*it)->SendAudioData(std::move(audio_frame_copy));}// Send the original frame to the first stream w/o copying.(*sending_streams_.begin())->SendAudioData(std::move(audio_frame));}}return 0; } . . . . . . void AudioTransportImpl::UpdateSendingStreams(std::vector<AudioSendStream*> streams,int send_sample_rate_hz,size_t send_num_channels) {rtc::CritScope lock(&capture_lock_);sending_streams_ = std::move(streams);send_sample_rate_hz_ = send_sample_rate_hz;send_num_channels_ = send_num_channels; }

    AudioTransportImpl 的 RecordedDataIsAvailable() 拿到錄音得到的音頻數據,對音頻數據執行如下操作:

  • 重采樣。AudioDeviceModule 送進來的音頻數據的采樣率是設備采集的原始采樣率,這里需要把采集的音頻數據轉換為適合 WebRTC 的音頻處理模塊如降噪和回聲消除等模塊及編碼器處理的采樣率。
  • 對音頻數據做回聲消除,降噪等處理。
  • 把音頻數據送出去,給注入的 AudioSendStream。
  • AudioState

    AudioState 既是 WebRTC 音頻流水線節點的容器,也可用于對這些節點做一些管理和控制,以及將不同節點粘起來構成流水線。AudioState 接口的定義 (位于 webrtc/src/call/audio_state.h) 如下:

    namespace webrtc {class AudioTransport;// AudioState holds the state which must be shared between multiple instances of // webrtc::Call for audio processing purposes. class AudioState : public rtc::RefCountInterface {public:struct Config {Config();~Config();// The audio mixer connected to active receive streams. One per// AudioState.rtc::scoped_refptr<AudioMixer> audio_mixer;// The audio processing module.rtc::scoped_refptr<webrtc::AudioProcessing> audio_processing;// TODO(solenberg): Temporary: audio device module.rtc::scoped_refptr<webrtc::AudioDeviceModule> audio_device_module;};struct Stats {// Audio peak level (max(abs())), linearly on the interval [0,32767].int32_t audio_level = -1;// See:// https://w3c.github.io/webrtc-stats/#dom-rtcmediastreamtrackstats-totalaudioenergydouble total_energy = 0.0f;double total_duration = 0.0f;};virtual AudioProcessing* audio_processing() = 0;virtual AudioTransport* audio_transport() = 0;// Enable/disable playout of the audio channels. Enabled by default.// This will stop playout of the underlying audio device but start a task// which will poll for audio data every 10ms to ensure that audio processing// happens and the audio stats are updated.virtual void SetPlayout(bool enabled) = 0;// Enable/disable recording of the audio channels. Enabled by default.// This will stop recording of the underlying audio device and no audio// packets will be encoded or transmitted.virtual void SetRecording(bool enabled) = 0;virtual Stats GetAudioInputStats() const = 0;virtual void SetStereoChannelSwapping(bool enable) = 0;static rtc::scoped_refptr<AudioState> Create(const AudioState::Config& config);~AudioState() override {} }; } // namespace webrtc

    作為音頻處理流水線的容器,AudioState 接口主要包含了 AudioMixer、AudioProcessing 和 AudioDeviceModule。AudioState 對象的創建主要通過靜態工廠方法完成。

    WebRTC 提供了 AudioState 接口的實現,該實現的定義 (位于 webrtc/src/audio/audio_state.h ) 如下:

    namespace webrtc {class AudioSendStream; class AudioReceiveStream;namespace internal {class AudioState : public webrtc::AudioState {public:explicit AudioState(const AudioState::Config& config);~AudioState() override;AudioProcessing* audio_processing() override;AudioTransport* audio_transport() override;void SetPlayout(bool enabled) override;void SetRecording(bool enabled) override;Stats GetAudioInputStats() const override;void SetStereoChannelSwapping(bool enable) override;AudioDeviceModule* audio_device_module() {RTC_DCHECK(config_.audio_device_module);return config_.audio_device_module.get();}bool typing_noise_detected() const;void AddReceivingStream(webrtc::AudioReceiveStream* stream);void RemoveReceivingStream(webrtc::AudioReceiveStream* stream);void AddSendingStream(webrtc::AudioSendStream* stream,int sample_rate_hz,size_t num_channels);void RemoveSendingStream(webrtc::AudioSendStream* stream);private:void UpdateAudioTransportWithSendingStreams();rtc::ThreadChecker thread_checker_;rtc::ThreadChecker process_thread_checker_;const webrtc::AudioState::Config config_;bool recording_enabled_ = true;bool playout_enabled_ = true;// Transports mixed audio from the mixer to the audio device and// recorded audio to the sending streams.AudioTransportImpl audio_transport_;// Null audio poller is used to continue polling the audio streams if audio// playout is disabled so that audio processing still happens and the audio// stats are still updated.std::unique_ptr<NullAudioPoller> null_audio_poller_;std::unordered_set<webrtc::AudioReceiveStream*> receiving_streams_;struct StreamProperties {int sample_rate_hz = 0;size_t num_channels = 0;};std::map<webrtc::AudioSendStream*, StreamProperties> sending_streams_;RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(AudioState); }; } // namespace internal

    如上面看到的,AudioState 對象在 WebRtcVoiceEngine::Init() 創建。AudioState 的 Create() 函數定義 (位于 webrtc/src/audio/audio_state.cpp) 如下:

    rtc::scoped_refptr<AudioState> AudioState::Create(const AudioState::Config& config) {return new rtc::RefCountedObject<internal::AudioState>(config); }

    WebRTC 中實際在用的 AudioState 對象為 webrtc::internal::AudioState 的對象。

    AudioState 完成的工作主要為:

  • 音頻流水線的節點的 Getter
  • 管理音頻靜態流水線和動態流水線的連接,也就是添加和移除 AudioSendStream 和 AudioReceiveStream:把 AudioSendStream 注入給 AudioTransportImpl 或從中移除,把 AudioReceiveStream 添加到 AudioMixer 或從中移除。這些接口主要由 webrtc::Call 使用。
  • 控制播放和錄制的啟動和停止。
  • webrtc::internal::AudioState 的具體實現 (位于 webrtc/src/audio/audio_state.cpp) 如下:

    namespace webrtc { namespace internal {AudioState::AudioState(const AudioState::Config& config): config_(config),audio_transport_(config_.audio_mixer, config_.audio_processing.get()) {process_thread_checker_.Detach();RTC_DCHECK(config_.audio_mixer);RTC_DCHECK(config_.audio_device_module); }AudioState::~AudioState() {RTC_DCHECK(thread_checker_.IsCurrent());RTC_DCHECK(receiving_streams_.empty());RTC_DCHECK(sending_streams_.empty()); }AudioProcessing* AudioState::audio_processing() {RTC_DCHECK(config_.audio_processing);return config_.audio_processing.get(); }AudioTransport* AudioState::audio_transport() {return &audio_transport_; }bool AudioState::typing_noise_detected() const {RTC_DCHECK(thread_checker_.IsCurrent());return audio_transport_.typing_noise_detected(); }void AudioState::AddReceivingStream(webrtc::AudioReceiveStream* stream) {RTC_DCHECK(thread_checker_.IsCurrent());RTC_DCHECK_EQ(0, receiving_streams_.count(stream));receiving_streams_.insert(stream);if (!config_.audio_mixer->AddSource(static_cast<internal::AudioReceiveStream*>(stream))) {RTC_DLOG(LS_ERROR) << "Failed to add source to mixer.";}// Make sure playback is initialized; start playing if enabled.auto* adm = config_.audio_device_module.get();if (!adm->Playing()) {if (adm->InitPlayout() == 0) {if (playout_enabled_) {adm->StartPlayout();}} else {RTC_DLOG_F(LS_ERROR) << "Failed to initialize playout.";}} }void AudioState::RemoveReceivingStream(webrtc::AudioReceiveStream* stream) {RTC_DCHECK(thread_checker_.IsCurrent());auto count = receiving_streams_.erase(stream);RTC_DCHECK_EQ(1, count);config_.audio_mixer->RemoveSource(static_cast<internal::AudioReceiveStream*>(stream));if (receiving_streams_.empty()) {config_.audio_device_module->StopPlayout();} }void AudioState::AddSendingStream(webrtc::AudioSendStream* stream,int sample_rate_hz,size_t num_channels) {RTC_DCHECK(thread_checker_.IsCurrent());auto& properties = sending_streams_[stream];properties.sample_rate_hz = sample_rate_hz;properties.num_channels = num_channels;UpdateAudioTransportWithSendingStreams();// Make sure recording is initialized; start recording if enabled.auto* adm = config_.audio_device_module.get();if (!adm->Recording()) {if (adm->InitRecording() == 0) {if (recording_enabled_) {adm->StartRecording();}} else {RTC_DLOG_F(LS_ERROR) << "Failed to initialize recording.";}} }void AudioState::RemoveSendingStream(webrtc::AudioSendStream* stream) {RTC_DCHECK(thread_checker_.IsCurrent());auto count = sending_streams_.erase(stream);RTC_DCHECK_EQ(1, count);UpdateAudioTransportWithSendingStreams();if (sending_streams_.empty()) {config_.audio_device_module->StopRecording();} }void AudioState::SetPlayout(bool enabled) {RTC_LOG(INFO) << "SetPlayout(" << enabled << ")";RTC_DCHECK(thread_checker_.IsCurrent());if (playout_enabled_ != enabled) {playout_enabled_ = enabled;if (enabled) {null_audio_poller_.reset();if (!receiving_streams_.empty()) {config_.audio_device_module->StartPlayout();}} else {config_.audio_device_module->StopPlayout();null_audio_poller_ =absl::make_unique<NullAudioPoller>(&audio_transport_);}} }void AudioState::SetRecording(bool enabled) {RTC_LOG(INFO) << "SetRecording(" << enabled << ")";RTC_DCHECK(thread_checker_.IsCurrent());if (recording_enabled_ != enabled) {recording_enabled_ = enabled;if (enabled) {if (!sending_streams_.empty()) {config_.audio_device_module->StartRecording();}} else {config_.audio_device_module->StopRecording();}} }AudioState::Stats AudioState::GetAudioInputStats() const {RTC_DCHECK(thread_checker_.IsCurrent());const voe::AudioLevel& audio_level = audio_transport_.audio_level();Stats result;result.audio_level = audio_level.LevelFullRange();RTC_DCHECK_LE(0, result.audio_level);RTC_DCHECK_GE(32767, result.audio_level);result.total_energy = audio_level.TotalEnergy();result.total_duration = audio_level.TotalDuration();return result; }void AudioState::SetStereoChannelSwapping(bool enable) {RTC_DCHECK(thread_checker_.IsCurrent());audio_transport_.SetStereoChannelSwapping(enable); }void AudioState::UpdateAudioTransportWithSendingStreams() {RTC_DCHECK(thread_checker_.IsCurrent());std::vector<webrtc::AudioSendStream*> sending_streams;int max_sample_rate_hz = 8000;size_t max_num_channels = 1;for (const auto& kv : sending_streams_) {sending_streams.push_back(kv.first);max_sample_rate_hz = std::max(max_sample_rate_hz, kv.second.sample_rate_hz);max_num_channels = std::max(max_num_channels, kv.second.num_channels);}audio_transport_.UpdateSendingStreams(std::move(sending_streams),max_sample_rate_hz, max_num_channels); } } // namespace internal

    總結一下:

    • WebRtcVoiceEngine::Init() 建立了 WebRTC 音頻處理的靜態流水線。
    • AudioTransport 將音頻處理靜態流水線的各個節點粘起來。
    • webrtc::Call 使用 AudioState 將音頻處理流水線的靜態部分和動態部分連接起來。
    超強干貨來襲 云風專訪:近40年碼齡,通宵達旦的技術人生

    總結

    以上是生活随笔為你收集整理的WebRTC 的音频处理流水线的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    五月婷婷av在线 | 激情视频二区 | 国产高清中文字幕 | av免费网页 | 亚洲精品久久久蜜臀下载官网 | 狠狠色丁香久久婷婷综合五月 | 2023亚洲精品国偷拍自产在线 | 中文字幕av最新更新 | av无限看 | 伊人va | 人人添人人澡人人澡人人人爽 | 五月婷婷导航 | 国产96在线观看 | 日韩一二三在线 | 亚洲一区二区高潮无套美女 | 激情五月婷婷激情 | 国内精品久久久久久中文字幕 | 午夜av片| 色欲综合视频天天天 | 国产美女在线精品免费观看 | 一级黄色片毛片 | 18网站在线观看 | 色视频网站在线观看一=区 a视频免费在线观看 | 激情丁香久久 | 狠狠干我 | 伊人国产在线观看 | 国产成人综合图片 | 国产高清视频免费最新在线 | 四虎免费av | 天天干天天做天天爱 | 久久综合久久综合这里只有精品 | 日本91在线 | 久草久草视频 | 国产一级二级三级在线观看 | 成人免费视频免费观看 | 日韩欧美一区二区三区在线观看 | 在线小视频 | 亚洲精品短视频 | av一级久久| 四虎在线视频免费观看 | 成人在线黄色电影 | 免费av在线网站 | 国产精品 亚洲精品 | 国产成人精品亚洲日本在线观看 | 超薄丝袜一二三区 | 亚洲综合在线五月 | 久久五月天综合 | japanesexxxxfreehd乱熟 | 在线免费av网站 | 国产高清一 | 狠狠色丁香久久婷婷综合_中 | 久色小说 | 夜夜高潮夜夜爽国产伦精品 | av免费看av | 色欧美综合| 色偷偷88欧美精品久久久 | 天天射天天艹 | 欧美精品久久久 | 黄色三级久久 | 久久黄色网页 | 九九热久久免费视频 | 亚洲一级黄色 | 亚洲欧美日韩国产 | 中文字幕在线看视频 | 日韩免费| 亚洲爱av | 国产精成人品免费观看 | 免费网站在线观看成人 | 国产老太婆免费交性大片 | 日韩在线观看你懂的 | 中文字幕在线观看不卡 | 99久久精| 久久婷婷开心 | 精品久久久久久国产 | 国产精品成久久久久 | av中文字幕在线播放 | 亚洲精品456在线播放乱码 | av三级av| 美女视频黄色免费 | 国产精品日韩欧美 | 黄色精品一区二区 | 亚洲成av人影院 | 免费观看日韩 | 丁香六月婷婷开心 | 久久不卡视频 | 中文字幕在线播放第一页 | 91精品少妇偷拍99 | 亚洲在线a | 中文字幕丝袜制服 | 天天操天天干天天爽 | 黄色免费视频在线观看 | 欧美日韩一区二区三区免费视频 | 欧美久久久久久久久中文字幕 | 婷婷在线视频观看 | 久久久免费看视频 | 久草网免费 | 亚洲精品乱码 | 日韩av中文在线 | 精品久久一级片 | 国产亚洲91 | 中文字幕免费在线看 | 免费欧美精品 | 婷婷资源站 | 九九久久国产 | 亚洲综合视频在线播放 | 久久亚洲婷婷 | 99久久这里有精品 | 狠狠色狠狠色终合网 | 一区二区精品在线观看 | 五月开心婷婷网 | 亚洲精品国精品久久99热 | 日韩精品中文字幕在线不卡尤物 | 精品久久久久久亚洲综合网 | 99精品在线视频播放 | 亚洲久草视频 | 国产精品2020| 国产精品福利在线观看 | 超碰精品在线 | 在线观看国产91 | 午夜精品久久久久久久久久久 | 国产高清视频色在线www | 日韩国产欧美在线播放 | 波多野结衣电影久久 | 97精品国产一二三产区 | 亚洲国产精品电影 | 久久免费国产视频 | 香蕉精品视频在线观看 | 日韩在线视 | 天天干天天天 | 激情网站| 国产精品综合久久久久久 | 日韩久久午夜一级啪啪 | 在线v片免费观看视频 | 亚洲伊人成综合网 | av高清一区二区三区 | 黄色精品一区 | 新版资源中文在线观看 | 国产精品一区二区免费看 | 狠狠色丁香婷婷综合基地 | 香蕉影视在线观看 | 中文字幕免费高清av | 五月婷婷丁香在线观看 | 久久私人影院 | 狠狠色丁香九九婷婷综合五月 | 四虎永久精品在线 | 亚洲成人一二三 | 色夜视频 | 免费亚洲黄色 | 国产vs久久| 国产专区视频在线观看 | 丁香色婷婷 | 久久99亚洲热视 | 亚洲国产wwwccc36天堂 | 成人午夜在线电影 | 日韩免费视频一区二区 | 在线观看91精品视频 | 精品欧美一区二区在线观看 | 日韩av五月天 | 色综合久久综合中文综合网 | 天天插视频 | 国产精品青草综合久久久久99 | 国产精品久久久久一区二区国产 | 综合婷婷 | 久久成人综合 | 五月婷婷六月丁香在线观看 | 欧美一级艳片视频免费观看 | 黄色三级在线 | 国产在线资源 | 日韩免费在线观看视频 | 午夜色婷婷 | 免费看污在线观看 | av电影久久| 激情综合色图 | 天天干干 | 国产精品女主播一区二区三区 | 婷婷精品国产一区二区三区日韩 | 在线观看国产福利片 | 欧美日韩亚洲精品在线 | 色综合天天在线 | 去干成人网 | 国产视频在线观看免费 | 久久久久国产成人精品亚洲午夜 | 人人狠狠 | 精品久久久久久综合 | 久久福利精品 | 精品久久一区 | 亚洲黄色在线播放 | ww亚洲ww亚在线观看 | 国产小视频在线 | 伊人成人久久 | 夜夜狠狠 | 美女视频黄频大全免费 | 91大神在线观看视频 | 久二影院 | 91尤物在线播放 | 久久国产免| a电影在线观看 | 国产亚洲综合精品 | 91自拍视频在线观看 | 久久国产精品偷 | 国产美腿白丝袜足在线av | 欧美日韩国产精品一区二区 | 五月婷婷综合色拍 | 91免费在线 | 国产成人一区二区啪在线观看 | 久久综合狠狠综合 | 精品视频在线看 | 国产精品欧美久久久久无广告 | 香蕉视频国产在线 | 亚洲综合少妇 | 中文字幕人成不卡一区 | 亚洲精品 在线视频 | 91九色在线观看视频 | 日韩精品一区二区三区免费视频观看 | 亚洲色图美腿丝袜 | 96久久久| 欧美一级裸体视频 | 国产精品久久久久久久久久ktv | 国产成人91 | 99视频精品免费观看, | 婷婷狠狠操 | 五月婷婷色播 | 国产一区二区三区免费在线观看 | 日韩视频二区 | 国产黄色精品在线 | 新版资源中文在线观看 | 久久久这里有精品 | 中文字幕在线观看视频一区二区三区 | 久久久久麻豆 | 五月婷婷毛片 | 九九热精品视频在线观看 | 亚洲网久久 | 九色91福利 | 国产国语在线 | 欧美一区二区在线刺激视频 | 一区二区三区电影大全 | 久久久www | 天天综合色 | 免费成人av在线看 | av线上免费观看 | 97电影网手机版 | 欧美一级欧美一级 | 国产色啪| 国产中文字幕在线视频 | 黄色精品久久久 | 国产女人40精品一区毛片视频 | 毛片网站免费 | 日日夜夜精品视频 | 国产精品入口麻豆 | 日韩在线一级 | 国产成人一区二区三区在线观看 | 日韩在线高清 | 亚洲男男gaygayxxxgv | 日韩欧美xxx | 人人爱人人爽 | 黄色三级免费网址 | 成人午夜精品久久久久久久3d | 久久字幕精品一区 | 国产99在线播放 | 国产91在线观 | 69av视频在线观看 | 91香蕉视频 mp4| 欧美精品做受xxx性少妇 | 日日草视频 | 51久久成人国产精品麻豆 | 99在线观看视频 | 狠狠躁夜夜a产精品视频 | 亚洲撸撸 | 99久久婷婷国产精品综合 | 久久国产精品99久久人人澡 | 日韩精品一区二区久久 | 精品国产一区二区三区在线观看 | 波多野结衣精品 | 欧美精品网站 | 天天色天| 亚洲电影黄色 | 亚洲精品在线观看视频 | 97在线观看免费观看 | 久草国产在线 | 91精品啪在线观看国产 | 午夜久操 | 精品国产一区二区三区久久久 | 91香蕉视频在线下载 | 国产一区欧美二区 | 欧美日比视频 | 一区二区电影在线观看 | 国产字幕av | 超碰日韩在线 | 国际精品网 | 国产在线精 | 9在线观看免费高清完整版在线观看明 | 亚洲精品videossex少妇 | 91九色精品| 日韩免费播放 | 国产精品久久久一区二区 | 成人精品视频 | 国产成人免费av电影 | 久久国产三级 | 在线观看视频免费播放 | 黄色大全免费观看 | 久草在线免费看视频 | 成人黄色一级视频 | 色吊丝在线永久观看最新版本 | 香蕉一区 | 乱男乱女www7788 | 国产综合视频在线观看 | 在线观看黄色 | 精品 激情| 美女精品在线观看 | 久久久久久久国产精品视频 | a视频免费看 | 欧美资源在线观看 | 亚洲欧美日韩一区二区三区在线观看 | 96视频在线 | 激情av在线资源 | 欧美精品久久久久久久久久白贞 | 成人app在线播放 | 亚洲九九爱 | 一区免费在线 | 91精品国产麻豆国产自产影视 | 精品福利视频在线观看 | 美女啪啪图片 | 久久福利在线 | 成人一级片免费看 | 中文字幕视频一区二区 | 黄色福利网 | 黄色免费网战 | 91av蜜桃 | 久久久综合精品 | 亚洲另类视频在线 | 樱空桃av | 伊色综合久久之综合久久 | 国产精品久久久久久久av大片 | 久久99这里只有精品 | 91中文字幕永久在线 | 久草com| 在线观看国产亚洲 | 国产精品www| 五月婷婷六月丁香激情 | av不卡免费在线观看 | 久久亚洲欧美日韩精品专区 | 免费看片网页 | 亚洲国产欧美一区二区三区丁香婷 | 免费在线日韩 | 欧美日韩高清在线观看 | 中文字幕在线免费观看视频 | 国产美女免费观看 | 97超碰资源总站 | 亚洲黄色在线 | 国产一区二区三精品久久久无广告 | 久草免费在线视频观看 | 久久综合久久伊人 | 亚洲国产欧美在线人成大黄瓜 | 在线观看自拍 | 亚洲午夜小视频 | 久久久久久国产精品亚洲78 | 91丨九色丨高潮丰满 | 国产成人精品一区二区三区网站观看 | 日韩精品一区二区三区丰满 | 一级做a视频 | 中文字幕永久在线 | 日本性生活一级片 | 在线观看视频国产 | 国产国语在线 | sm免费xx网站 | 欧美日韩一区二区视频在线观看 | 国产午夜一区二区 | av品善网 | 亚洲精品免费在线播放 | 91成人精品 | 天天操福利视频 | 国产精品视频在线观看 | 久久深爱网 | 一区二区视频免费在线观看 | 五月天电影免费在线观看一区 | 久久久久久久久久网站 | 玖玖视频国产 | 精品国产亚洲在线 | 国产精品久久久久免费 | 91探花系列在线播放 | 久久久久免费精品视频 | 色婷婷国产精品一区在线观看 | 毛片1000部免费看 | 国产又黄又爽又猛视频日本 | 成年人在线播放视频 | 伊人成人精品 | 久久视频这里有精品 | 亚洲欧美国产精品18p | 日韩精品一区二区三区在线视频 | 特级毛片爽www免费版 | 夜夜骑日日操 | 激情av资源网 | 中文字幕电影高清在线观看 | 国产黄色片免费 | 国产精品久久二区 | 婷婷精品 | 午夜精品一区二区三区免费 | 国内精品久久久久久久久久 | 探花在线观看 | 国产在线国偷精品产拍 | 久久这里只有精品视频首页 | 午夜av色| 黄色网址中文字幕 | 黄色字幕网 | 午夜精品视频免费在线观看 | 日韩免| 日韩在线观看的 | 国产日本高清 | 99热最新网址 | 亚洲精品国产精品国自产 | 狠狠狠色丁香综合久久天下网 | 亚洲国产精彩中文乱码av | 午夜精品99久久免费 | 久久在线视频精品 | 黄色一级性片 | 伊人久久av| 亚洲色图av| 成人精品国产免费网站 | 六月丁香六月婷婷 | 中文字幕在线观看一区 | 色姑娘综合天天 | 亚洲欧美视频在线播放 | 国产在线一区二区三区播放 | 中文字幕在线观看视频一区二区三区 | 91色九色 | 婷婷久久综合网 | 久久a久久 | 久久短视频 | 欧美精品成人在线 | 欧美成年人在线视频 | 日韩免费二区 | 国产专区精品视频 | 国产黄色片久久久 | av在线免费观看不卡 | 69视频网站 | 香蕉影院在线播放 | 国产精品区一区 | 97品白浆高清久久久久久 | 日韩精品综合在线 | av+在线播放在线播放 | 四虎影视8848dvd | 国产精品婷婷 | 欧洲色综合 | 国产精品扒开做爽爽的视频 | 国产精品久久久久免费a∨ 欧美一级性生活片 | 97自拍超碰 | 久久久久久久久久网站 | 精品久久亚洲 | 成人动漫视频在线 | 在线观看一级片 | 99这里只有精品视频 | 婷婷丁香色 | 久久r精品 | 日产av在线播放 | 日韩精品在线视频免费观看 | 91亚洲精品视频 | 午夜视频在线观看一区二区三区 | 国产亚洲精品bv在线观看 | 日韩免费视频播放 | 久久8精品 | 在线之家官网 | 色综合久久天天 | 日本中文字幕一二区观 | 国产精品婷婷午夜在线观看 | 欧美一级性生活片 | 婷婷色在线视频 | 国产精品1区 | 久久激情网站 | 国产一区二区电影在线观看 | 亚洲视频久久久久 | 国产精品久久网站 | 97免费在线观看视频 | 91大神免费视频 | 久久久久久久久久久久久影院 | 亚洲全部视频 | 日韩在线 一区二区 | 日韩另类在线 | 久草在线综合网 | 波多野结衣一区三区 | 国产亚洲精品女人久久久久久 | 久久伦理 | 国产精品毛片一区二区 | 国产玖玖精品视频 | 日本黄色大片免费看 | 久久久久久久久亚洲精品 | 九九热免费观看 | 免费国产在线视频 | 毛片二区| 久久精品一区二区三区中文字幕 | 国产一区在线免费 | 在线中文字幕电影 | 亚洲欧美在线视频免费 | 少妇bbw揉bbb欧美 | 国产香蕉视频在线观看 | 九九免费在线视频 | 欧美日韩在线观看一区二区 | 香蕉视频色 | 伊人日日干 | 国产一二区在线观看 | 福利区在线观看 | 一级成人免费视频 | 中文字幕日韩免费视频 | 超碰97国产在线 | 丁香在线 | 天天综合人人 | 日本丰满少妇免费一区 | 黄色一级大片在线免费看产 | 在线免费观看视频你懂的 | 日韩在线 一区二区 | 黄色亚洲大片免费在线观看 | 美女免费视频网站 | 欧美国产日韩在线视频 | 日韩理论电影在线 | 国产精品久久久久高潮 | 黄色软件视频大全免费下载 | 久久综合网色—综合色88 | 99精品国产99久久久久久福利 | 五月天伊人网 | 91免费在线 | 91人人干| 97国产在线观看 | 99综合久久 | 综合网欧美 | 91网在线观看 | 国产午夜精品久久 | www免费网站在线观看 | 日日噜噜噜噜夜夜爽亚洲精品 | 热re99久久精品国产66热 | 国产精品嫩草69影院 | 国产精品视频在线观看 | 欧美日韩视频一区二区三区 | 婷婷视频在线观看 | 九九免费在线视频 | 国产精品99久久久精品免费观看 | 成人av在线直播 | 麻豆视频免费在线 | 精品福利视频在线观看 | 日韩高清免费电影 | 亚洲另类交 | 亚洲久草在线 | 九九精品视频在线观看 | 成人小视频免费在线观看 | 欧美大片大全 | 蜜桃av综合网 | 国产精品尤物视频 | 日本在线观看中文字幕无线观看 | 国产免费一区二区三区网站免费 | 久久久久亚洲精品国产 | 精品国产乱码久久久久久1区2匹 | 日日操操 | 国产破处在线视频 | 欧美激情va永久在线播放 | 久久99精品久久久久久久久久久久 | 国产精品综合av一区二区国产馆 | 在线日韩视频 | 黄在线免费观看 | 亚洲精品欧洲精品 | 97电影网站| 曰本免费av| 国产精品美女久久久久久 | www免费网站在线观看 | 免费进去里的视频 | 欧美一级性生活视频 | 亚洲视频每日更新 | 伊人午夜视频 | 91亚洲成人 | 成年人天堂com | 日本精品一区二区三区在线观看 | 亚洲激情小视频 | 国产视频 久久久 | 欧美激情综合五月色丁香 | 国产欧美综合在线观看 | 国产精品一区免费观看 | 国产精品成人aaaaa网站 | 在线免费观看黄色大片 | 日韩av高清在线观看 | 激情综合电影网 | 国产在线色视频 | 日日夜夜天天 | 亚洲欧美va | 久久y| 国产直播av | 国产大陆亚洲精品国产 | 国产精品久久久久国产精品日日 | 久久天天躁狠狠躁夜夜不卡公司 | 狠狠干婷婷色 | 一区二区三区在线免费观看视频 | 成在人线av | 天天草综合网 | 激情欧美一区二区三区免费看 | 午夜久久久久久久久久影院 | 欧美在线不卡一区 | 国产一级久久久 | 久久天堂影院 | 六月丁香婷 | 91大神免费视频 | 区一区二区三区中文字幕 | 黄色影院在线免费观看 | 国产高清网站 | 国产黄色免费电影 | 欧美一区日韩精品 | 国产91全国探花系列在线播放 | 最新色站 | 久久国产精品99国产精 | 国产精品欧美久久久久无广告 | 99热日本 | 亚洲第五色综合网 | 久久99视频免费 | 国产成人精品一区二区在线观看 | 黄色一区三区 | 最新av网址在线观看 | 在线看免费 | 五月综合激情婷婷 | 久久久久久蜜桃一区二区 | 美女网站色| 欧美亚洲国产日韩 | 伊人伊成久久人综合网站 | 日韩视频在线不卡 | 久久精品久久久久久久 | 91av视频在线免费观看 | 日韩精品高清不卡 | 九色在线视频 | 青青草国产在线 | 日韩欧美一区二区不卡 | 免费看三级 | 人人澡人人干 | 国产色综合 | 日韩特黄一级欧美毛片特黄 | 国产成人61精品免费看片 | 天堂资源在线观看视频 | 91九色最新 | 色多多污污在线观看 | 精品视频专区 | 精品福利在线视频 | 天天操狠狠操 | 91网址在线观看 | 国产午夜亚洲精品 | 国内精品久久久久久久97牛牛 | 在线高清一区 | 国产一区国产二区在线观看 | 亚洲综合日韩在线 | 在线观看国产一区 | 丝袜美腿亚洲 | 国产精品久久精品 | 国产精品资源在线观看 | 在线国产激情视频 | 国产精品剧情在线亚洲 | 亚洲精品字幕在线观看 | 黄色毛片在线观看 | 欧美日韩国产一二三区 | 91热这里只有精品 | 狠狠久久婷婷 | 亚州精品在线视频 | 在线直播av | 国产精品久久久久久高潮 | 亚洲黄色小说网址 | 国产成人av福利 | 在线电影日韩 | 久久一线| www久| 国产成人精品亚洲日本在线观看 | 国产精品激情偷乱一区二区∴ | 男女拍拍免费视频 | 亚洲一级二级 | 欧美日韩调教 | 免费黄色网址网站 | 日本aaa在线观看 | 国产精品黄色 | 中文字幕在线观看第一区 | 日本中文字幕免费观看 | 三级午夜片| 西西www4444大胆视频 | www.香蕉视频在线观看 | 国产精品高潮呻吟久久av无 | av天天干 | 国产精品一区二区久久久久 | 97超碰免费在线观看 | 最近高清中文在线字幕在线观看 | 91原创在线观看 | 99久久综合狠狠综合久久 | 免费在线一区二区三区 | 不卡电影一区二区三区 | 国产精品黄 | 天天操比 | 久草在线免费色站 | 免费亚洲黄色 | 亚洲欧美成aⅴ人在线观看 四虎在线观看 | 手机在线永久免费观看av片 | 免费人做人爱www的视 | 亚洲精品在线观看的 | 蜜臀av性久久久久蜜臀av | 国产91在线免费视频 | 国产精品一区二区在线看 | 亚洲国产成人在线播放 | 欧美精选一区二区三区 | 久草资源在线 | 五月天激情视频在线观看 | 亚洲精品在线观看的 | av大片免费在线观看 | 国产精品麻豆视频 | 欧美午夜精品久久久久 | 在线亚洲播放 | 五月亚洲 | 国产精品久久久久免费a∨ 欧美一级性生活片 | 国产精品久久久久久久久久不蜜月 | 久久婷亚洲五月一区天天躁 | 啪嗒啪嗒免费观看完整版 | 91在线网站| 一区二区三区免费在线观看 | 91看片网址 | 国产美女精品视频 | 一级黄色片在线免费观看 | 二区中文字幕 | 免费看特级毛片 | 久久一区精品 | 欧美日韩免费在线视频 | 深夜国产在线 | 黄色福利网站 | 成人久久精品 | 国产精品久久久久久吹潮天美传媒 | 美女久久久| 久久男人中文字幕资源站 | 在线观看日韩免费视频 | 99精品毛片| 亚洲黄色免费在线 | 欧美一区日韩一区 | 国产精品久久久久久久av大片 | 91九色视频在线播放 | 四虎国产免费 | 91最新在线观看 | 亚洲天堂网视频在线观看 | 日韩av成人在线观看 | 成人欧美一区二区三区在线观看 | 婷婷六月综合网 | 2023亚洲精品国偷拍自产在线 | 成人免费在线观看电影 | 久久视频网址 | 韩国av永久免费 | 国产 欧美 在线 | 激情片av | 久久伦理网 | 五月婷婷激情六月 | 五月天视频网 | 国产精品高潮呻吟久久久久 | 国产精品1000 | 亚洲美女免费精品视频在线观看 | 美女露久久 | 国产国语在线 | 久久福利国产 | 超碰com | 欧美日韩高清一区二区三区 | 色婷av| 少妇搡bbbb搡bbb搡69 | 91在线播放视频 | 欧美日韩一区二区三区在线免费观看 | 国产自在线 | 视频在线观看入口黄最新永久免费国产 | 不卡的av在线 | av网站在线免费观看 | 精品视频www | 在线激情小视频 | 一级欧美日韩 | 天天干天天做天天操 | 99精彩视频在线观看免费 | 成人免费看电影 | 久久午夜精品 | 狠狠干电影 | 九九久久婷婷 | 99精品国产一区二区 | 久草成人在线 | 伊人五月天综合 | 欧美精品一区二区在线观看 | 香蕉久久国产 | 午夜精品一二三区 | 午夜婷婷在线播放 | av网站手机在线观看 | 五月天狠狠操 | 97在线播放| 国产在线播放一区二区三区 | 亚洲91精品在线观看 | 久久超级碰视频 | 免费开视频 | 成人中文字幕在线观看 | 天堂视频中文在线 | 国产精品18久久久久久不卡孕妇 | 国产精品中文在线 | 黄色在线成人 | 国产99久久久国产精品成人免费 | 在线播放第一页 | 狠狠躁18三区二区一区ai明星 | 丁香六月伊人 | 国产一级片直播 | 国内少妇自拍视频一区 | 毛片美女网站 | av电影av在线 | 999热线在线观看 | 国产麻豆果冻传媒在线观看 | 亚洲国产一区二区精品专区 | 精品亚洲午夜久久久久91 | 人人干网 | 国产一线天在线观看 | 高清av在线免费观看 | 欧美在线视频日韩 | 18网站在线观看 | 国产成人久久精品77777 | 五月综合网站 | 国产视频1| 91热| www.黄色小说.com | 色婷婷久久一区二区 | 久久久www成人免费毛片麻豆 | 在线观看91av| 久久黄色美女 | 亚洲精品视频在线观看网站 | 中文字幕在线免费看 | 日韩欧美有码在线 | 99热手机在线观看 | 五月婷久久 | 免费在线观看午夜视频 | 国产视频精品久久 | 欧美在线91 | 在线成人免费 | 欧美日韩在线视频观看 | 久久精品国产精品 | 成人h视频| 国产高清专区 | 国产中文 | 国产人成看黄久久久久久久久 | 国产资源中文字幕 | 欧美精品你懂的 | 日韩精品一区二区三区免费观看视频 | 18国产精品白浆在线观看免费 | 国产综合精品一区二区三区 | 国产成人不卡 | 亚洲第二色 | 欧美日韩在线免费观看视频 | 国产很黄很色的视频 | 日韩精品一区二区三区高清免费 | 97av影院| 最近2019好看的中文字幕免费 | 日韩中文字幕免费看 | 99色免费 | 99在线精品视频观看 | 国产区 在线 | 国产精品久久嫩一区二区免费 | 在线看黄色av | 久久天天躁夜夜躁狠狠躁2022 | 色综合久久综合网 | 国产99精品| 狠狠色网| 国语麻豆 | 97伊人网 | 免费人成在线观看网站 | 日韩免费二区 | 91完整版观看 | 中文字幕亚洲国产 | 成人在线电影观看 | 色久综合 | 日韩久久一区 | 五月激情片| 欧美精品中文字幕亚洲专区 | 亚洲综合视频在线播放 | 日韩精品第一区 | 久久综合久久综合久久综合 | 九九三级毛片 | 69国产成人综合久久精品欧美 | 亚洲欧美日韩在线一区二区 | 成人免费观看大片 | 美女国内精品自产拍在线播放 | 国产在线播放一区二区三区 | 69亚洲视频 | 人人插人人草 | 不卡的av电影 | 国产一二三区av | 精品电影一区 | 一区二区三区在线观看免费视频 | 国产日产精品一区二区三区四区的观看方式 | 玖玖爱国产在线 | 日本系列中文字幕 | 久久99热久久99精品 | 二区三区在线视频 | 在线免费观看国产黄色 | 亚洲春色综合另类校园电影 | 91亚洲激情| 99精品久久99久久久久 | 亚洲成人黄色在线 | 婷婷在线资源 | 成年人视频在线免费 | 一区二区伦理电影 | 婷婷色网 | 国产精品黄网站在线观看 | 一级片免费观看 | 日韩大片在线免费观看 | 国产精品乱码久久 | 激情久久一区二区三区 | 久久免费试看 | 久久不射影院 | 久草免费手机视频 | 日韩精品一区二区三区免费观看视频 | 国产第一页在线播放 | 精品网站999www | 亚洲人在线7777777精品 | 在线观看日韩一区 | 97国产大学生情侣白嫩酒店 | 欧美日韩有码 | 中文字幕av在线免费 | 国产精品aⅴ| 久久精品79国产精品 | 久久a久久 | 91成人免费在线视频 | 粉嫩aⅴ一区二区三区 | 在线观看成人av | 视频在线观看入口黄最新永久免费国产 | 国产成人精品一区在线 | 国产中文字幕亚洲 | 国产精品小视频网站 | 欧美大片在线看免费观看 | 久久精品电影 | 精品国产乱码久久久久久三级人 | 日日狠狠 | 91亚洲精品久久久久图片蜜桃 | 成人在线视频在线观看 | 色婷婷综合久色 | 免费欧美精品 | 国产精品观看 | 日韩av在线小说 | 天天爱天天操天天爽 | 九九日韩 | 一区二区国产精品 | 久精品在线 | 亚洲欧美日韩在线一区二区 | 欧美日韩伦理一区 | 亚洲日本成人 | 午夜精品久久久久久久99热影院 | 国产人免费人成免费视频 | 久久成人国产精品入口 | 国产伦精品一区二区三区四区视频 | 色婷五月天 | 日韩影视大全 | 六月婷婷网 | 中文字幕在线视频国产 | 丁香视频全集免费观看 | 欧美日韩国产在线精品 | 韩国精品福利一区二区三区 | 五月婷婷丁香在线观看 | 91av视频导航| 亚洲精品国产精品久久99 | 99精品久久只有精品 | 欧洲精品久久久久毛片完整版 | 九色视频自拍 | 国产麻豆精品95视频 | 久草在线视频网 | 久久99久久99精品中文字幕 | 国产精品白丝av | 午夜久操| 亚洲人天堂 | 亚洲午夜精品一区二区三区电影院 | 国精产品999国精产品视频 | 激情网站网址 | 人人爽久久涩噜噜噜网站 | 在线日本看片免费人成视久网 | www色综合| 日本高清中文字幕有码在线 | 国产亚洲无 | 国产系列 在线观看 | 中文字幕在线高清 | 99国产情侣在线播放 | 日韩免费观看高清 | 国产一区电影在线观看 | 国产精品福利在线观看 | 久久亚洲热 | 中文乱幕日产无线码1区 | 成人一区二区三区在线 | 精品免费一区 | 五月导航 | 婷婷在线视频观看 | 九九免费在线观看视频 | 超碰人人99 | 911精品美国片911久久久 | 精品久久福利 | 五月婷婷开心 | 国产在线色视频 | 久久综合九色综合久99 | 久爱精品在线 | 久久久久久久99精品免费观看 | 日韩av黄 | 99精品免费视频 | 狠狠干综合 | 午夜精品久久久久99热app | 久草影视在线 | 国产这里只有精品 | 色婷婷天天干 | 免费高清在线观看成人 | 永久av免费在线观看 | 在线免费观看欧美日韩 | 国产亚洲片 | 91在线视频观看 | 成人v|