blob: 40b6f42309bd79cab87ee98ce077bd27d96cfba8 [file] [log] [blame]
Alan Viverette3da604b2020-06-10 18:34:39 +00001/*
2 * Copyright 2019 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16package com.android.server.audio;
17
18import android.annotation.NonNull;
19import android.bluetooth.BluetoothA2dp;
20import android.bluetooth.BluetoothDevice;
21import android.bluetooth.BluetoothHeadset;
22import android.bluetooth.BluetoothHearingAid;
23import android.bluetooth.BluetoothProfile;
24import android.content.ContentResolver;
25import android.content.Context;
26import android.content.Intent;
27import android.media.AudioDeviceAttributes;
28import android.media.AudioRoutesInfo;
29import android.media.AudioSystem;
30import android.media.IAudioRoutesObserver;
31import android.media.IStrategyPreferredDeviceDispatcher;
32import android.media.MediaMetrics;
33import android.os.Binder;
34import android.os.Handler;
35import android.os.IBinder;
36import android.os.Looper;
37import android.os.Message;
38import android.os.PowerManager;
39import android.os.SystemClock;
40import android.util.Log;
41import android.util.PrintWriterPrinter;
42
43import com.android.internal.annotations.GuardedBy;
44
45import java.io.PrintWriter;
46
47/** @hide */
48/*package*/ final class AudioDeviceBroker {
49
50 private static final String TAG = "AS.AudioDeviceBroker";
51
52 private static final long BROKER_WAKELOCK_TIMEOUT_MS = 5000; //5s
53
54 /*package*/ static final int BTA2DP_DOCK_TIMEOUT_MS = 8000;
55 // Timeout for connection to bluetooth headset service
56 /*package*/ static final int BT_HEADSET_CNCT_TIMEOUT_MS = 3000;
57
58 private final @NonNull AudioService mAudioService;
59 private final @NonNull Context mContext;
60
61 /** Forced device usage for communications sent to AudioSystem */
62 private int mForcedUseForComm;
63 /**
64 * Externally reported force device usage state returned by getters: always consistent
65 * with requests by setters */
66 private int mForcedUseForCommExt;
67
68 // Manages all connected devices, only ever accessed on the message loop
69 private final AudioDeviceInventory mDeviceInventory;
70 // Manages notifications to BT service
71 private final BtHelper mBtHelper;
72 // Adapter for system_server-reserved operations
73 private final SystemServerAdapter mSystemServer;
74
75
76 //-------------------------------------------------------------------
77 // we use a different lock than mDeviceStateLock so as not to create
78 // lock contention between enqueueing a message and handling them
79 private static final Object sLastDeviceConnectionMsgTimeLock = new Object();
80 @GuardedBy("sLastDeviceConnectionMsgTimeLock")
81 private static long sLastDeviceConnectMsgTime = 0;
82
83 // General lock to be taken whenever the state of the audio devices is to be checked or changed
84 private final Object mDeviceStateLock = new Object();
85
86 // Request to override default use of A2DP for media.
87 @GuardedBy("mDeviceStateLock")
88 private boolean mBluetoothA2dpEnabled;
89
90 // lock always taken when accessing AudioService.mSetModeDeathHandlers
91 // TODO do not "share" the lock between AudioService and BtHelpr, see b/123769055
92 /*package*/ final Object mSetModeLock = new Object();
93
94 //-------------------------------------------------------------------
95 /*package*/ AudioDeviceBroker(@NonNull Context context, @NonNull AudioService service) {
96 mContext = context;
97 mAudioService = service;
98 mBtHelper = new BtHelper(this);
99 mDeviceInventory = new AudioDeviceInventory(this);
100 mSystemServer = SystemServerAdapter.getDefaultAdapter(mContext);
101
102 init();
103 }
104
105 /** for test purposes only, inject AudioDeviceInventory and adapter for operations running
106 * in system_server */
107 AudioDeviceBroker(@NonNull Context context, @NonNull AudioService service,
108 @NonNull AudioDeviceInventory mockDeviceInventory,
109 @NonNull SystemServerAdapter mockSystemServer) {
110 mContext = context;
111 mAudioService = service;
112 mBtHelper = new BtHelper(this);
113 mDeviceInventory = mockDeviceInventory;
114 mSystemServer = mockSystemServer;
115
116 init();
117 }
118
119 private void init() {
120 setupMessaging(mContext);
121
122 mForcedUseForComm = AudioSystem.FORCE_NONE;
123 mForcedUseForCommExt = mForcedUseForComm;
124 }
125
126 /*package*/ Context getContext() {
127 return mContext;
128 }
129
130 //---------------------------------------------------------------------
131 // Communication from AudioService
132 // All methods are asynchronous and never block
133 // All permission checks are done in AudioService, all incoming calls are considered "safe"
134 // All post* methods are asynchronous
135
136 /*package*/ void onSystemReady() {
137 synchronized (mSetModeLock) {
138 synchronized (mDeviceStateLock) {
139 mBtHelper.onSystemReady();
140 }
141 }
142 }
143
144 /*package*/ void onAudioServerDied() {
145 // Restore forced usage for communications and record
146 synchronized (mDeviceStateLock) {
147 AudioSystem.setParameters(
148 "BT_SCO=" + (mForcedUseForComm == AudioSystem.FORCE_BT_SCO ? "on" : "off"));
149 onSetForceUse(AudioSystem.FOR_COMMUNICATION, mForcedUseForComm, "onAudioServerDied");
150 onSetForceUse(AudioSystem.FOR_RECORD, mForcedUseForComm, "onAudioServerDied");
151 }
152 // restore devices
153 sendMsgNoDelay(MSG_RESTORE_DEVICES, SENDMSG_REPLACE);
154 }
155
156 /*package*/ void setForceUse_Async(int useCase, int config, String eventSource) {
157 sendIILMsgNoDelay(MSG_IIL_SET_FORCE_USE, SENDMSG_QUEUE,
158 useCase, config, eventSource);
159 }
160
161 /*package*/ void toggleHdmiIfConnected_Async() {
162 sendMsgNoDelay(MSG_TOGGLE_HDMI, SENDMSG_QUEUE);
163 }
164
165 /*package*/ void disconnectAllBluetoothProfiles() {
166 synchronized (mDeviceStateLock) {
167 mBtHelper.disconnectAllBluetoothProfiles();
168 }
169 }
170
171 /**
172 * Handle BluetoothHeadset intents where the action is one of
173 * {@link BluetoothHeadset#ACTION_ACTIVE_DEVICE_CHANGED} or
174 * {@link BluetoothHeadset#ACTION_AUDIO_STATE_CHANGED}.
175 * @param intent
176 */
177 /*package*/ void receiveBtEvent(@NonNull Intent intent) {
178 synchronized (mSetModeLock) {
179 synchronized (mDeviceStateLock) {
180 mBtHelper.receiveBtEvent(intent);
181 }
182 }
183 }
184
185 /*package*/ void setBluetoothA2dpOn_Async(boolean on, String source) {
186 synchronized (mDeviceStateLock) {
187 if (mBluetoothA2dpEnabled == on) {
188 return;
189 }
190 mBluetoothA2dpEnabled = on;
191 mBrokerHandler.removeMessages(MSG_IIL_SET_FORCE_BT_A2DP_USE);
192 sendIILMsgNoDelay(MSG_IIL_SET_FORCE_BT_A2DP_USE, SENDMSG_QUEUE,
193 AudioSystem.FOR_MEDIA,
194 mBluetoothA2dpEnabled ? AudioSystem.FORCE_NONE : AudioSystem.FORCE_NO_BT_A2DP,
195 source);
196 }
197 }
198
199 /**
200 * Turns speakerphone on/off
201 * @param on
202 * @param eventSource for logging purposes
203 * @return true if speakerphone state changed
204 */
205 /*package*/ boolean setSpeakerphoneOn(boolean on, String eventSource) {
206 synchronized (mDeviceStateLock) {
207 final boolean wasOn = isSpeakerphoneOn();
208 if (on) {
209 if (mForcedUseForComm == AudioSystem.FORCE_BT_SCO) {
210 setForceUse_Async(AudioSystem.FOR_RECORD, AudioSystem.FORCE_NONE, eventSource);
211 }
212 mForcedUseForComm = AudioSystem.FORCE_SPEAKER;
213 } else if (mForcedUseForComm == AudioSystem.FORCE_SPEAKER) {
214 mForcedUseForComm = AudioSystem.FORCE_NONE;
215 }
216
217 mForcedUseForCommExt = mForcedUseForComm;
218 setForceUse_Async(AudioSystem.FOR_COMMUNICATION, mForcedUseForComm, eventSource);
219 return (wasOn != isSpeakerphoneOn());
220 }
221 }
222
223 /*package*/ boolean isSpeakerphoneOn() {
224 synchronized (mDeviceStateLock) {
225 return (mForcedUseForCommExt == AudioSystem.FORCE_SPEAKER);
226 }
227 }
228
229 /*package*/ void setWiredDeviceConnectionState(int type,
230 @AudioService.ConnectionState int state, String address, String name,
231 String caller) {
232 //TODO move logging here just like in setBluetooth* methods
233 synchronized (mDeviceStateLock) {
234 mDeviceInventory.setWiredDeviceConnectionState(type, state, address, name, caller);
235 }
236 }
237
238 private static final class BtDeviceConnectionInfo {
239 final @NonNull BluetoothDevice mDevice;
240 final @AudioService.BtProfileConnectionState int mState;
241 final int mProfile;
242 final boolean mSupprNoisy;
243 final int mVolume;
244
245 BtDeviceConnectionInfo(@NonNull BluetoothDevice device,
246 @AudioService.BtProfileConnectionState int state,
247 int profile, boolean suppressNoisyIntent, int vol) {
248 mDevice = device;
249 mState = state;
250 mProfile = profile;
251 mSupprNoisy = suppressNoisyIntent;
252 mVolume = vol;
253 }
254
255 // redefine equality op so we can match messages intended for this device
256 @Override
257 public boolean equals(Object o) {
258 if (o == null) {
259 return false;
260 }
261 if (this == o) {
262 return true;
263 }
264 if (o instanceof BtDeviceConnectionInfo) {
265 return mDevice.equals(((BtDeviceConnectionInfo) o).mDevice);
266 }
267 return false;
268 }
269
270 @Override
271 public String toString() {
272 return "BtDeviceConnectionInfo dev=" + mDevice.toString();
273 }
274 }
275
276
277 /*package*/ void postBluetoothA2dpDeviceConnectionStateSuppressNoisyIntent(
278 @NonNull BluetoothDevice device, @AudioService.BtProfileConnectionState int state,
279 int profile, boolean suppressNoisyIntent, int a2dpVolume) {
280 final BtDeviceConnectionInfo info = new BtDeviceConnectionInfo(device, state, profile,
281 suppressNoisyIntent, a2dpVolume);
282
283 // operations of removing and posting messages related to A2DP device state change must be
284 // mutually exclusive
285 synchronized (mDeviceStateLock) {
286 // when receiving a request to change the connection state of a device, this last
287 // request is the source of truth, so cancel all previous requests that are already in
288 // the handler
289 removeScheduledA2dpEvents(device);
290
291 sendLMsgNoDelay(
292 state == BluetoothProfile.STATE_CONNECTED
293 ? MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_CONNECTION
294 : MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_DISCONNECTION,
295 SENDMSG_QUEUE, info);
296 }
297 }
298
299 /** remove all previously scheduled connection and state change events for the given device */
300 @GuardedBy("mDeviceStateLock")
301 private void removeScheduledA2dpEvents(@NonNull BluetoothDevice device) {
302 mBrokerHandler.removeEqualMessages(MSG_L_A2DP_DEVICE_CONFIG_CHANGE, device);
303
304 final BtDeviceConnectionInfo connectionInfoToRemove = new BtDeviceConnectionInfo(device,
305 // the next parameters of the constructor will be ignored when finding the message
306 // to remove as the equality of the message's object is tested on the device itself
307 // (see BtDeviceConnectionInfo.equals() method override)
308 BluetoothProfile.STATE_CONNECTED, 0, false, -1);
309 mBrokerHandler.removeEqualMessages(MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_DISCONNECTION,
310 connectionInfoToRemove);
311 mBrokerHandler.removeEqualMessages(MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_CONNECTION,
312 connectionInfoToRemove);
313
314 final BtHelper.BluetoothA2dpDeviceInfo devInfoToRemove =
315 new BtHelper.BluetoothA2dpDeviceInfo(device);
316 mBrokerHandler.removeEqualMessages(MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_CONNECTED,
317 devInfoToRemove);
318 mBrokerHandler.removeEqualMessages(MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_DISCONNECTED,
319 devInfoToRemove);
320 mBrokerHandler.removeEqualMessages(MSG_L_A2DP_ACTIVE_DEVICE_CHANGE,
321 devInfoToRemove);
322 }
323
324 private static final class HearingAidDeviceConnectionInfo {
325 final @NonNull BluetoothDevice mDevice;
326 final @AudioService.BtProfileConnectionState int mState;
327 final boolean mSupprNoisy;
328 final int mMusicDevice;
329 final @NonNull String mEventSource;
330
331 HearingAidDeviceConnectionInfo(@NonNull BluetoothDevice device,
332 @AudioService.BtProfileConnectionState int state,
333 boolean suppressNoisyIntent, int musicDevice, @NonNull String eventSource) {
334 mDevice = device;
335 mState = state;
336 mSupprNoisy = suppressNoisyIntent;
337 mMusicDevice = musicDevice;
338 mEventSource = eventSource;
339 }
340 }
341
342 /*package*/ void postBluetoothHearingAidDeviceConnectionState(
343 @NonNull BluetoothDevice device, @AudioService.BtProfileConnectionState int state,
344 boolean suppressNoisyIntent, int musicDevice, @NonNull String eventSource) {
345 final HearingAidDeviceConnectionInfo info = new HearingAidDeviceConnectionInfo(
346 device, state, suppressNoisyIntent, musicDevice, eventSource);
347 sendLMsgNoDelay(MSG_L_HEARING_AID_DEVICE_CONNECTION_CHANGE_EXT, SENDMSG_QUEUE, info);
348 }
349
350 // never called by system components
351 /*package*/ void setBluetoothScoOnByApp(boolean on) {
352 synchronized (mDeviceStateLock) {
353 mForcedUseForCommExt = on ? AudioSystem.FORCE_BT_SCO : AudioSystem.FORCE_NONE;
354 }
355 }
356
357 /*package*/ boolean isBluetoothScoOnForApp() {
358 synchronized (mDeviceStateLock) {
359 return mForcedUseForCommExt == AudioSystem.FORCE_BT_SCO;
360 }
361 }
362
363 /*package*/ void setBluetoothScoOn(boolean on, String eventSource) {
364 //Log.i(TAG, "setBluetoothScoOn: " + on + " " + eventSource);
365 synchronized (mDeviceStateLock) {
366 if (on) {
367 // do not accept SCO ON if SCO audio is not connected
368 if (!mBtHelper.isBluetoothScoOn()) {
369 mForcedUseForCommExt = AudioSystem.FORCE_BT_SCO;
370 return;
371 }
372 mForcedUseForComm = AudioSystem.FORCE_BT_SCO;
373 } else if (mForcedUseForComm == AudioSystem.FORCE_BT_SCO) {
374 mForcedUseForComm = AudioSystem.FORCE_NONE;
375 }
376 mForcedUseForCommExt = mForcedUseForComm;
377 AudioSystem.setParameters("BT_SCO=" + (on ? "on" : "off"));
378 sendIILMsgNoDelay(MSG_IIL_SET_FORCE_USE, SENDMSG_QUEUE,
379 AudioSystem.FOR_COMMUNICATION, mForcedUseForComm, eventSource);
380 sendIILMsgNoDelay(MSG_IIL_SET_FORCE_USE, SENDMSG_QUEUE,
381 AudioSystem.FOR_RECORD, mForcedUseForComm, eventSource);
382 }
383 // Un-mute ringtone stream volume
384 mAudioService.postUpdateRingerModeServiceInt();
385 }
386
387 /*package*/ AudioRoutesInfo startWatchingRoutes(IAudioRoutesObserver observer) {
388 synchronized (mDeviceStateLock) {
389 return mDeviceInventory.startWatchingRoutes(observer);
390 }
391 }
392
393 /*package*/ AudioRoutesInfo getCurAudioRoutes() {
394 synchronized (mDeviceStateLock) {
395 return mDeviceInventory.getCurAudioRoutes();
396 }
397 }
398
399 /*package*/ boolean isAvrcpAbsoluteVolumeSupported() {
400 synchronized (mDeviceStateLock) {
401 return mBtHelper.isAvrcpAbsoluteVolumeSupported();
402 }
403 }
404
405 /*package*/ boolean isBluetoothA2dpOn() {
406 synchronized (mDeviceStateLock) {
407 return mBluetoothA2dpEnabled;
408 }
409 }
410
411 /*package*/ void postSetAvrcpAbsoluteVolumeIndex(int index) {
412 sendIMsgNoDelay(MSG_I_SET_AVRCP_ABSOLUTE_VOLUME, SENDMSG_REPLACE, index);
413 }
414
415 /*package*/ void postSetHearingAidVolumeIndex(int index, int streamType) {
416 sendIIMsgNoDelay(MSG_II_SET_HEARING_AID_VOLUME, SENDMSG_REPLACE, index, streamType);
417 }
418
419 /*package*/ void postDisconnectBluetoothSco(int exceptPid) {
420 sendIMsgNoDelay(MSG_I_DISCONNECT_BT_SCO, SENDMSG_REPLACE, exceptPid);
421 }
422
423 /*package*/ void postBluetoothA2dpDeviceConfigChange(@NonNull BluetoothDevice device) {
424 sendLMsgNoDelay(MSG_L_A2DP_DEVICE_CONFIG_CHANGE, SENDMSG_QUEUE, device);
425 }
426
427 @GuardedBy("mSetModeLock")
428 /*package*/ void startBluetoothScoForClient_Sync(IBinder cb, int scoAudioMode,
429 @NonNull String eventSource) {
430 synchronized (mDeviceStateLock) {
431 mBtHelper.startBluetoothScoForClient(cb, scoAudioMode, eventSource);
432 }
433 }
434
435 @GuardedBy("mSetModeLock")
436 /*package*/ void stopBluetoothScoForClient_Sync(IBinder cb, @NonNull String eventSource) {
437 synchronized (mDeviceStateLock) {
438 mBtHelper.stopBluetoothScoForClient(cb, eventSource);
439 }
440 }
441
442 /*package*/ int setPreferredDeviceForStrategySync(int strategy,
443 @NonNull AudioDeviceAttributes device) {
444 return mDeviceInventory.setPreferredDeviceForStrategySync(strategy, device);
445 }
446
447 /*package*/ int removePreferredDeviceForStrategySync(int strategy) {
448 return mDeviceInventory.removePreferredDeviceForStrategySync(strategy);
449 }
450
451 /*package*/ void registerStrategyPreferredDeviceDispatcher(
452 @NonNull IStrategyPreferredDeviceDispatcher dispatcher) {
453 mDeviceInventory.registerStrategyPreferredDeviceDispatcher(dispatcher);
454 }
455
456 /*package*/ void unregisterStrategyPreferredDeviceDispatcher(
457 @NonNull IStrategyPreferredDeviceDispatcher dispatcher) {
458 mDeviceInventory.unregisterStrategyPreferredDeviceDispatcher(dispatcher);
459 }
460
461 //---------------------------------------------------------------------
462 // Communication with (to) AudioService
463 //TODO check whether the AudioService methods are candidates to move here
464 /*package*/ void postAccessoryPlugMediaUnmute(int device) {
465 mAudioService.postAccessoryPlugMediaUnmute(device);
466 }
467
468 /*package*/ int getVssVolumeForDevice(int streamType, int device) {
469 return mAudioService.getVssVolumeForDevice(streamType, device);
470 }
471
472 /*package*/ int getModeOwnerPid() {
473 return mAudioService.getModeOwnerPid();
474 }
475
476 /*package*/ int getDeviceForStream(int streamType) {
477 return mAudioService.getDeviceForStream(streamType);
478 }
479
480 /*package*/ void postApplyVolumeOnDevice(int streamType, int device, String caller) {
481 mAudioService.postApplyVolumeOnDevice(streamType, device, caller);
482 }
483
484 /*package*/ void postSetVolumeIndexOnDevice(int streamType, int vssVolIndex, int device,
485 String caller) {
486 mAudioService.postSetVolumeIndexOnDevice(streamType, vssVolIndex, device, caller);
487 }
488
489 /*packages*/ void postObserveDevicesForAllStreams() {
490 mAudioService.postObserveDevicesForAllStreams();
491 }
492
493 /*package*/ boolean isInCommunication() {
494 return mAudioService.isInCommunication();
495 }
496
497 /*package*/ boolean hasMediaDynamicPolicy() {
498 return mAudioService.hasMediaDynamicPolicy();
499 }
500
501 /*package*/ ContentResolver getContentResolver() {
502 return mAudioService.getContentResolver();
503 }
504
505 /*package*/ void checkMusicActive(int deviceType, String caller) {
506 mAudioService.checkMusicActive(deviceType, caller);
507 }
508
509 /*package*/ void checkVolumeCecOnHdmiConnection(
510 @AudioService.ConnectionState int state, String caller) {
511 mAudioService.postCheckVolumeCecOnHdmiConnection(state, caller);
512 }
513
514 /*package*/ boolean hasAudioFocusUsers() {
515 return mAudioService.hasAudioFocusUsers();
516 }
517
518 //---------------------------------------------------------------------
519 // Message handling on behalf of helper classes
520 /*package*/ void postBroadcastScoConnectionState(int state) {
521 sendIMsgNoDelay(MSG_I_BROADCAST_BT_CONNECTION_STATE, SENDMSG_QUEUE, state);
522 }
523
524 /*package*/ void postBroadcastBecomingNoisy() {
525 sendMsgNoDelay(MSG_BROADCAST_AUDIO_BECOMING_NOISY, SENDMSG_REPLACE);
526 }
527
528 @GuardedBy("mDeviceStateLock")
529 /*package*/ void postA2dpSinkConnection(@AudioService.BtProfileConnectionState int state,
530 @NonNull BtHelper.BluetoothA2dpDeviceInfo btDeviceInfo, int delay) {
531 sendILMsg(state == BluetoothA2dp.STATE_CONNECTED
532 ? MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_CONNECTED
533 : MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_DISCONNECTED,
534 SENDMSG_QUEUE,
535 state, btDeviceInfo, delay);
536 }
537
538 /*package*/ void postA2dpSourceConnection(@AudioService.BtProfileConnectionState int state,
539 @NonNull BtHelper.BluetoothA2dpDeviceInfo btDeviceInfo, int delay) {
540 sendILMsg(MSG_IL_SET_A2DP_SOURCE_CONNECTION_STATE, SENDMSG_QUEUE,
541 state, btDeviceInfo, delay);
542 }
543
544 /*package*/ void postSetWiredDeviceConnectionState(
545 AudioDeviceInventory.WiredDeviceConnectionState connectionState, int delay) {
546 sendLMsg(MSG_L_SET_WIRED_DEVICE_CONNECTION_STATE, SENDMSG_QUEUE, connectionState, delay);
547 }
548
549 /*package*/ void postSetHearingAidConnectionState(
550 @AudioService.BtProfileConnectionState int state,
551 @NonNull BluetoothDevice device, int delay) {
552 sendILMsg(MSG_IL_SET_HEARING_AID_CONNECTION_STATE, SENDMSG_QUEUE,
553 state,
554 device,
555 delay);
556 }
557
558 /*package*/ void postDisconnectA2dp() {
559 sendMsgNoDelay(MSG_DISCONNECT_A2DP, SENDMSG_QUEUE);
560 }
561
562 /*package*/ void postDisconnectA2dpSink() {
563 sendMsgNoDelay(MSG_DISCONNECT_A2DP_SINK, SENDMSG_QUEUE);
564 }
565
566 /*package*/ void postDisconnectHearingAid() {
567 sendMsgNoDelay(MSG_DISCONNECT_BT_HEARING_AID, SENDMSG_QUEUE);
568 }
569
570 /*package*/ void postDisconnectHeadset() {
571 sendMsgNoDelay(MSG_DISCONNECT_BT_HEADSET, SENDMSG_QUEUE);
572 }
573
574 /*package*/ void postBtA2dpProfileConnected(BluetoothA2dp a2dpProfile) {
575 sendLMsgNoDelay(MSG_L_BT_SERVICE_CONNECTED_PROFILE_A2DP, SENDMSG_QUEUE, a2dpProfile);
576 }
577
578 /*package*/ void postBtA2dpSinkProfileConnected(BluetoothProfile profile) {
579 sendLMsgNoDelay(MSG_L_BT_SERVICE_CONNECTED_PROFILE_A2DP_SINK, SENDMSG_QUEUE, profile);
580 }
581
582 /*package*/ void postBtHeasetProfileConnected(BluetoothHeadset headsetProfile) {
583 sendLMsgNoDelay(MSG_L_BT_SERVICE_CONNECTED_PROFILE_HEADSET, SENDMSG_QUEUE, headsetProfile);
584 }
585
586 /*package*/ void postBtHearingAidProfileConnected(BluetoothHearingAid hearingAidProfile) {
587 sendLMsgNoDelay(MSG_L_BT_SERVICE_CONNECTED_PROFILE_HEARING_AID, SENDMSG_QUEUE,
588 hearingAidProfile);
589 }
590
591 /*package*/ void postScoClientDied(Object obj) {
592 sendLMsgNoDelay(MSG_L_SCOCLIENT_DIED, SENDMSG_QUEUE, obj);
593 }
594
595 /*package*/ void postSaveSetPreferredDeviceForStrategy(int strategy,
596 AudioDeviceAttributes device)
597 {
598 sendILMsgNoDelay(MSG_IL_SAVE_PREF_DEVICE_FOR_STRATEGY, SENDMSG_QUEUE, strategy, device);
599 }
600
601 /*package*/ void postSaveRemovePreferredDeviceForStrategy(int strategy) {
602 sendIMsgNoDelay(MSG_I_SAVE_REMOVE_PREF_DEVICE_FOR_STRATEGY, SENDMSG_QUEUE, strategy);
603 }
604
605 //---------------------------------------------------------------------
606 // Method forwarding between the helper classes (BtHelper, AudioDeviceInventory)
607 // only call from a "handle"* method or "on"* method
608
609 // Handles request to override default use of A2DP for media.
610 //@GuardedBy("mConnectedDevices")
611 /*package*/ void setBluetoothA2dpOnInt(boolean on, String source) {
612 // for logging only
613 final String eventSource = new StringBuilder("setBluetoothA2dpOn(").append(on)
614 .append(") from u/pid:").append(Binder.getCallingUid()).append("/")
615 .append(Binder.getCallingPid()).append(" src:").append(source).toString();
616
617 synchronized (mDeviceStateLock) {
618 mBluetoothA2dpEnabled = on;
619 mBrokerHandler.removeMessages(MSG_IIL_SET_FORCE_BT_A2DP_USE);
620 onSetForceUse(
621 AudioSystem.FOR_MEDIA,
622 mBluetoothA2dpEnabled ? AudioSystem.FORCE_NONE : AudioSystem.FORCE_NO_BT_A2DP,
623 eventSource);
624 }
625 }
626
627 /*package*/ boolean handleDeviceConnection(boolean connect, int device, String address,
628 String deviceName) {
629 synchronized (mDeviceStateLock) {
630 return mDeviceInventory.handleDeviceConnection(connect, device, address, deviceName);
631 }
632 }
633
634 /*package*/ void postSetA2dpSourceConnectionState(@BluetoothProfile.BtProfileState int state,
635 @NonNull BtHelper.BluetoothA2dpDeviceInfo btDeviceInfo) {
636 final int intState = (state == BluetoothA2dp.STATE_CONNECTED) ? 1 : 0;
637 sendILMsgNoDelay(MSG_IL_SET_A2DP_SOURCE_CONNECTION_STATE, SENDMSG_QUEUE, state,
638 btDeviceInfo);
639 }
640
641 /*package*/ void handleFailureToConnectToBtHeadsetService(int delay) {
642 sendMsg(MSG_BT_HEADSET_CNCT_FAILED, SENDMSG_REPLACE, delay);
643 }
644
645 /*package*/ void handleCancelFailureToConnectToBtHeadsetService() {
646 mBrokerHandler.removeMessages(MSG_BT_HEADSET_CNCT_FAILED);
647 }
648
649 /*package*/ void postReportNewRoutes() {
650 sendMsgNoDelay(MSG_REPORT_NEW_ROUTES, SENDMSG_NOOP);
651 }
652
653 /*package*/ void postA2dpActiveDeviceChange(
654 @NonNull BtHelper.BluetoothA2dpDeviceInfo btDeviceInfo) {
655 sendLMsgNoDelay(MSG_L_A2DP_ACTIVE_DEVICE_CHANGE, SENDMSG_QUEUE, btDeviceInfo);
656 }
657
658 // must be called synchronized on mConnectedDevices
659 /*package*/ boolean hasScheduledA2dpSinkConnectionState(BluetoothDevice btDevice) {
660 final BtHelper.BluetoothA2dpDeviceInfo devInfoToCheck =
661 new BtHelper.BluetoothA2dpDeviceInfo(btDevice);
662 return (mBrokerHandler.hasEqualMessages(
663 MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_CONNECTED, devInfoToCheck)
664 || mBrokerHandler.hasEqualMessages(
665 MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_DISCONNECTED, devInfoToCheck));
666 }
667
668 /*package*/ void setA2dpTimeout(String address, int a2dpCodec, int delayMs) {
669 sendILMsg(MSG_IL_BTA2DP_TIMEOUT, SENDMSG_QUEUE, a2dpCodec, address, delayMs);
670 }
671
672 /*package*/ void setAvrcpAbsoluteVolumeSupported(boolean supported) {
673 synchronized (mDeviceStateLock) {
674 mBtHelper.setAvrcpAbsoluteVolumeSupported(supported);
675 }
676 }
677
678 /*package*/ boolean getBluetoothA2dpEnabled() {
679 synchronized (mDeviceStateLock) {
680 return mBluetoothA2dpEnabled;
681 }
682 }
683
684 /*package*/ int getA2dpCodec(@NonNull BluetoothDevice device) {
685 synchronized (mDeviceStateLock) {
686 return mBtHelper.getA2dpCodec(device);
687 }
688 }
689
690 /*package*/ void dump(PrintWriter pw, String prefix) {
691 if (mBrokerHandler != null) {
692 pw.println(prefix + "Message handler (watch for unhandled messages):");
693 mBrokerHandler.dump(new PrintWriterPrinter(pw), prefix + " ");
694 } else {
695 pw.println("Message handler is null");
696 }
697 mDeviceInventory.dump(pw, prefix);
698 }
699
700 //---------------------------------------------------------------------
701 // Internal handling of messages
702 // These methods are ALL synchronous, in response to message handling in BrokerHandler
703 // Blocking in any of those will block the message queue
704
705 private void onSetForceUse(int useCase, int config, String eventSource) {
706 if (useCase == AudioSystem.FOR_MEDIA) {
707 postReportNewRoutes();
708 }
709 AudioService.sForceUseLogger.log(
710 new AudioServiceEvents.ForceUseEvent(useCase, config, eventSource));
711 new MediaMetrics.Item(MediaMetrics.Name.AUDIO_FORCE_USE + MediaMetrics.SEPARATOR
712 + AudioSystem.forceUseUsageToString(useCase))
713 .set(MediaMetrics.Property.EVENT, "onSetForceUse")
714 .set(MediaMetrics.Property.FORCE_USE_DUE_TO, eventSource)
715 .set(MediaMetrics.Property.FORCE_USE_MODE,
716 AudioSystem.forceUseConfigToString(config))
717 .record();
718 AudioSystem.setForceUse(useCase, config);
719 }
720
721 private void onSendBecomingNoisyIntent() {
722 AudioService.sDeviceLogger.log((new AudioEventLogger.StringEvent(
723 "broadcast ACTION_AUDIO_BECOMING_NOISY")).printLog(TAG));
724 mSystemServer.sendDeviceBecomingNoisyIntent();
725 }
726
727 //---------------------------------------------------------------------
728 // Message handling
729 private BrokerHandler mBrokerHandler;
730 private BrokerThread mBrokerThread;
731 private PowerManager.WakeLock mBrokerEventWakeLock;
732
733 private void setupMessaging(Context ctxt) {
734 final PowerManager pm = (PowerManager) ctxt.getSystemService(Context.POWER_SERVICE);
735 mBrokerEventWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK,
736 "handleAudioDeviceEvent");
737 mBrokerThread = new BrokerThread();
738 mBrokerThread.start();
739 waitForBrokerHandlerCreation();
740 }
741
742 private void waitForBrokerHandlerCreation() {
743 synchronized (this) {
744 while (mBrokerHandler == null) {
745 try {
746 wait();
747 } catch (InterruptedException e) {
748 Log.e(TAG, "Interruption while waiting on BrokerHandler");
749 }
750 }
751 }
752 }
753
754 /** Class that handles the device broker's message queue */
755 private class BrokerThread extends Thread {
756 BrokerThread() {
757 super("AudioDeviceBroker");
758 }
759
760 @Override
761 public void run() {
762 // Set this thread up so the handler will work on it
763 Looper.prepare();
764
765 synchronized (AudioDeviceBroker.this) {
766 mBrokerHandler = new BrokerHandler();
767
768 // Notify that the handler has been created
769 AudioDeviceBroker.this.notify();
770 }
771
772 Looper.loop();
773 }
774 }
775
776 /** Class that handles the message queue */
777 private class BrokerHandler extends Handler {
778
779 @Override
780 public void handleMessage(Message msg) {
781 switch (msg.what) {
782 case MSG_RESTORE_DEVICES:
783 synchronized (mDeviceStateLock) {
784 mDeviceInventory.onRestoreDevices();
785 mBtHelper.onAudioServerDiedRestoreA2dp();
786 }
787 break;
788 case MSG_L_SET_WIRED_DEVICE_CONNECTION_STATE:
789 synchronized (mDeviceStateLock) {
790 mDeviceInventory.onSetWiredDeviceConnectionState(
791 (AudioDeviceInventory.WiredDeviceConnectionState) msg.obj);
792 }
793 break;
794 case MSG_I_BROADCAST_BT_CONNECTION_STATE:
795 synchronized (mDeviceStateLock) {
796 mBtHelper.onBroadcastScoConnectionState(msg.arg1);
797 }
798 break;
799 case MSG_IIL_SET_FORCE_USE: // intended fall-through
800 case MSG_IIL_SET_FORCE_BT_A2DP_USE:
801 onSetForceUse(msg.arg1, msg.arg2, (String) msg.obj);
802 break;
803 case MSG_REPORT_NEW_ROUTES:
804 synchronized (mDeviceStateLock) {
805 mDeviceInventory.onReportNewRoutes();
806 }
807 break;
808 case MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_CONNECTED:
809 case MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_DISCONNECTED:
810 synchronized (mDeviceStateLock) {
811 mDeviceInventory.onSetA2dpSinkConnectionState(
812 (BtHelper.BluetoothA2dpDeviceInfo) msg.obj, msg.arg1);
813 }
814 break;
815 case MSG_IL_SET_A2DP_SOURCE_CONNECTION_STATE:
816 synchronized (mDeviceStateLock) {
817 mDeviceInventory.onSetA2dpSourceConnectionState(
818 (BtHelper.BluetoothA2dpDeviceInfo) msg.obj, msg.arg1);
819 }
820 break;
821 case MSG_IL_SET_HEARING_AID_CONNECTION_STATE:
822 synchronized (mDeviceStateLock) {
823 mDeviceInventory.onSetHearingAidConnectionState(
824 (BluetoothDevice) msg.obj, msg.arg1,
825 mAudioService.getHearingAidStreamType());
826 }
827 break;
828 case MSG_BT_HEADSET_CNCT_FAILED:
829 synchronized (mSetModeLock) {
830 synchronized (mDeviceStateLock) {
831 mBtHelper.resetBluetoothSco();
832 }
833 }
834 break;
835 case MSG_IL_BTA2DP_TIMEOUT:
836 // msg.obj == address of BTA2DP device
837 synchronized (mDeviceStateLock) {
838 mDeviceInventory.onMakeA2dpDeviceUnavailableNow((String) msg.obj, msg.arg1);
839 }
840 break;
841 case MSG_L_A2DP_DEVICE_CONFIG_CHANGE:
842 final int a2dpCodec;
843 final BluetoothDevice btDevice = (BluetoothDevice) msg.obj;
844 synchronized (mDeviceStateLock) {
845 a2dpCodec = mBtHelper.getA2dpCodec(btDevice);
846 // TODO: name of method being called on AudioDeviceInventory is currently
847 // misleading (config change vs active device change), to be
848 // reconciliated once the BT side has been updated.
849 mDeviceInventory.onBluetoothA2dpActiveDeviceChange(
850 new BtHelper.BluetoothA2dpDeviceInfo(btDevice, -1, a2dpCodec),
851 BtHelper.EVENT_DEVICE_CONFIG_CHANGE);
852 }
853 break;
854 case MSG_BROADCAST_AUDIO_BECOMING_NOISY:
855 onSendBecomingNoisyIntent();
856 break;
857 case MSG_II_SET_HEARING_AID_VOLUME:
858 synchronized (mDeviceStateLock) {
859 mBtHelper.setHearingAidVolume(msg.arg1, msg.arg2);
860 }
861 break;
862 case MSG_I_SET_AVRCP_ABSOLUTE_VOLUME:
863 synchronized (mDeviceStateLock) {
864 mBtHelper.setAvrcpAbsoluteVolumeIndex(msg.arg1);
865 }
866 break;
867 case MSG_I_DISCONNECT_BT_SCO:
868 synchronized (mSetModeLock) {
869 synchronized (mDeviceStateLock) {
870 mBtHelper.disconnectBluetoothSco(msg.arg1);
871 }
872 }
873 break;
874 case MSG_L_SCOCLIENT_DIED:
875 synchronized (mSetModeLock) {
876 synchronized (mDeviceStateLock) {
877 mBtHelper.scoClientDied(msg.obj);
878 }
879 }
880 break;
881 case MSG_TOGGLE_HDMI:
882 synchronized (mDeviceStateLock) {
883 mDeviceInventory.onToggleHdmi();
884 }
885 break;
886 case MSG_L_A2DP_ACTIVE_DEVICE_CHANGE:
887 synchronized (mDeviceStateLock) {
888 mDeviceInventory.onBluetoothA2dpActiveDeviceChange(
889 (BtHelper.BluetoothA2dpDeviceInfo) msg.obj,
890 BtHelper.EVENT_ACTIVE_DEVICE_CHANGE);
891 }
892 break;
893 case MSG_DISCONNECT_A2DP:
894 synchronized (mDeviceStateLock) {
895 mDeviceInventory.disconnectA2dp();
896 }
897 break;
898 case MSG_DISCONNECT_A2DP_SINK:
899 synchronized (mDeviceStateLock) {
900 mDeviceInventory.disconnectA2dpSink();
901 }
902 break;
903 case MSG_DISCONNECT_BT_HEARING_AID:
904 synchronized (mDeviceStateLock) {
905 mDeviceInventory.disconnectHearingAid();
906 }
907 break;
908 case MSG_DISCONNECT_BT_HEADSET:
909 synchronized (mSetModeLock) {
910 synchronized (mDeviceStateLock) {
911 mBtHelper.disconnectHeadset();
912 }
913 }
914 break;
915 case MSG_L_BT_SERVICE_CONNECTED_PROFILE_A2DP:
916 synchronized (mDeviceStateLock) {
917 mBtHelper.onA2dpProfileConnected((BluetoothA2dp) msg.obj);
918 }
919 break;
920 case MSG_L_BT_SERVICE_CONNECTED_PROFILE_A2DP_SINK:
921 synchronized (mDeviceStateLock) {
922 mBtHelper.onA2dpSinkProfileConnected((BluetoothProfile) msg.obj);
923 }
924 break;
925 case MSG_L_BT_SERVICE_CONNECTED_PROFILE_HEARING_AID:
926 synchronized (mDeviceStateLock) {
927 mBtHelper.onHearingAidProfileConnected((BluetoothHearingAid) msg.obj);
928 }
929 break;
930 case MSG_L_BT_SERVICE_CONNECTED_PROFILE_HEADSET:
931 synchronized (mSetModeLock) {
932 synchronized (mDeviceStateLock) {
933 mBtHelper.onHeadsetProfileConnected((BluetoothHeadset) msg.obj);
934 }
935 }
936 break;
937 case MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_CONNECTION:
938 case MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_DISCONNECTION: {
939 final BtDeviceConnectionInfo info = (BtDeviceConnectionInfo) msg.obj;
940 AudioService.sDeviceLogger.log((new AudioEventLogger.StringEvent(
941 "msg: setBluetoothA2dpDeviceConnectionStateSuppressNoisyIntent "
942 + " state=" + info.mState
943 // only querying address as this is the only readily available
944 // field on the device
945 + " addr=" + info.mDevice.getAddress()
946 + " prof=" + info.mProfile + " supprNoisy=" + info.mSupprNoisy
947 + " vol=" + info.mVolume)).printLog(TAG));
948 synchronized (mDeviceStateLock) {
949 mDeviceInventory.setBluetoothA2dpDeviceConnectionState(
950 info.mDevice, info.mState, info.mProfile, info.mSupprNoisy,
951 AudioSystem.DEVICE_NONE, info.mVolume);
952 }
953 } break;
954 case MSG_L_HEARING_AID_DEVICE_CONNECTION_CHANGE_EXT: {
955 final HearingAidDeviceConnectionInfo info =
956 (HearingAidDeviceConnectionInfo) msg.obj;
957 AudioService.sDeviceLogger.log((new AudioEventLogger.StringEvent(
958 "msg: setHearingAidDeviceConnectionState state=" + info.mState
959 + " addr=" + info.mDevice.getAddress()
960 + " supprNoisy=" + info.mSupprNoisy
961 + " src=" + info.mEventSource)).printLog(TAG));
962 synchronized (mDeviceStateLock) {
963 mDeviceInventory.setBluetoothHearingAidDeviceConnectionState(
964 info.mDevice, info.mState, info.mSupprNoisy, info.mMusicDevice);
965 }
966 } break;
967 case MSG_IL_SAVE_PREF_DEVICE_FOR_STRATEGY: {
968 final int strategy = msg.arg1;
969 final AudioDeviceAttributes device = (AudioDeviceAttributes) msg.obj;
970 mDeviceInventory.onSaveSetPreferredDevice(strategy, device);
971 } break;
972 case MSG_I_SAVE_REMOVE_PREF_DEVICE_FOR_STRATEGY: {
973 final int strategy = msg.arg1;
974 mDeviceInventory.onSaveRemovePreferredDevice(strategy);
975 } break;
976 default:
977 Log.wtf(TAG, "Invalid message " + msg.what);
978 }
979 if (isMessageHandledUnderWakelock(msg.what)) {
980 try {
981 mBrokerEventWakeLock.release();
982 } catch (Exception e) {
983 Log.e(TAG, "Exception releasing wakelock", e);
984 }
985 }
986 }
987 }
988
989 // List of all messages. If a message has be handled under wakelock, add it to
990 // the isMessageHandledUnderWakelock(int) method
991 // Naming of msg indicates arguments, using JNI argument grammar
992 // (e.g. II indicates two int args, IL indicates int and Obj arg)
993 private static final int MSG_RESTORE_DEVICES = 1;
994 private static final int MSG_L_SET_WIRED_DEVICE_CONNECTION_STATE = 2;
995 private static final int MSG_I_BROADCAST_BT_CONNECTION_STATE = 3;
996 private static final int MSG_IIL_SET_FORCE_USE = 4;
997 private static final int MSG_IIL_SET_FORCE_BT_A2DP_USE = 5;
998 private static final int MSG_TOGGLE_HDMI = 6;
999 private static final int MSG_IL_SET_A2DP_SOURCE_CONNECTION_STATE = 7;
1000 private static final int MSG_IL_SET_HEARING_AID_CONNECTION_STATE = 8;
1001 private static final int MSG_BT_HEADSET_CNCT_FAILED = 9;
1002 private static final int MSG_IL_BTA2DP_TIMEOUT = 10;
1003
1004 // process change of A2DP device configuration, obj is BluetoothDevice
1005 private static final int MSG_L_A2DP_DEVICE_CONFIG_CHANGE = 11;
1006
1007 private static final int MSG_BROADCAST_AUDIO_BECOMING_NOISY = 12;
1008 private static final int MSG_REPORT_NEW_ROUTES = 13;
1009 private static final int MSG_II_SET_HEARING_AID_VOLUME = 14;
1010 private static final int MSG_I_SET_AVRCP_ABSOLUTE_VOLUME = 15;
1011 private static final int MSG_I_DISCONNECT_BT_SCO = 16;
1012
1013 // process active A2DP device change, obj is BtHelper.BluetoothA2dpDeviceInfo
1014 private static final int MSG_L_A2DP_ACTIVE_DEVICE_CHANGE = 18;
1015
1016 private static final int MSG_DISCONNECT_A2DP = 19;
1017 private static final int MSG_DISCONNECT_A2DP_SINK = 20;
1018 private static final int MSG_DISCONNECT_BT_HEARING_AID = 21;
1019 private static final int MSG_DISCONNECT_BT_HEADSET = 22;
1020 private static final int MSG_L_BT_SERVICE_CONNECTED_PROFILE_A2DP = 23;
1021 private static final int MSG_L_BT_SERVICE_CONNECTED_PROFILE_A2DP_SINK = 24;
1022 private static final int MSG_L_BT_SERVICE_CONNECTED_PROFILE_HEARING_AID = 25;
1023 private static final int MSG_L_BT_SERVICE_CONNECTED_PROFILE_HEADSET = 26;
1024
1025 // process change of state, obj is BtHelper.BluetoothA2dpDeviceInfo
1026 private static final int MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_CONNECTED = 27;
1027 private static final int MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_DISCONNECTED = 28;
1028
1029 // process external command to (dis)connect an A2DP device, obj is BtDeviceConnectionInfo
1030 private static final int MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_CONNECTION = 29;
1031 private static final int MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_DISCONNECTION = 30;
1032
1033 // process external command to (dis)connect a hearing aid device
1034 private static final int MSG_L_HEARING_AID_DEVICE_CONNECTION_CHANGE_EXT = 31;
1035
1036 // a ScoClient died in BtHelper
1037 private static final int MSG_L_SCOCLIENT_DIED = 32;
1038 private static final int MSG_IL_SAVE_PREF_DEVICE_FOR_STRATEGY = 33;
1039 private static final int MSG_I_SAVE_REMOVE_PREF_DEVICE_FOR_STRATEGY = 34;
1040
1041
1042 private static boolean isMessageHandledUnderWakelock(int msgId) {
1043 switch(msgId) {
1044 case MSG_L_SET_WIRED_DEVICE_CONNECTION_STATE:
1045 case MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_CONNECTED:
1046 case MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_DISCONNECTED:
1047 case MSG_IL_SET_A2DP_SOURCE_CONNECTION_STATE:
1048 case MSG_IL_SET_HEARING_AID_CONNECTION_STATE:
1049 case MSG_IL_BTA2DP_TIMEOUT:
1050 case MSG_L_A2DP_DEVICE_CONFIG_CHANGE:
1051 case MSG_TOGGLE_HDMI:
1052 case MSG_L_A2DP_ACTIVE_DEVICE_CHANGE:
1053 case MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_CONNECTION:
1054 case MSG_L_A2DP_DEVICE_CONNECTION_CHANGE_EXT_DISCONNECTION:
1055 case MSG_L_HEARING_AID_DEVICE_CONNECTION_CHANGE_EXT:
1056 return true;
1057 default:
1058 return false;
1059 }
1060 }
1061
1062 // Message helper methods
1063
1064 // sendMsg() flags
1065 /** If the msg is already queued, replace it with this one. */
1066 private static final int SENDMSG_REPLACE = 0;
1067 /** If the msg is already queued, ignore this one and leave the old. */
1068 private static final int SENDMSG_NOOP = 1;
1069 /** If the msg is already queued, queue this one and leave the old. */
1070 private static final int SENDMSG_QUEUE = 2;
1071
1072 private void sendMsg(int msg, int existingMsgPolicy, int delay) {
1073 sendIILMsg(msg, existingMsgPolicy, 0, 0, null, delay);
1074 }
1075
1076 private void sendILMsg(int msg, int existingMsgPolicy, int arg, Object obj, int delay) {
1077 sendIILMsg(msg, existingMsgPolicy, arg, 0, obj, delay);
1078 }
1079
1080 private void sendLMsg(int msg, int existingMsgPolicy, Object obj, int delay) {
1081 sendIILMsg(msg, existingMsgPolicy, 0, 0, obj, delay);
1082 }
1083
1084 private void sendIMsg(int msg, int existingMsgPolicy, int arg, int delay) {
1085 sendIILMsg(msg, existingMsgPolicy, arg, 0, null, delay);
1086 }
1087
1088 private void sendMsgNoDelay(int msg, int existingMsgPolicy) {
1089 sendIILMsg(msg, existingMsgPolicy, 0, 0, null, 0);
1090 }
1091
1092 private void sendIMsgNoDelay(int msg, int existingMsgPolicy, int arg) {
1093 sendIILMsg(msg, existingMsgPolicy, arg, 0, null, 0);
1094 }
1095
1096 private void sendIIMsgNoDelay(int msg, int existingMsgPolicy, int arg1, int arg2) {
1097 sendIILMsg(msg, existingMsgPolicy, arg1, arg2, null, 0);
1098 }
1099
1100 private void sendILMsgNoDelay(int msg, int existingMsgPolicy, int arg, Object obj) {
1101 sendIILMsg(msg, existingMsgPolicy, arg, 0, obj, 0);
1102 }
1103
1104 private void sendLMsgNoDelay(int msg, int existingMsgPolicy, Object obj) {
1105 sendIILMsg(msg, existingMsgPolicy, 0, 0, obj, 0);
1106 }
1107
1108 private void sendIILMsgNoDelay(int msg, int existingMsgPolicy, int arg1, int arg2, Object obj) {
1109 sendIILMsg(msg, existingMsgPolicy, arg1, arg2, obj, 0);
1110 }
1111
1112 private void sendIILMsg(int msg, int existingMsgPolicy, int arg1, int arg2, Object obj,
1113 int delay) {
1114 if (existingMsgPolicy == SENDMSG_REPLACE) {
1115 mBrokerHandler.removeMessages(msg);
1116 } else if (existingMsgPolicy == SENDMSG_NOOP && mBrokerHandler.hasMessages(msg)) {
1117 return;
1118 }
1119
1120 if (isMessageHandledUnderWakelock(msg)) {
1121 final long identity = Binder.clearCallingIdentity();
1122 try {
1123 mBrokerEventWakeLock.acquire(BROKER_WAKELOCK_TIMEOUT_MS);
1124 } catch (Exception e) {
1125 Log.e(TAG, "Exception acquiring wakelock", e);
1126 }
1127 Binder.restoreCallingIdentity(identity);
1128 }
1129
1130 synchronized (sLastDeviceConnectionMsgTimeLock) {
1131 long time = SystemClock.uptimeMillis() + delay;
1132
1133 switch (msg) {
1134 case MSG_IL_SET_A2DP_SOURCE_CONNECTION_STATE:
1135 case MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_CONNECTED:
1136 case MSG_IL_SET_A2DP_SINK_CONNECTION_STATE_DISCONNECTED:
1137 case MSG_IL_SET_HEARING_AID_CONNECTION_STATE:
1138 case MSG_L_SET_WIRED_DEVICE_CONNECTION_STATE:
1139 case MSG_IL_BTA2DP_TIMEOUT:
1140 case MSG_L_A2DP_DEVICE_CONFIG_CHANGE:
1141 case MSG_L_A2DP_ACTIVE_DEVICE_CHANGE:
1142 if (sLastDeviceConnectMsgTime >= time) {
1143 // add a little delay to make sure messages are ordered as expected
1144 time = sLastDeviceConnectMsgTime + 30;
1145 }
1146 sLastDeviceConnectMsgTime = time;
1147 break;
1148 default:
1149 break;
1150 }
1151
1152 mBrokerHandler.sendMessageAtTime(mBrokerHandler.obtainMessage(msg, arg1, arg2, obj),
1153 time);
1154 }
1155 }
1156}