Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 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 #include "media/audio/cras/audio_manager_cras.h" | 5 #include "media/audio/cras/audio_manager_cras.h" |
| 6 | 6 |
| 7 #include <algorithm> | 7 #include <algorithm> |
| 8 | 8 |
| 9 #include "base/command_line.h" | 9 #include "base/command_line.h" |
| 10 #include "base/environment.h" | 10 #include "base/environment.h" |
| 11 #include "base/logging.h" | 11 #include "base/logging.h" |
| 12 #include "base/nix/xdg_util.h" | 12 #include "base/nix/xdg_util.h" |
| 13 #include "base/stl_util.h" | 13 #include "base/stl_util.h" |
| 14 #include "chromeos/audio/audio_device.h" | 14 #include "chromeos/audio/audio_device.h" |
| 15 #include "chromeos/audio/cras_audio_handler.h" | 15 #include "chromeos/audio/cras_audio_handler.h" |
| 16 #include "media/audio/cras/cras_input.h" | 16 #include "media/audio/cras/cras_input.h" |
| 17 #include "media/audio/cras/cras_unified.h" | 17 #include "media/audio/cras/cras_unified.h" |
| 18 #include "media/base/channel_layout.h" | 18 #include "media/base/channel_layout.h" |
| 19 | 19 |
| 20 // cras_util.h headers pull in min/max macros... | 20 // cras_util.h headers pull in min/max macros... |
| 21 // TODO(dgreid): Fix headers such that these aren't imported. | 21 // TODO(dgreid): Fix headers such that these aren't imported. |
| 22 #undef min | 22 #undef min |
| 23 #undef max | 23 #undef max |
| 24 | 24 |
| 25 namespace media { | 25 namespace media { |
| 26 namespace { | |
| 26 | 27 |
| 27 static void AddDefaultDevice(AudioDeviceNames* device_names) { | 28 void AddDefaultDevice(AudioDeviceNames* device_names) { |
| 28 DCHECK(device_names->empty()); | |
|
Henrik Grunell
2015/08/27 08:38:18
Why remove the dcheck?
| |
| 29 | |
| 30 // Cras will route audio from a proper physical device automatically. | 29 // Cras will route audio from a proper physical device automatically. |
| 31 device_names->push_back( | 30 device_names->push_back( |
| 32 AudioDeviceName(AudioManagerBase::kDefaultDeviceName, | 31 AudioDeviceName(AudioManagerBase::kDefaultDeviceName, |
| 33 AudioManagerBase::kDefaultDeviceId)); | 32 AudioManagerBase::kDefaultDeviceId)); |
| 34 } | 33 } |
| 35 | 34 |
| 35 std::string GetMicPositions() { | |
|
Henrik Grunell
2015/08/27 08:38:18
Add comment.
Can the function be called on any th
| |
| 36 // Get the list of devices from CRAS. An internal mic with a non-empty | |
| 37 // positions field indicates the machine has a beamforming capable mic array. | |
| 38 chromeos::AudioDeviceList devices; | |
| 39 chromeos::CrasAudioHandler::Get()->GetAudioDevices(&devices); | |
| 40 for (const auto& device : devices) { | |
| 41 if (device.type == chromeos::AUDIO_TYPE_INTERNAL_MIC) { | |
| 42 // There should be only one internal mic device. | |
| 43 return device.mic_positions; | |
| 44 } | |
| 45 } | |
| 46 return ""; | |
| 47 } | |
| 48 | |
| 36 // Maximum number of output streams that can be open simultaneously. | 49 // Maximum number of output streams that can be open simultaneously. |
| 37 static const int kMaxOutputStreams = 50; | 50 const int kMaxOutputStreams = 50; |
| 38 | 51 |
| 39 // Default sample rate for input and output streams. | 52 // Default sample rate for input and output streams. |
| 40 static const int kDefaultSampleRate = 48000; | 53 const int kDefaultSampleRate = 48000; |
| 41 | 54 |
| 42 // Define bounds for the output buffer size. | 55 // Define bounds for the output buffer size. |
| 43 static const int kMinimumOutputBufferSize = 512; | 56 const int kMinimumOutputBufferSize = 512; |
| 44 static const int kMaximumOutputBufferSize = 8192; | 57 const int kMaximumOutputBufferSize = 8192; |
| 45 | 58 |
| 46 // Default input buffer size. | 59 // Default input buffer size. |
| 47 static const int kDefaultInputBufferSize = 1024; | 60 const int kDefaultInputBufferSize = 1024; |
| 61 | |
| 62 const char kBeamformingDeviceNameSuffix[] = " with beamforming"; | |
| 63 const char kBeamformingDeviceIdSuffix[] = "-beamforming"; | |
|
aluebs-chromium
2015/08/28 19:14:30
Since they are user-facing strings, is this the ri
ajm
2015/08/28 20:17:18
Great question, I wonder the same thing! The exist
dgreid
2015/08/28 20:24:55
I don't think we ever bothered to translate "Defau
| |
| 64 | |
| 65 } // namespace | |
| 48 | 66 |
| 49 bool AudioManagerCras::HasAudioOutputDevices() { | 67 bool AudioManagerCras::HasAudioOutputDevices() { |
| 50 return true; | 68 return true; |
| 51 } | 69 } |
| 52 | 70 |
| 53 bool AudioManagerCras::HasAudioInputDevices() { | 71 bool AudioManagerCras::HasAudioInputDevices() { |
| 54 chromeos::AudioDeviceList devices; | 72 chromeos::AudioDeviceList devices; |
| 55 chromeos::CrasAudioHandler::Get()->GetAudioDevices(&devices); | 73 chromeos::CrasAudioHandler::Get()->GetAudioDevices(&devices); |
| 56 for (size_t i = 0; i < devices.size(); ++i) { | 74 for (size_t i = 0; i < devices.size(); ++i) { |
| 57 if (devices[i].is_input && devices[i].is_for_simple_usage()) | 75 if (devices[i].is_input && devices[i].is_for_simple_usage()) |
| 58 return true; | 76 return true; |
| 59 } | 77 } |
| 60 return false; | 78 return false; |
| 61 } | 79 } |
| 62 | 80 |
| 63 AudioManagerCras::AudioManagerCras(AudioLogFactory* audio_log_factory) | 81 AudioManagerCras::AudioManagerCras(AudioLogFactory* audio_log_factory) |
| 64 : AudioManagerBase(audio_log_factory), | 82 : AudioManagerBase(audio_log_factory), |
| 65 has_keyboard_mic_(false) { | 83 has_keyboard_mic_(false), |
| 84 beamforming_device_name_( | |
| 85 std::string(AudioManagerBase::kDefaultDeviceName) + | |
| 86 kBeamformingDeviceNameSuffix, | |
| 87 std::string(AudioManagerBase::kDefaultDeviceId) + | |
| 88 kBeamformingDeviceIdSuffix), | |
| 89 mic_positions_(GetMicPositions()) { | |
| 66 SetMaxOutputStreamsAllowed(kMaxOutputStreams); | 90 SetMaxOutputStreamsAllowed(kMaxOutputStreams); |
| 67 } | 91 } |
| 68 | 92 |
| 69 AudioManagerCras::~AudioManagerCras() { | 93 AudioManagerCras::~AudioManagerCras() { |
| 70 Shutdown(); | 94 Shutdown(); |
| 71 } | 95 } |
| 72 | 96 |
| 73 void AudioManagerCras::ShowAudioInputSettings() { | 97 void AudioManagerCras::ShowAudioInputSettings() { |
| 74 NOTIMPLEMENTED(); | 98 NOTIMPLEMENTED(); |
| 75 } | 99 } |
| 76 | 100 |
| 77 void AudioManagerCras::GetAudioInputDeviceNames( | 101 void AudioManagerCras::GetAudioInputDeviceNames( |
| 78 AudioDeviceNames* device_names) { | 102 AudioDeviceNames* device_names) { |
| 103 // A non-empty mic_positions indicates we have a beamforming capable mic | |
| 104 // array. Add the virtual beamforming device to the list. When this device is | |
| 105 // queried through GetInputStreamParameters, provide the cached mic positions. | |
| 106 if (!mic_positions_.empty()) { | |
|
aluebs-chromium
2015/08/28 19:14:30
Shouldn't we be more restrictive here? For instanc
ajm
2015/08/28 23:38:31
That's a good point. I have those checks in MSAP f
aluebs-chromium
2015/08/29 00:00:26
Yes, I know it was in MSAP, but if we do it here w
ajm
2015/09/02 20:38:41
Forgot to switch the check to > 1 mic. Done in PS#
| |
| 107 device_names->push_back(beamforming_device_name_); | |
| 108 } | |
| 109 | |
| 79 AddDefaultDevice(device_names); | 110 AddDefaultDevice(device_names); |
| 80 } | 111 } |
| 81 | 112 |
| 82 void AudioManagerCras::GetAudioOutputDeviceNames( | 113 void AudioManagerCras::GetAudioOutputDeviceNames( |
| 83 AudioDeviceNames* device_names) { | 114 AudioDeviceNames* device_names) { |
| 84 AddDefaultDevice(device_names); | 115 AddDefaultDevice(device_names); |
| 85 } | 116 } |
| 86 | 117 |
| 87 AudioParameters AudioManagerCras::GetInputStreamParameters( | 118 AudioParameters AudioManagerCras::GetInputStreamParameters( |
| 88 const std::string& device_id) { | 119 const std::string& device_id) { |
| 89 DCHECK(GetTaskRunner()->BelongsToCurrentThread()); | 120 DCHECK(GetTaskRunner()->BelongsToCurrentThread()); |
| 90 | 121 |
| 91 int user_buffer_size = GetUserBufferSize(); | 122 int user_buffer_size = GetUserBufferSize(); |
| 92 int buffer_size = user_buffer_size ? | 123 int buffer_size = user_buffer_size ? |
| 93 user_buffer_size : kDefaultInputBufferSize; | 124 user_buffer_size : kDefaultInputBufferSize; |
| 94 AudioParameters::PlatformEffectsMask effects = | 125 AudioParameters::PlatformEffectsMask effects = |
| 95 has_keyboard_mic_ ? AudioParameters::KEYBOARD_MIC | 126 has_keyboard_mic_ ? AudioParameters::KEYBOARD_MIC |
| 96 : AudioParameters::NO_EFFECTS; | 127 : AudioParameters::NO_EFFECTS; |
| 128 // Return the cached mic positions in the case of the beamforming device. | |
| 129 const std::string& mic_positions = | |
| 130 device_id == beamforming_device_name_.unique_id ? mic_positions_ : ""; | |
| 97 | 131 |
| 98 // TODO(hshi): Fine-tune audio parameters based on |device_id|. The optimal | 132 // TODO(hshi): Fine-tune audio parameters based on |device_id|. The optimal |
| 99 // parameters for the loopback stream may differ from the default. | 133 // parameters for the loopback stream may differ from the default. |
| 100 return AudioParameters( | 134 return AudioParameters(AudioParameters::AUDIO_PCM_LOW_LATENCY, |
| 101 AudioParameters::AUDIO_PCM_LOW_LATENCY, CHANNEL_LAYOUT_STEREO, | 135 CHANNEL_LAYOUT_STEREO, kDefaultSampleRate, 16, |
|
aluebs-chromium
2015/08/28 19:14:30
What does CHANNEL_LAYOUT_STEREO represent? Is this
ajm
2015/08/28 23:38:31
This represents the channel layout for the audio c
aluebs-chromium
2015/08/29 00:00:26
Agreed that this CL is big enough. But it needs to
| |
| 102 kDefaultSampleRate, 16, buffer_size, effects); | 136 buffer_size, mic_positions, effects); |
| 103 } | 137 } |
| 104 | 138 |
| 105 void AudioManagerCras::SetHasKeyboardMic() { | 139 void AudioManagerCras::SetHasKeyboardMic() { |
| 106 DCHECK(GetTaskRunner()->BelongsToCurrentThread()); | 140 DCHECK(GetTaskRunner()->BelongsToCurrentThread()); |
| 107 has_keyboard_mic_ = true; | 141 has_keyboard_mic_ = true; |
| 108 } | 142 } |
| 109 | 143 |
| 110 AudioOutputStream* AudioManagerCras::MakeLinearOutputStream( | 144 AudioOutputStream* AudioManagerCras::MakeLinearOutputStream( |
| 111 const AudioParameters& params) { | 145 const AudioParameters& params) { |
| 112 DCHECK_EQ(AudioParameters::AUDIO_PCM_LINEAR, params.format()); | 146 DCHECK_EQ(AudioParameters::AUDIO_PCM_LINEAR, params.format()); |
| (...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 149 channel_layout = input_params.channel_layout(); | 183 channel_layout = input_params.channel_layout(); |
| 150 buffer_size = | 184 buffer_size = |
| 151 std::min(kMaximumOutputBufferSize, | 185 std::min(kMaximumOutputBufferSize, |
| 152 std::max(buffer_size, input_params.frames_per_buffer())); | 186 std::max(buffer_size, input_params.frames_per_buffer())); |
| 153 } | 187 } |
| 154 | 188 |
| 155 int user_buffer_size = GetUserBufferSize(); | 189 int user_buffer_size = GetUserBufferSize(); |
| 156 if (user_buffer_size) | 190 if (user_buffer_size) |
| 157 buffer_size = user_buffer_size; | 191 buffer_size = user_buffer_size; |
| 158 | 192 |
| 159 return AudioParameters( | 193 return AudioParameters(AudioParameters::AUDIO_PCM_LOW_LATENCY, channel_layout, |
| 160 AudioParameters::AUDIO_PCM_LOW_LATENCY, channel_layout, | 194 sample_rate, bits_per_sample, buffer_size); |
| 161 sample_rate, bits_per_sample, buffer_size, AudioParameters::NO_EFFECTS); | |
| 162 } | 195 } |
| 163 | 196 |
| 164 AudioOutputStream* AudioManagerCras::MakeOutputStream( | 197 AudioOutputStream* AudioManagerCras::MakeOutputStream( |
| 165 const AudioParameters& params) { | 198 const AudioParameters& params) { |
| 166 return new CrasUnifiedStream(params, this); | 199 return new CrasUnifiedStream(params, this); |
| 167 } | 200 } |
| 168 | 201 |
| 169 AudioInputStream* AudioManagerCras::MakeInputStream( | 202 AudioInputStream* AudioManagerCras::MakeInputStream( |
| 170 const AudioParameters& params, const std::string& device_id) { | 203 const AudioParameters& params, const std::string& device_id) { |
| 171 return new CrasInputStream(params, this, device_id); | 204 return new CrasInputStream(params, this, device_id); |
| 172 } | 205 } |
| 173 | 206 |
| 174 snd_pcm_format_t AudioManagerCras::BitsToFormat(int bits_per_sample) { | 207 snd_pcm_format_t AudioManagerCras::BitsToFormat(int bits_per_sample) { |
| 175 switch (bits_per_sample) { | 208 switch (bits_per_sample) { |
| 176 case 8: | 209 case 8: |
| 177 return SND_PCM_FORMAT_U8; | 210 return SND_PCM_FORMAT_U8; |
| 178 case 16: | 211 case 16: |
| 179 return SND_PCM_FORMAT_S16; | 212 return SND_PCM_FORMAT_S16; |
| 180 case 24: | 213 case 24: |
| 181 return SND_PCM_FORMAT_S24; | 214 return SND_PCM_FORMAT_S24; |
| 182 case 32: | 215 case 32: |
| 183 return SND_PCM_FORMAT_S32; | 216 return SND_PCM_FORMAT_S32; |
| 184 default: | 217 default: |
| 185 return SND_PCM_FORMAT_UNKNOWN; | 218 return SND_PCM_FORMAT_UNKNOWN; |
| 186 } | 219 } |
| 187 } | 220 } |
| 188 | 221 |
| 189 } // namespace media | 222 } // namespace media |
| OLD | NEW |