| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 // | 4 // |
| 5 // AudioConverter implementation. Uses MultiChannelSincResampler for resampling | 5 // AudioConverter implementation. Uses MultiChannelSincResampler for resampling |
| 6 // audio, ChannelMixer for channel mixing, and AudioPullFifo for buffering. | 6 // audio, ChannelMixer for channel mixing, and AudioPullFifo for buffering. |
| 7 // | 7 // |
| 8 // Delay estimates are provided to InputCallbacks based on the frame delay | 8 // Delay estimates are provided to InputCallbacks based on the frame delay |
| 9 // information reported via the resampler and FIFO units. | 9 // information reported via the resampler and FIFO units. |
| 10 | 10 |
| (...skipping 10 matching lines...) Expand all Loading... |
| 21 #include "media/base/vector_math.h" | 21 #include "media/base/vector_math.h" |
| 22 | 22 |
| 23 namespace media { | 23 namespace media { |
| 24 | 24 |
| 25 AudioConverter::AudioConverter(const AudioParameters& input_params, | 25 AudioConverter::AudioConverter(const AudioParameters& input_params, |
| 26 const AudioParameters& output_params, | 26 const AudioParameters& output_params, |
| 27 bool disable_fifo) | 27 bool disable_fifo) |
| 28 : chunk_size_(output_params.frames_per_buffer()), | 28 : chunk_size_(output_params.frames_per_buffer()), |
| 29 downmix_early_(false), | 29 downmix_early_(false), |
| 30 resampler_frame_delay_(0), | 30 resampler_frame_delay_(0), |
| 31 input_channel_count_(input_params.channels()) { | 31 input_channel_count_(input_params.channels()), |
| 32 io_sample_rate_ratio_(input_params.sample_rate() / |
| 33 static_cast<double>(output_params.sample_rate())) { |
| 32 CHECK(input_params.IsValid()); | 34 CHECK(input_params.IsValid()); |
| 33 CHECK(output_params.IsValid()); | 35 CHECK(output_params.IsValid()); |
| 34 | 36 |
| 35 // Handle different input and output channel layouts. | 37 // Handle different input and output channel layouts. |
| 36 if (input_params.channel_layout() != output_params.channel_layout()) { | 38 if (input_params.channel_layout() != output_params.channel_layout()) { |
| 37 DVLOG(1) << "Remixing channel layout from " << input_params.channel_layout() | 39 DVLOG(1) << "Remixing channel layout from " << input_params.channel_layout() |
| 38 << " to " << output_params.channel_layout() << "; from " | 40 << " to " << output_params.channel_layout() << "; from " |
| 39 << input_params.channels() << " channels to " | 41 << input_params.channels() << " channels to " |
| 40 << output_params.channels() << " channels."; | 42 << output_params.channels() << " channels."; |
| 41 channel_mixer_.reset(new ChannelMixer(input_params, output_params)); | 43 channel_mixer_.reset(new ChannelMixer(input_params, output_params)); |
| 42 | 44 |
| 43 // Pare off data as early as we can for efficiency. | 45 // Pare off data as early as we can for efficiency. |
| 44 downmix_early_ = input_params.channels() > output_params.channels(); | 46 downmix_early_ = input_params.channels() > output_params.channels(); |
| 45 } | 47 } |
| 46 | 48 |
| 47 // Only resample if necessary since it's expensive. | 49 // Only resample if necessary since it's expensive. |
| 48 if (input_params.sample_rate() != output_params.sample_rate()) { | 50 if (input_params.sample_rate() != output_params.sample_rate()) { |
| 49 DVLOG(1) << "Resampling from " << input_params.sample_rate() << " to " | 51 DVLOG(1) << "Resampling from " << input_params.sample_rate() << " to " |
| 50 << output_params.sample_rate(); | 52 << output_params.sample_rate(); |
| 51 const double io_sample_rate_ratio = input_params.sample_rate() / | |
| 52 static_cast<double>(output_params.sample_rate()); | |
| 53 const int request_size = disable_fifo ? SincResampler::kDefaultRequestSize : | 53 const int request_size = disable_fifo ? SincResampler::kDefaultRequestSize : |
| 54 input_params.frames_per_buffer(); | 54 input_params.frames_per_buffer(); |
| 55 resampler_.reset(new MultiChannelResampler( | 55 resampler_.reset(new MultiChannelResampler( |
| 56 downmix_early_ ? output_params.channels() : | 56 downmix_early_ ? output_params.channels() : |
| 57 input_params.channels(), | 57 input_params.channels(), |
| 58 io_sample_rate_ratio, request_size, base::Bind( | 58 io_sample_rate_ratio_, request_size, base::Bind( |
| 59 &AudioConverter::ProvideInput, base::Unretained(this)))); | 59 &AudioConverter::ProvideInput, base::Unretained(this)))); |
| 60 } | 60 } |
| 61 | 61 |
| 62 input_frame_duration_ = base::TimeDelta::FromMicroseconds( | 62 input_frame_duration_ = base::TimeDelta::FromMicroseconds( |
| 63 base::Time::kMicrosecondsPerSecond / | 63 base::Time::kMicrosecondsPerSecond / |
| 64 static_cast<double>(input_params.sample_rate())); | 64 static_cast<double>(input_params.sample_rate())); |
| 65 output_frame_duration_ = base::TimeDelta::FromMicroseconds( | 65 output_frame_duration_ = base::TimeDelta::FromMicroseconds( |
| 66 base::Time::kMicrosecondsPerSecond / | 66 base::Time::kMicrosecondsPerSecond / |
| 67 static_cast<double>(output_params.sample_rate())); | 67 static_cast<double>(output_params.sample_rate())); |
| 68 | 68 |
| (...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 108 if (resampler_) | 108 if (resampler_) |
| 109 resampler_->Flush(); | 109 resampler_->Flush(); |
| 110 } | 110 } |
| 111 | 111 |
| 112 int AudioConverter::ChunkSize() const { | 112 int AudioConverter::ChunkSize() const { |
| 113 if (!resampler_) | 113 if (!resampler_) |
| 114 return chunk_size_; | 114 return chunk_size_; |
| 115 return resampler_->ChunkSize(); | 115 return resampler_->ChunkSize(); |
| 116 } | 116 } |
| 117 | 117 |
| 118 int AudioConverter::RequestSize() const { |
| 119 if (resampler_) |
| 120 return audio_fifo_ ? chunk_size_ |
| 121 : SincResampler::kDefaultRequestSize; |
| 122 return chunk_size_; |
| 123 } |
| 124 |
| 118 void AudioConverter::ConvertWithDelay(const base::TimeDelta& initial_delay, | 125 void AudioConverter::ConvertWithDelay(const base::TimeDelta& initial_delay, |
| 119 AudioBus* dest) { | 126 AudioBus* dest) { |
| 120 initial_delay_ = initial_delay; | 127 initial_delay_ = initial_delay; |
| 121 | 128 |
| 122 if (transform_inputs_.empty()) { | 129 if (transform_inputs_.empty()) { |
| 123 dest->Zero(); | 130 dest->Zero(); |
| 124 return; | 131 return; |
| 125 } | 132 } |
| 126 | 133 |
| 127 // Determine if channel mixing should be done and if it should be done before | 134 // Determine if channel mixing should be done and if it should be done before |
| (...skipping 112 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 240 else | 247 else |
| 241 SourceCallback(0, dest); | 248 SourceCallback(0, dest); |
| 242 } | 249 } |
| 243 | 250 |
| 244 void AudioConverter::CreateUnmixedAudioIfNecessary(int frames) { | 251 void AudioConverter::CreateUnmixedAudioIfNecessary(int frames) { |
| 245 if (!unmixed_audio_ || unmixed_audio_->frames() != frames) | 252 if (!unmixed_audio_ || unmixed_audio_->frames() != frames) |
| 246 unmixed_audio_ = AudioBus::Create(input_channel_count_, frames); | 253 unmixed_audio_ = AudioBus::Create(input_channel_count_, frames); |
| 247 } | 254 } |
| 248 | 255 |
| 249 } // namespace media | 256 } // namespace media |
| OLD | NEW |