Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(17)

Side by Side Diff: media/audio/android/audio_manager_android.cc

Issue 131503006: Initialization of audio manager for Android is now done on the audio thread (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: All methods in unit tests should now be on the audio thread Created 6 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
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 #include "media/audio/android/audio_manager_android.h" 5 #include "media/audio/android/audio_manager_android.h"
6 6
7 #include "base/android/build_info.h" 7 #include "base/android/build_info.h"
8 #include "base/android/jni_array.h" 8 #include "base/android/jni_array.h"
9 #include "base/android/jni_string.h" 9 #include "base/android/jni_string.h"
10 #include "base/android/scoped_java_ref.h" 10 #include "base/android/scoped_java_ref.h"
11 #include "base/logging.h" 11 #include "base/logging.h"
12 #include "base/message_loop/message_loop.h"
12 #include "base/strings/string_number_conversions.h" 13 #include "base/strings/string_number_conversions.h"
13 #include "jni/AudioManagerAndroid_jni.h" 14 #include "jni/AudioManagerAndroid_jni.h"
14 #include "media/audio/android/audio_record_input.h" 15 #include "media/audio/android/audio_record_input.h"
15 #include "media/audio/android/opensles_input.h" 16 #include "media/audio/android/opensles_input.h"
16 #include "media/audio/android/opensles_output.h" 17 #include "media/audio/android/opensles_output.h"
17 #include "media/audio/audio_manager.h" 18 #include "media/audio/audio_manager.h"
18 #include "media/audio/audio_parameters.h" 19 #include "media/audio/audio_parameters.h"
19 #include "media/audio/fake_audio_input_stream.h" 20 #include "media/audio/fake_audio_input_stream.h"
20 #include "media/base/channel_layout.h" 21 #include "media/base/channel_layout.h"
21 22
(...skipping 18 matching lines...) Expand all
40 static const int kDefaultInputBufferSize = 1024; 41 static const int kDefaultInputBufferSize = 1024;
41 static const int kDefaultOutputBufferSize = 2048; 42 static const int kDefaultOutputBufferSize = 2048;
42 43
43 AudioManager* CreateAudioManager(AudioLogFactory* audio_log_factory) { 44 AudioManager* CreateAudioManager(AudioLogFactory* audio_log_factory) {
44 return new AudioManagerAndroid(audio_log_factory); 45 return new AudioManagerAndroid(audio_log_factory);
45 } 46 }
46 47
47 AudioManagerAndroid::AudioManagerAndroid(AudioLogFactory* audio_log_factory) 48 AudioManagerAndroid::AudioManagerAndroid(AudioLogFactory* audio_log_factory)
48 : AudioManagerBase(audio_log_factory) { 49 : AudioManagerBase(audio_log_factory) {
49 SetMaxOutputStreamsAllowed(kMaxOutputStreams); 50 SetMaxOutputStreamsAllowed(kMaxOutputStreams);
50
51 j_audio_manager_.Reset(
52 Java_AudioManagerAndroid_createAudioManagerAndroid(
53 base::android::AttachCurrentThread(),
54 base::android::GetApplicationContext(),
55 reinterpret_cast<intptr_t>(this)));
56 Init();
57 } 51 }
58 52
59 AudioManagerAndroid::~AudioManagerAndroid() { 53 AudioManagerAndroid::~AudioManagerAndroid() {
60 Close();
61 Shutdown(); 54 Shutdown();
55 // Verify that WillDestroyCurrentMessageLoop() has been called.
56 DCHECK(j_audio_manager_.is_null());
62 } 57 }
63 58
64 bool AudioManagerAndroid::HasAudioOutputDevices() { 59 bool AudioManagerAndroid::HasAudioOutputDevices() {
65 return true; 60 return true;
66 } 61 }
67 62
68 bool AudioManagerAndroid::HasAudioInputDevices() { 63 bool AudioManagerAndroid::HasAudioInputDevices() {
69 return true; 64 return true;
70 } 65 }
71 66
72 void AudioManagerAndroid::GetAudioInputDeviceNames( 67 void AudioManagerAndroid::GetAudioInputDeviceNames(
73 AudioDeviceNames* device_names) { 68 AudioDeviceNames* device_names) {
69 CreateAndInitOnAudioThread();
70
74 // Always add default device parameters as first element. 71 // Always add default device parameters as first element.
75 DCHECK(device_names->empty()); 72 DCHECK(device_names->empty());
76 AddDefaultDevice(device_names); 73 AddDefaultDevice(device_names);
77 74
75 // Get list of available audio devices.
78 JNIEnv* env = AttachCurrentThread(); 76 JNIEnv* env = AttachCurrentThread();
79 ScopedJavaLocalRef<jobjectArray> j_device_array = 77 ScopedJavaLocalRef<jobjectArray> j_device_array =
80 Java_AudioManagerAndroid_getAudioInputDeviceNames( 78 Java_AudioManagerAndroid_getAudioInputDeviceNames(
81 env, j_audio_manager_.obj()); 79 env, j_audio_manager_.obj());
82 jsize len = env->GetArrayLength(j_device_array.obj()); 80 jsize len = env->GetArrayLength(j_device_array.obj());
83 AudioDeviceName device; 81 AudioDeviceName device;
84 for (jsize i = 0; i < len; ++i) { 82 for (jsize i = 0; i < len; ++i) {
85 ScopedJavaLocalRef<jobject> j_device( 83 ScopedJavaLocalRef<jobject> j_device(
86 env, env->GetObjectArrayElement(j_device_array.obj(), i)); 84 env, env->GetObjectArrayElement(j_device_array.obj(), i));
87 ScopedJavaLocalRef<jstring> j_device_name = 85 ScopedJavaLocalRef<jstring> j_device_name =
88 Java_AudioDeviceName_name(env, j_device.obj()); 86 Java_AudioDeviceName_name(env, j_device.obj());
89 ConvertJavaStringToUTF8(env, j_device_name.obj(), &device.device_name); 87 ConvertJavaStringToUTF8(env, j_device_name.obj(), &device.device_name);
90 ScopedJavaLocalRef<jstring> j_device_id = 88 ScopedJavaLocalRef<jstring> j_device_id =
91 Java_AudioDeviceName_id(env, j_device.obj()); 89 Java_AudioDeviceName_id(env, j_device.obj());
92 ConvertJavaStringToUTF8(env, j_device_id.obj(), &device.unique_id); 90 ConvertJavaStringToUTF8(env, j_device_id.obj(), &device.unique_id);
93 device_names->push_back(device); 91 device_names->push_back(device);
94 } 92 }
95 } 93 }
96 94
97 void AudioManagerAndroid::GetAudioOutputDeviceNames( 95 void AudioManagerAndroid::GetAudioOutputDeviceNames(
98 AudioDeviceNames* device_names) { 96 AudioDeviceNames* device_names) {
99 // TODO(henrika): enumerate using GetAudioInputDeviceNames(). 97 // TODO(henrika): enumerate using GetAudioInputDeviceNames().
100 AddDefaultDevice(device_names); 98 AddDefaultDevice(device_names);
101 } 99 }
102 100
103 AudioParameters AudioManagerAndroid::GetInputStreamParameters( 101 AudioParameters AudioManagerAndroid::GetInputStreamParameters(
104 const std::string& device_id) { 102 const std::string& device_id) {
105 JNIEnv* env = AttachCurrentThread(); 103 CreateAndInitOnAudioThread();
104
106 // Use mono as preferred number of input channels on Android to save 105 // Use mono as preferred number of input channels on Android to save
107 // resources. Using mono also avoids a driver issue seen on Samsung 106 // resources. Using mono also avoids a driver issue seen on Samsung
108 // Galaxy S3 and S4 devices. See http://crbug.com/256851 for details. 107 // Galaxy S3 and S4 devices. See http://crbug.com/256851 for details.
108 JNIEnv* env = AttachCurrentThread();
109 CHECK(env);
109 ChannelLayout channel_layout = CHANNEL_LAYOUT_MONO; 110 ChannelLayout channel_layout = CHANNEL_LAYOUT_MONO;
110 int buffer_size = Java_AudioManagerAndroid_getMinInputFrameSize( 111 int buffer_size = Java_AudioManagerAndroid_getMinInputFrameSize(
111 env, GetNativeOutputSampleRate(), 112 env, GetNativeOutputSampleRate(),
112 ChannelLayoutToChannelCount(channel_layout)); 113 ChannelLayoutToChannelCount(channel_layout));
113 int effects = AudioParameters::NO_EFFECTS; 114 int effects = AudioParameters::NO_EFFECTS;
114 effects |= Java_AudioManagerAndroid_shouldUseAcousticEchoCanceler(env) ? 115 effects |= Java_AudioManagerAndroid_shouldUseAcousticEchoCanceler(env) ?
115 AudioParameters::ECHO_CANCELLER : AudioParameters::NO_EFFECTS; 116 AudioParameters::ECHO_CANCELLER : AudioParameters::NO_EFFECTS;
116 AudioParameters params( 117 AudioParameters params(
117 AudioParameters::AUDIO_PCM_LOW_LATENCY, channel_layout, 0, 118 AudioParameters::AUDIO_PCM_LOW_LATENCY, channel_layout, 0,
118 GetNativeOutputSampleRate(), 16, 119 GetNativeOutputSampleRate(), 16,
119 buffer_size <= 0 ? kDefaultInputBufferSize : buffer_size, effects); 120 buffer_size <= 0 ? kDefaultInputBufferSize : buffer_size, effects);
120 return params; 121 return params;
121 } 122 }
122 123
123 AudioOutputStream* AudioManagerAndroid::MakeAudioOutputStream( 124 AudioOutputStream* AudioManagerAndroid::MakeAudioOutputStream(
124 const AudioParameters& params, 125 const AudioParameters& params,
125 const std::string& device_id) { 126 const std::string& device_id) {
126 bool had_no_streams = HadNoAudioStreams(); 127 CreateAndInitOnAudioThread();
128 bool has_no_streams = HasNoAudioStreams();
127 AudioOutputStream* stream = 129 AudioOutputStream* stream =
128 AudioManagerBase::MakeAudioOutputStream(params, std::string()); 130 AudioManagerBase::MakeAudioOutputStream(params, std::string());
129 131
130 // The audio manager for Android creates streams intended for real-time 132 // The audio manager for Android creates streams intended for real-time
131 // VoIP sessions and therefore sets the audio mode to MODE_IN_COMMUNICATION. 133 // VoIP sessions and therefore sets the audio mode to MODE_IN_COMMUNICATION.
132 // If a Bluetooth headset is used, the audio stream will use the SCO 134 // If a Bluetooth headset is used, the audio stream will use the SCO
133 // channel and therefore have a limited bandwidth (8-16kHz). 135 // channel and therefore have a limited bandwidth (8-16kHz).
134 if (stream && had_no_streams) 136 if (stream && has_no_streams)
135 SetCommunicationAudioModeOn(true); 137 SetCommunicationAudioModeOn(true);
136 138
137 { 139 {
138 base::AutoLock lock(streams_lock_); 140 base::AutoLock lock(streams_lock_);
139 streams_.insert(static_cast<OpenSLESOutputStream*>(stream)); 141 streams_.insert(static_cast<OpenSLESOutputStream*>(stream));
140 } 142 }
141 143
142 return stream; 144 return stream;
143 } 145 }
144 146
145 AudioInputStream* AudioManagerAndroid::MakeAudioInputStream( 147 AudioInputStream* AudioManagerAndroid::MakeAudioInputStream(
146 const AudioParameters& params, const std::string& device_id) { 148 const AudioParameters& params, const std::string& device_id) {
147 bool had_no_streams = HadNoAudioStreams(); 149 CreateAndInitOnAudioThread();
150 bool has_no_streams = HasNoAudioStreams();
148 AudioInputStream* stream = 151 AudioInputStream* stream =
149 AudioManagerBase::MakeAudioInputStream(params, device_id); 152 AudioManagerBase::MakeAudioInputStream(params, device_id);
150 153
151 // The audio manager for Android creates streams intended for real-time 154 // The audio manager for Android creates streams intended for real-time
152 // VoIP sessions and therefore sets the audio mode to MODE_IN_COMMUNICATION. 155 // VoIP sessions and therefore sets the audio mode to MODE_IN_COMMUNICATION.
153 // If a Bluetooth headset is used, the audio stream will use the SCO 156 // If a Bluetooth headset is used, the audio stream will use the SCO
154 // channel and therefore have a limited bandwidth (8kHz). 157 // channel and therefore have a limited bandwidth (8kHz).
155 if (stream && had_no_streams) 158 if (stream && has_no_streams)
156 SetCommunicationAudioModeOn(true); 159 SetCommunicationAudioModeOn(true);
157 return stream; 160 return stream;
158 } 161 }
159 162
160 void AudioManagerAndroid::ReleaseOutputStream(AudioOutputStream* stream) { 163 void AudioManagerAndroid::ReleaseOutputStream(AudioOutputStream* stream) {
tommi (sloooow) - chröme 2014/02/17 13:24:55 If this method is expected to be called on the aud
henrika (OOO until Aug 14) 2014/02/17 15:14:41 Done.
161 AudioManagerBase::ReleaseOutputStream(stream); 164 AudioManagerBase::ReleaseOutputStream(stream);
162 165
163 // Restore the audio mode which was used before the first communication- 166 // Restore the audio mode which was used before the first communication-
164 // mode stream was created. 167 // mode stream was created.
165 if (HadNoAudioStreams()) 168 if (HasNoAudioStreams())
166 SetCommunicationAudioModeOn(false); 169 SetCommunicationAudioModeOn(false);
167 base::AutoLock lock(streams_lock_); 170 base::AutoLock lock(streams_lock_);
168 streams_.erase(static_cast<OpenSLESOutputStream*>(stream)); 171 streams_.erase(static_cast<OpenSLESOutputStream*>(stream));
169 } 172 }
170 173
171 void AudioManagerAndroid::ReleaseInputStream(AudioInputStream* stream) { 174 void AudioManagerAndroid::ReleaseInputStream(AudioInputStream* stream) {
175 DCHECK(GetTaskRunner()->BelongsToCurrentThread());
176 DCHECK(!j_audio_manager_.is_null());
172 AudioManagerBase::ReleaseInputStream(stream); 177 AudioManagerBase::ReleaseInputStream(stream);
173 178
174 // Restore the audio mode which was used before the first communication- 179 // Restore the audio mode which was used before the first communication-
175 // mode stream was created. 180 // mode stream was created.
176 if (HadNoAudioStreams()) 181 if (HasNoAudioStreams())
177 SetCommunicationAudioModeOn(false); 182 SetCommunicationAudioModeOn(false);
178 } 183 }
179 184
180 AudioOutputStream* AudioManagerAndroid::MakeLinearOutputStream( 185 AudioOutputStream* AudioManagerAndroid::MakeLinearOutputStream(
181 const AudioParameters& params) { 186 const AudioParameters& params) {
182 DCHECK_EQ(AudioParameters::AUDIO_PCM_LINEAR, params.format()); 187 DCHECK_EQ(AudioParameters::AUDIO_PCM_LINEAR, params.format());
183 return new OpenSLESOutputStream(this, params); 188 return new OpenSLESOutputStream(this, params);
184 } 189 }
185 190
186 AudioOutputStream* AudioManagerAndroid::MakeLowLatencyOutputStream( 191 AudioOutputStream* AudioManagerAndroid::MakeLowLatencyOutputStream(
187 const AudioParameters& params, 192 const AudioParameters& params,
188 const std::string& device_id) { 193 const std::string& device_id) {
189 DLOG_IF(ERROR, !device_id.empty()) << "Not implemented!"; 194 DLOG_IF(ERROR, !device_id.empty()) << "Not implemented!";
190 DCHECK_EQ(AudioParameters::AUDIO_PCM_LOW_LATENCY, params.format()); 195 DCHECK_EQ(AudioParameters::AUDIO_PCM_LOW_LATENCY, params.format());
196 CreateAndInitOnAudioThread();
191 return new OpenSLESOutputStream(this, params); 197 return new OpenSLESOutputStream(this, params);
192 } 198 }
193 199
194 AudioInputStream* AudioManagerAndroid::MakeLinearInputStream( 200 AudioInputStream* AudioManagerAndroid::MakeLinearInputStream(
195 const AudioParameters& params, const std::string& device_id) { 201 const AudioParameters& params, const std::string& device_id) {
196 // TODO(henrika): add support for device selection if/when any client 202 // TODO(henrika): add support for device selection if/when any client
197 // needs it. 203 // needs it.
198 DLOG_IF(ERROR, !device_id.empty()) << "Not implemented!"; 204 DLOG_IF(ERROR, !device_id.empty()) << "Not implemented!";
199 DCHECK_EQ(AudioParameters::AUDIO_PCM_LINEAR, params.format()); 205 DCHECK_EQ(AudioParameters::AUDIO_PCM_LINEAR, params.format());
206 CreateAndInitOnAudioThread();
200 return new OpenSLESInputStream(this, params); 207 return new OpenSLESInputStream(this, params);
201 } 208 }
202 209
203 AudioInputStream* AudioManagerAndroid::MakeLowLatencyInputStream( 210 AudioInputStream* AudioManagerAndroid::MakeLowLatencyInputStream(
204 const AudioParameters& params, const std::string& device_id) { 211 const AudioParameters& params, const std::string& device_id) {
205 DCHECK_EQ(AudioParameters::AUDIO_PCM_LOW_LATENCY, params.format()); 212 DCHECK_EQ(AudioParameters::AUDIO_PCM_LOW_LATENCY, params.format());
206 DLOG_IF(ERROR, device_id.empty()) << "Invalid device ID!"; 213 DLOG_IF(ERROR, device_id.empty()) << "Invalid device ID!";
214 CreateAndInitOnAudioThread();
215
207 // Use the device ID to select the correct input device. 216 // Use the device ID to select the correct input device.
208 // Note that the input device is always associated with a certain output 217 // Note that the input device is always associated with a certain output
209 // device, i.e., this selection does also switch the output device. 218 // device, i.e., this selection does also switch the output device.
210 // All input and output streams will be affected by the device selection. 219 // All input and output streams will be affected by the device selection.
211 if (!SetAudioDevice(device_id)) { 220 if (!SetAudioDevice(device_id)) {
212 LOG(ERROR) << "Unable to select audio device!"; 221 LOG(ERROR) << "Unable to select audio device!";
213 return NULL; 222 return NULL;
214 } 223 }
215 224
216 if (params.effects() != AudioParameters::NO_EFFECTS) { 225 if (params.effects() != AudioParameters::NO_EFFECTS) {
217 // Platform effects can only be enabled through the AudioRecord path. 226 // Platform effects can only be enabled through the AudioRecord path.
218 // An effect should only have been requested here if recommended by 227 // An effect should only have been requested here if recommended by
219 // AudioManagerAndroid.shouldUse<Effect>. 228 // AudioManagerAndroid.shouldUse<Effect>.
220 // 229 //
221 // Creating this class requires Jelly Bean, which is already guaranteed by 230 // Creating this class requires Jelly Bean, which is already guaranteed by
222 // shouldUse<Effect>. Only DCHECK on that condition to allow tests to use 231 // shouldUse<Effect>. Only DCHECK on that condition to allow tests to use
223 // the effect settings as a way to select the input path. 232 // the effect settings as a way to select the input path.
224 DCHECK_GE(base::android::BuildInfo::GetInstance()->sdk_int(), 16); 233 DCHECK_GE(base::android::BuildInfo::GetInstance()->sdk_int(), 16);
225 DVLOG(1) << "Creating AudioRecordInputStream"; 234 DVLOG(1) << "Creating AudioRecordInputStream";
226 return new AudioRecordInputStream(this, params); 235 return new AudioRecordInputStream(this, params);
227 } 236 }
228 DVLOG(1) << "Creating OpenSLESInputStream"; 237 DVLOG(1) << "Creating OpenSLESInputStream";
229 return new OpenSLESInputStream(this, params); 238 return new OpenSLESInputStream(this, params);
230 } 239 }
231 240
232 int AudioManagerAndroid::GetOptimalOutputFrameSize(int sample_rate, 241 // static
233 int channels) { 242 bool AudioManagerAndroid::RegisterAudioManager(JNIEnv* env) {
234 if (IsAudioLowLatencySupported()) { 243 return RegisterNativesImpl(env);
235 return GetAudioLowLatencyOutputFrameSize(); 244 }
236 } else { 245
237 return std::max(kDefaultOutputBufferSize, 246 void AudioManagerAndroid::SetMute(JNIEnv* env, jobject obj, jboolean muted) {
238 Java_AudioManagerAndroid_getMinOutputFrameSize( 247 GetTaskRunner()->PostTask(
tommi (sloooow) - chröme 2014/02/17 13:24:55 On which thread is this method called? Shouldn't
henrika (OOO until Aug 14) 2014/02/17 15:14:41 It is a callback from Java and is not on the audio
239 base::android::AttachCurrentThread(), 248 FROM_HERE,
240 sample_rate, channels)); 249 base::Bind(
241 } 250 &AudioManagerAndroid::DoSetMuteOnAudioThread,
251 base::Unretained(this),
252 muted));
242 } 253 }
243 254
244 AudioParameters AudioManagerAndroid::GetPreferredOutputStreamParameters( 255 AudioParameters AudioManagerAndroid::GetPreferredOutputStreamParameters(
245 const std::string& output_device_id, 256 const std::string& output_device_id,
246 const AudioParameters& input_params) { 257 const AudioParameters& input_params) {
258 CreateAndInitOnAudioThread();
259
247 // TODO(tommi): Support |output_device_id|. 260 // TODO(tommi): Support |output_device_id|.
248 DLOG_IF(ERROR, !output_device_id.empty()) << "Not implemented!"; 261 DLOG_IF(ERROR, !output_device_id.empty()) << "Not implemented!";
249 ChannelLayout channel_layout = CHANNEL_LAYOUT_STEREO; 262 ChannelLayout channel_layout = CHANNEL_LAYOUT_STEREO;
250 int sample_rate = GetNativeOutputSampleRate(); 263 int sample_rate = GetNativeOutputSampleRate();
251 int buffer_size = GetOptimalOutputFrameSize(sample_rate, 2); 264 int buffer_size = GetOptimalOutputFrameSize(sample_rate, 2);
252 int bits_per_sample = 16; 265 int bits_per_sample = 16;
253 int input_channels = 0; 266 int input_channels = 0;
254 if (input_params.IsValid()) { 267 if (input_params.IsValid()) {
255 // Use the client's input parameters if they are valid. 268 // Use the client's input parameters if they are valid.
256 sample_rate = input_params.sample_rate(); 269 sample_rate = input_params.sample_rate();
257 bits_per_sample = input_params.bits_per_sample(); 270 bits_per_sample = input_params.bits_per_sample();
258 channel_layout = input_params.channel_layout(); 271 channel_layout = input_params.channel_layout();
259 input_channels = input_params.input_channels(); 272 input_channels = input_params.input_channels();
260 buffer_size = GetOptimalOutputFrameSize( 273 buffer_size = GetOptimalOutputFrameSize(
261 sample_rate, ChannelLayoutToChannelCount(channel_layout)); 274 sample_rate, ChannelLayoutToChannelCount(channel_layout));
262 } 275 }
263 276
264 int user_buffer_size = GetUserBufferSize(); 277 int user_buffer_size = GetUserBufferSize();
265 if (user_buffer_size) 278 if (user_buffer_size)
266 buffer_size = user_buffer_size; 279 buffer_size = user_buffer_size;
267 280
268 return AudioParameters( 281 return AudioParameters(
269 AudioParameters::AUDIO_PCM_LOW_LATENCY, channel_layout, input_channels, 282 AudioParameters::AUDIO_PCM_LOW_LATENCY, channel_layout, input_channels,
270 sample_rate, bits_per_sample, buffer_size, AudioParameters::NO_EFFECTS); 283 sample_rate, bits_per_sample, buffer_size, AudioParameters::NO_EFFECTS);
271 } 284 }
272 285
273 bool AudioManagerAndroid::HadNoAudioStreams() { 286 void AudioManagerAndroid::WillDestroyCurrentMessageLoop() {
287 CloseOnAudioThread();
288 }
289
290 void AudioManagerAndroid::CreateAndInitOnAudioThread() {
291 DCHECK(GetTaskRunner()->BelongsToCurrentThread());
292
293 // Ensure that we only create and initialize the Java part once.
294 if (!j_audio_manager_.is_null()) {
295 return;
296 }
297
298 JNIEnv* env = base::android::AttachCurrentThread();
299 CHECK(env);
300
301 // Create the Android audio manager on the audio thread.
302 DVLOG(2) << "Creating Java part of the audio manager";
303 j_audio_manager_.Reset(
304 Java_AudioManagerAndroid_createAudioManagerAndroid(
305 env,
306 base::android::GetApplicationContext(),
307 reinterpret_cast<intptr_t>(this)));
308
309 // Prepare the list of audio devices and register receivers for device
310 // notifications.
311 Init();
312
313 // Ensure that we are notified when the audio thread dies.
314 base::MessageLoop::current()->AddDestructionObserver(this);
315 }
316
317 void AudioManagerAndroid::CloseOnAudioThread() {
318 DCHECK(GetTaskRunner()->BelongsToCurrentThread());
319 if (j_audio_manager_.is_null())
320 return;
321 Close();
322 DVLOG(2) << "Destroying Java part of the audio manager";
323 j_audio_manager_.Reset();
324 }
325
326 bool AudioManagerAndroid::HasNoAudioStreams() {
274 return output_stream_count() == 0 && input_stream_count() == 0; 327 return output_stream_count() == 0 && input_stream_count() == 0;
275 } 328 }
276 329
277 // static
278 bool AudioManagerAndroid::RegisterAudioManager(JNIEnv* env) {
279 return RegisterNativesImpl(env);
280 }
281
282 void AudioManagerAndroid::Init() { 330 void AudioManagerAndroid::Init() {
283 Java_AudioManagerAndroid_init( 331 Java_AudioManagerAndroid_init(
284 base::android::AttachCurrentThread(), 332 base::android::AttachCurrentThread(),
285 j_audio_manager_.obj()); 333 j_audio_manager_.obj());
286 } 334 }
287 335
288 void AudioManagerAndroid::Close() { 336 void AudioManagerAndroid::Close() {
289 Java_AudioManagerAndroid_close( 337 Java_AudioManagerAndroid_close(
290 base::android::AttachCurrentThread(), 338 base::android::AttachCurrentThread(),
291 j_audio_manager_.obj()); 339 j_audio_manager_.obj());
292 } 340 }
293 341
294 void AudioManagerAndroid::SetMute(JNIEnv* env, jobject obj, jboolean muted) {
295 GetTaskRunner()->PostTask(
296 FROM_HERE,
297 base::Bind(
298 &AudioManagerAndroid::DoSetMuteOnAudioThread,
299 base::Unretained(this),
300 muted));
301 }
302
303 void AudioManagerAndroid::DoSetMuteOnAudioThread(bool muted) {
304 base::AutoLock lock(streams_lock_);
305 for (OutputStreams::iterator it = streams_.begin();
306 it != streams_.end(); ++it) {
307 (*it)->SetMute(muted);
308 }
309 }
310
311 void AudioManagerAndroid::SetCommunicationAudioModeOn(bool on) { 342 void AudioManagerAndroid::SetCommunicationAudioModeOn(bool on) {
312 Java_AudioManagerAndroid_setCommunicationAudioModeOn( 343 Java_AudioManagerAndroid_setCommunicationAudioModeOn(
313 base::android::AttachCurrentThread(), 344 base::android::AttachCurrentThread(),
314 j_audio_manager_.obj(), on); 345 j_audio_manager_.obj(), on);
315 } 346 }
316 347
317 bool AudioManagerAndroid::SetAudioDevice(const std::string& device_id) { 348 bool AudioManagerAndroid::SetAudioDevice(const std::string& device_id) {
318 JNIEnv* env = AttachCurrentThread(); 349 DCHECK(GetTaskRunner()->BelongsToCurrentThread());
319 350
320 // Send the unique device ID to the Java audio manager and make the 351 // Send the unique device ID to the Java audio manager and make the
321 // device switch. Provide an empty string to the Java audio manager 352 // device switch. Provide an empty string to the Java audio manager
322 // if the default device is selected. 353 // if the default device is selected.
354 JNIEnv* env = AttachCurrentThread();
355 CHECK(env);
323 ScopedJavaLocalRef<jstring> j_device_id = ConvertUTF8ToJavaString( 356 ScopedJavaLocalRef<jstring> j_device_id = ConvertUTF8ToJavaString(
324 env, 357 env,
325 device_id == AudioManagerBase::kDefaultDeviceId ? 358 device_id == AudioManagerBase::kDefaultDeviceId ?
326 std::string() : device_id); 359 std::string() : device_id);
327 return Java_AudioManagerAndroid_setDevice( 360 return Java_AudioManagerAndroid_setDevice(
328 env, j_audio_manager_.obj(), j_device_id.obj()); 361 env, j_audio_manager_.obj(), j_device_id.obj());
329 } 362 }
330 363
331 int AudioManagerAndroid::GetNativeOutputSampleRate() { 364 int AudioManagerAndroid::GetNativeOutputSampleRate() {
332 return Java_AudioManagerAndroid_getNativeOutputSampleRate( 365 return Java_AudioManagerAndroid_getNativeOutputSampleRate(
333 base::android::AttachCurrentThread(), 366 base::android::AttachCurrentThread(),
334 j_audio_manager_.obj()); 367 j_audio_manager_.obj());
335 } 368 }
336 369
337 bool AudioManagerAndroid::IsAudioLowLatencySupported() { 370 bool AudioManagerAndroid::IsAudioLowLatencySupported() {
338 return Java_AudioManagerAndroid_isAudioLowLatencySupported( 371 return Java_AudioManagerAndroid_isAudioLowLatencySupported(
339 base::android::AttachCurrentThread(), 372 base::android::AttachCurrentThread(),
340 j_audio_manager_.obj()); 373 j_audio_manager_.obj());
341 } 374 }
342 375
343 int AudioManagerAndroid::GetAudioLowLatencyOutputFrameSize() { 376 int AudioManagerAndroid::GetAudioLowLatencyOutputFrameSize() {
344 return Java_AudioManagerAndroid_getAudioLowLatencyOutputFrameSize( 377 return Java_AudioManagerAndroid_getAudioLowLatencyOutputFrameSize(
345 base::android::AttachCurrentThread(), 378 base::android::AttachCurrentThread(),
346 j_audio_manager_.obj()); 379 j_audio_manager_.obj());
347 } 380 }
348 381
382 int AudioManagerAndroid::GetOptimalOutputFrameSize(int sample_rate,
383 int channels) {
384 if (IsAudioLowLatencySupported()) {
385 return GetAudioLowLatencyOutputFrameSize();
386 } else {
387 return std::max(kDefaultOutputBufferSize,
388 Java_AudioManagerAndroid_getMinOutputFrameSize(
389 base::android::AttachCurrentThread(),
390 sample_rate, channels));
391 }
392 }
393
394 void AudioManagerAndroid::DoSetMuteOnAudioThread(bool muted) {
395 DCHECK(GetTaskRunner()->BelongsToCurrentThread());
396 base::AutoLock lock(streams_lock_);
397 for (OutputStreams::iterator it = streams_.begin();
tommi (sloooow) - chröme 2014/02/17 13:24:55 it is starting to look to me like streams_ is only
henrika (OOO until Aug 14) 2014/02/17 15:14:41 It is in fact on only touched on the audio thread
398 it != streams_.end(); ++it) {
399 (*it)->SetMute(muted);
400 }
401 }
402
349 } // namespace media 403 } // namespace media
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698