1 /* 2 * Copyright (C) 2016 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 */ 16 17 package com.android.server.telecom.bluetooth; 18 19 import android.bluetooth.BluetoothAdapter; 20 import android.bluetooth.BluetoothDevice; 21 import android.bluetooth.BluetoothHeadset; 22 import android.bluetooth.BluetoothHearingAid; 23 import android.bluetooth.BluetoothProfile; 24 import android.content.Context; 25 import android.telecom.Log; 26 import android.util.LocalLog; 27 28 import com.android.internal.util.IndentingPrintWriter; 29 import com.android.server.telecom.BluetoothAdapterProxy; 30 import com.android.server.telecom.BluetoothHeadsetProxy; 31 32 import java.util.ArrayList; 33 import java.util.Collection; 34 import java.util.Collections; 35 import java.util.LinkedHashMap; 36 import java.util.LinkedHashSet; 37 import java.util.LinkedList; 38 import java.util.List; 39 import java.util.Set; 40 41 public class BluetoothDeviceManager { 42 private final BluetoothProfile.ServiceListener mBluetoothProfileServiceListener = 43 new BluetoothProfile.ServiceListener() { 44 @Override 45 public void onServiceConnected(int profile, BluetoothProfile proxy) { 46 Log.startSession("BMSL.oSC"); 47 try { 48 synchronized (mLock) { 49 String logString; 50 if (profile == BluetoothProfile.HEADSET) { 51 mBluetoothHeadsetService = 52 new BluetoothHeadsetProxy((BluetoothHeadset) proxy); 53 logString = "Got BluetoothHeadset: " + mBluetoothHeadsetService; 54 } else if (profile == BluetoothProfile.HEARING_AID) { 55 mBluetoothHearingAidService = (BluetoothHearingAid) proxy; 56 logString = "Got BluetoothHearingAid: " 57 + mBluetoothHearingAidService; 58 } else { 59 logString = "Connected to non-requested bluetooth service." + 60 " Not changing bluetooth headset."; 61 } 62 Log.i(BluetoothDeviceManager.this, logString); 63 mLocalLog.log(logString); 64 } 65 } finally { 66 Log.endSession(); 67 } 68 } 69 70 @Override 71 public void onServiceDisconnected(int profile) { 72 Log.startSession("BMSL.oSD"); 73 try { 74 synchronized (mLock) { 75 LinkedHashMap<String, BluetoothDevice> lostServiceDevices; 76 String logString; 77 if (profile == BluetoothProfile.HEADSET) { 78 mBluetoothHeadsetService = null; 79 lostServiceDevices = mHfpDevicesByAddress; 80 mBluetoothRouteManager.onActiveDeviceChanged(null, false); 81 logString = "Lost BluetoothHeadset service. " + 82 "Removing all tracked devices"; 83 } else if (profile == BluetoothProfile.HEARING_AID) { 84 mBluetoothHearingAidService = null; 85 logString = "Lost BluetoothHearingAid service. " + 86 "Removing all tracked devices."; 87 lostServiceDevices = mHearingAidDevicesByAddress; 88 mBluetoothRouteManager.onActiveDeviceChanged(null, true); 89 } else { 90 return; 91 } 92 Log.i(BluetoothDeviceManager.this, logString); 93 mLocalLog.log(logString); 94 95 List<BluetoothDevice> devicesToRemove = new LinkedList<>( 96 lostServiceDevices.values()); 97 lostServiceDevices.clear(); 98 for (BluetoothDevice device : devicesToRemove) { 99 mBluetoothRouteManager.onDeviceLost(device.getAddress()); 100 } 101 } 102 } finally { 103 Log.endSession(); 104 } 105 } 106 }; 107 108 private final LinkedHashMap<String, BluetoothDevice> mHfpDevicesByAddress = 109 new LinkedHashMap<>(); 110 private final LinkedHashMap<String, BluetoothDevice> mHearingAidDevicesByAddress = 111 new LinkedHashMap<>(); 112 private final LinkedHashMap<BluetoothDevice, Long> mHearingAidDeviceSyncIds = 113 new LinkedHashMap<>(); 114 private final LocalLog mLocalLog = new LocalLog(20); 115 116 // This lock only protects internal state -- it doesn't lock on anything going into Telecom. 117 private final Object mLock = new Object(); 118 119 private BluetoothRouteManager mBluetoothRouteManager; 120 private BluetoothHeadsetProxy mBluetoothHeadsetService; 121 private BluetoothHearingAid mBluetoothHearingAidService; 122 private BluetoothDevice mBluetoothHearingAidActiveDeviceCache; 123 private BluetoothAdapterProxy mBluetoothAdapterProxy; 124 125 public BluetoothDeviceManager(Context context, BluetoothAdapterProxy bluetoothAdapter) { 126 if (bluetoothAdapter != null) { 127 mBluetoothAdapterProxy = bluetoothAdapter; 128 bluetoothAdapter.getProfileProxy(context, mBluetoothProfileServiceListener, 129 BluetoothProfile.HEADSET); 130 bluetoothAdapter.getProfileProxy(context, mBluetoothProfileServiceListener, 131 BluetoothProfile.HEARING_AID); 132 } 133 } 134 135 public void setBluetoothRouteManager(BluetoothRouteManager brm) { 136 mBluetoothRouteManager = brm; 137 } 138 139 public int getNumConnectedDevices() { 140 synchronized (mLock) { 141 return mHfpDevicesByAddress.size() + mHearingAidDevicesByAddress.size(); 142 } 143 } 144 145 public Collection<BluetoothDevice> getConnectedDevices() { 146 synchronized (mLock) { 147 ArrayList<BluetoothDevice> result = new ArrayList<>(mHfpDevicesByAddress.values()); 148 result.addAll(mHearingAidDevicesByAddress.values()); 149 return Collections.unmodifiableCollection(result); 150 } 151 } 152 153 // Same as getConnectedDevices except it filters out the hearing aid devices that are linked 154 // together by their hiSyncId. 155 public Collection<BluetoothDevice> getUniqueConnectedDevices() { 156 ArrayList<BluetoothDevice> result; 157 synchronized (mLock) { 158 result = new ArrayList<>(mHfpDevicesByAddress.values()); 159 } 160 Set<Long> seenHiSyncIds = new LinkedHashSet<>(); 161 // Add the left-most active device to the seen list so that we match up with the list 162 // generated in BluetoothRouteManager. 163 if (mBluetoothHearingAidService != null) { 164 for (BluetoothDevice device : mBluetoothHearingAidService.getActiveDevices()) { 165 if (device != null) { 166 result.add(device); 167 seenHiSyncIds.add(mHearingAidDeviceSyncIds.getOrDefault(device, -1L)); 168 break; 169 } 170 } 171 } 172 synchronized (mLock) { 173 for (BluetoothDevice d : mHearingAidDevicesByAddress.values()) { 174 long hiSyncId = mHearingAidDeviceSyncIds.getOrDefault(d, -1L); 175 if (seenHiSyncIds.contains(hiSyncId)) { 176 continue; 177 } 178 result.add(d); 179 seenHiSyncIds.add(hiSyncId); 180 } 181 } 182 return Collections.unmodifiableCollection(result); 183 } 184 185 public BluetoothHeadsetProxy getHeadsetService() { 186 return mBluetoothHeadsetService; 187 } 188 189 public BluetoothHearingAid getHearingAidService() { 190 return mBluetoothHearingAidService; 191 } 192 193 public void setHeadsetServiceForTesting(BluetoothHeadsetProxy bluetoothHeadset) { 194 mBluetoothHeadsetService = bluetoothHeadset; 195 } 196 197 public void setHearingAidServiceForTesting(BluetoothHearingAid bluetoothHearingAid) { 198 mBluetoothHearingAidService = bluetoothHearingAid; 199 } 200 201 void onDeviceConnected(BluetoothDevice device, boolean isHearingAid) { 202 mLocalLog.log("Device connected -- address: " + device.getAddress() + " isHeadingAid: " 203 + isHearingAid); 204 synchronized (mLock) { 205 LinkedHashMap<String, BluetoothDevice> targetDeviceMap; 206 if (isHearingAid) { 207 if (mBluetoothHearingAidService == null) { 208 Log.w(this, "Hearing aid service null when receiving device added broadcast"); 209 return; 210 } 211 long hiSyncId = mBluetoothHearingAidService.getHiSyncId(device); 212 mHearingAidDeviceSyncIds.put(device, hiSyncId); 213 targetDeviceMap = mHearingAidDevicesByAddress; 214 } else { 215 if (mBluetoothHeadsetService == null) { 216 Log.w(this, "Headset service null when receiving device added broadcast"); 217 return; 218 } 219 targetDeviceMap = mHfpDevicesByAddress; 220 } 221 if (!targetDeviceMap.containsKey(device.getAddress())) { 222 targetDeviceMap.put(device.getAddress(), device); 223 mBluetoothRouteManager.onDeviceAdded(device.getAddress()); 224 } 225 } 226 } 227 228 void onDeviceDisconnected(BluetoothDevice device, boolean isHearingAid) { 229 mLocalLog.log("Device disconnected -- address: " + device.getAddress() + " isHeadingAid: " 230 + isHearingAid); 231 synchronized (mLock) { 232 LinkedHashMap<String, BluetoothDevice> targetDeviceMap; 233 if (isHearingAid) { 234 mHearingAidDeviceSyncIds.remove(device); 235 targetDeviceMap = mHearingAidDevicesByAddress; 236 } else { 237 targetDeviceMap = mHfpDevicesByAddress; 238 } 239 if (targetDeviceMap.containsKey(device.getAddress())) { 240 targetDeviceMap.remove(device.getAddress()); 241 mBluetoothRouteManager.onDeviceLost(device.getAddress()); 242 } 243 } 244 } 245 246 public void disconnectAudio() { 247 if (mBluetoothHearingAidService == null) { 248 Log.w(this, "Trying to disconnect audio but no hearing aid service exists"); 249 } else { 250 for (BluetoothDevice device : mBluetoothHearingAidService.getActiveDevices()) { 251 if (device != null) { 252 mBluetoothAdapterProxy.setActiveDevice(null, 253 BluetoothAdapter.ACTIVE_DEVICE_ALL); 254 } 255 } 256 } 257 disconnectSco(); 258 } 259 260 public void disconnectSco() { 261 if (mBluetoothHeadsetService == null) { 262 Log.w(this, "Trying to disconnect audio but no headset service exists."); 263 } else { 264 mBluetoothHeadsetService.disconnectAudio(); 265 } 266 } 267 268 // Connect audio to the bluetooth device at address, checking to see whether it's a hearing aid 269 // or a HFP device, and using the proper BT API. 270 public boolean connectAudio(String address) { 271 if (mHearingAidDevicesByAddress.containsKey(address)) { 272 if (mBluetoothHearingAidService == null) { 273 Log.w(this, "Attempting to turn on audio when the hearing aid service is null"); 274 return false; 275 } 276 return mBluetoothAdapterProxy.setActiveDevice( 277 mHearingAidDevicesByAddress.get(address), 278 BluetoothAdapter.ACTIVE_DEVICE_ALL); 279 } else if (mHfpDevicesByAddress.containsKey(address)) { 280 BluetoothDevice device = mHfpDevicesByAddress.get(address); 281 if (mBluetoothHeadsetService == null) { 282 Log.w(this, "Attempting to turn on audio when the headset service is null"); 283 return false; 284 } 285 boolean success = mBluetoothAdapterProxy.setActiveDevice(device, 286 BluetoothAdapter.ACTIVE_DEVICE_PHONE_CALL); 287 if (!success) { 288 Log.w(this, "Couldn't set active device to %s", address); 289 return false; 290 } 291 if (!mBluetoothHeadsetService.isAudioOn()) { 292 return mBluetoothHeadsetService.connectAudio(); 293 } 294 return true; 295 } else { 296 Log.w(this, "Attempting to turn on audio for a disconnected device"); 297 return false; 298 } 299 } 300 301 public void cacheHearingAidDevice() { 302 if (mBluetoothHearingAidService != null) { 303 for (BluetoothDevice device : mBluetoothHearingAidService.getActiveDevices()) { 304 if (device != null) { 305 mBluetoothHearingAidActiveDeviceCache = device; 306 } 307 } 308 } 309 } 310 311 public void restoreHearingAidDevice() { 312 if (mBluetoothHearingAidActiveDeviceCache != null && mBluetoothHearingAidService != null) { 313 mBluetoothAdapterProxy.setActiveDevice( 314 mBluetoothHearingAidActiveDeviceCache, 315 BluetoothAdapter.ACTIVE_DEVICE_ALL); 316 mBluetoothHearingAidActiveDeviceCache = null; 317 } 318 } 319 320 public void dump(IndentingPrintWriter pw) { 321 mLocalLog.dump(pw); 322 } 323 } 324