blob: 2095fd496d3088cb9dd3404b9fecaa85961fa1a2 [file] [log] [blame]
/*
* Copyright (C) 2012 The Android Open Source Project
* Copyright (C) 2016-2017 The Linux Foundation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.bluetooth.btservice;
import static android.Manifest.permission.BLUETOOTH_CONNECT;
import static android.Manifest.permission.BLUETOOTH_SCAN;
import android.annotation.NonNull;
import android.annotation.RequiresPermission;
import android.app.BroadcastOptions;
import android.bluetooth.BluetoothA2dp;
import android.bluetooth.BluetoothA2dpSink;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothAvrcpController;
import android.bluetooth.BluetoothClass;
import android.bluetooth.BluetoothDevice;
import android.bluetooth.BluetoothHeadset;
import android.bluetooth.BluetoothHeadsetClient;
import android.bluetooth.BluetoothHearingAid;
import android.bluetooth.BluetoothHidDevice;
import android.bluetooth.BluetoothHidHost;
import android.bluetooth.BluetoothLeAudio;
import android.bluetooth.BluetoothMap;
import android.bluetooth.BluetoothMapClient;
import android.bluetooth.BluetoothPan;
import android.bluetooth.BluetoothPbap;
import android.bluetooth.BluetoothPbapClient;
import android.bluetooth.BluetoothProfile;
import android.bluetooth.BluetoothSap;
import android.bluetooth.BufferConstraint;
import android.bluetooth.BufferConstraints;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.os.ParcelUuid;
import android.os.SystemProperties;
import android.os.UserHandle;
import android.util.Log;
import android.util.Pair;
import androidx.annotation.VisibleForTesting;
import com.android.bluetooth.BluetoothStatsLog;
import com.android.bluetooth.Utils;
import com.android.bluetooth.btservice.RemoteDevices.DeviceProperties;
import com.android.modules.utils.build.SdkLevel;
import com.google.common.collect.EvictingQueue;
import java.io.FileDescriptor;
import java.io.PrintWriter;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
class AdapterProperties {
private static final boolean DBG = true;
private static final boolean VDBG = false;
private static final String TAG = "AdapterProperties";
private static final String MAX_CONNECTED_AUDIO_DEVICES_PROPERTY =
"persist.bluetooth.maxconnectedaudiodevices";
private static final int MAX_CONNECTED_AUDIO_DEVICES_LOWER_BOUND = 1;
private static final int MAX_CONNECTED_AUDIO_DEVICES_UPPER_BOUND = 5;
private static final String A2DP_OFFLOAD_SUPPORTED_PROPERTY =
"ro.bluetooth.a2dp_offload.supported";
private static final String A2DP_OFFLOAD_DISABLED_PROPERTY =
"persist.bluetooth.a2dp_offload.disabled";
private static final long DEFAULT_DISCOVERY_TIMEOUT_MS = 12800;
@VisibleForTesting static final int BLUETOOTH_NAME_MAX_LENGTH_BYTES = 248;
private static final int BD_ADDR_LEN = 6; // in bytes
private volatile String mName;
private volatile byte[] mAddress;
private volatile BluetoothClass mBluetoothClass;
private volatile int mScanMode;
private volatile int mDiscoverableTimeout;
private volatile ParcelUuid[] mUuids;
private volatile int mLocalIOCapability = BluetoothAdapter.IO_CAPABILITY_UNKNOWN;
private CopyOnWriteArrayList<BluetoothDevice> mBondedDevices =
new CopyOnWriteArrayList<BluetoothDevice>();
private static final int SCAN_MODE_CHANGES_MAX_SIZE = 10;
private EvictingQueue<String> mScanModeChanges;
private CopyOnWriteArrayList<String> mAllowlistedPlayers =
new CopyOnWriteArrayList<String>();
private int mProfilesConnecting, mProfilesConnected, mProfilesDisconnecting;
private final HashMap<Integer, Pair<Integer, Integer>> mProfileConnectionState =
new HashMap<>();
private volatile int mConnectionState = BluetoothAdapter.STATE_DISCONNECTED;
private volatile int mState = BluetoothAdapter.STATE_OFF;
private int mMaxConnectedAudioDevices = 1;
private boolean mA2dpOffloadEnabled = false;
private AdapterService mService;
private boolean mDiscovering;
private long mDiscoveryEndMs; //< Time (ms since epoch) that discovery ended or will end.
private RemoteDevices mRemoteDevices;
private BluetoothAdapter mAdapter;
//TODO - all hw capabilities to be exposed as a class
private int mNumOfAdvertisementInstancesSupported;
private boolean mRpaOffloadSupported;
private int mNumOfOffloadedIrkSupported;
private int mNumOfOffloadedScanFilterSupported;
private int mOffloadedScanResultStorageBytes;
private int mVersSupported;
private int mTotNumOfTrackableAdv;
private boolean mIsExtendedScanSupported;
private boolean mIsDebugLogSupported;
private boolean mIsActivityAndEnergyReporting;
private boolean mIsLe2MPhySupported;
private boolean mIsLeCodedPhySupported;
private boolean mIsLeExtendedAdvertisingSupported;
private boolean mIsLePeriodicAdvertisingSupported;
private int mLeMaximumAdvertisingDataLength;
private boolean mIsOffloadedTransportDiscoveryDataScanSupported;
private int mIsDynamicAudioBufferSizeSupported;
private int mDynamicAudioBufferSizeSupportedCodecsGroup1;
private int mDynamicAudioBufferSizeSupportedCodecsGroup2;
private boolean mIsLePeriodicAdvertisingSyncTransferSenderSupported;
private boolean mIsLePeriodicAdvertisingSyncTransferRecipientSupported;
private boolean mIsLeConnectedIsochronousStreamCentralSupported;
private boolean mIsLeIsochronousBroadcasterSupported;
private List<BufferConstraint> mBufferConstraintList;
private boolean mReceiverRegistered;
private BroadcastReceiver mReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
if (action == null) {
Log.w(TAG, "Received intent with null action");
return;
}
switch (action) {
case BluetoothHeadset.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.HEADSET, intent);
break;
case BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.A2DP, intent);
break;
case BluetoothHeadsetClient.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.HEADSET_CLIENT, intent);
break;
case BluetoothHearingAid.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.HEARING_AID, intent);
break;
case BluetoothA2dpSink.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.A2DP_SINK, intent);
break;
case BluetoothHidDevice.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.HID_DEVICE, intent);
break;
case BluetoothHidHost.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.HID_HOST, intent);
break;
case BluetoothAvrcpController.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.AVRCP_CONTROLLER, intent);
break;
case BluetoothPan.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.PAN, intent);
break;
case BluetoothMap.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.MAP, intent);
break;
case BluetoothMapClient.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.MAP_CLIENT, intent);
break;
case BluetoothSap.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.SAP, intent);
break;
case BluetoothPbapClient.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.PBAP_CLIENT, intent);
break;
case BluetoothPbap.ACTION_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.PBAP, intent);
break;
case BluetoothLeAudio.ACTION_LE_AUDIO_CONNECTION_STATE_CHANGED:
sendConnectionStateChange(BluetoothProfile.LE_AUDIO, intent);
break;
default:
Log.w(TAG, "Received unknown intent " + intent);
break;
}
}
};
// Lock for all getters and setters.
// If finer grained locking is needer, more locks
// can be added here.
private final Object mObject = new Object();
AdapterProperties(AdapterService service) {
mService = service;
mAdapter = BluetoothAdapter.getDefaultAdapter();
mScanModeChanges = EvictingQueue.create(SCAN_MODE_CHANGES_MAX_SIZE);
invalidateBluetoothCaches();
}
public void init(RemoteDevices remoteDevices) {
mProfileConnectionState.clear();
mRemoteDevices = remoteDevices;
// Get default max connected audio devices from config.xml
int configDefaultMaxConnectedAudioDevices = mService.getResources().getInteger(
com.android.bluetooth.R.integer.config_bluetooth_max_connected_audio_devices);
// Override max connected audio devices if MAX_CONNECTED_AUDIO_DEVICES_PROPERTY is set
int propertyOverlayedMaxConnectedAudioDevices =
SystemProperties.getInt(MAX_CONNECTED_AUDIO_DEVICES_PROPERTY,
configDefaultMaxConnectedAudioDevices);
// Make sure the final value of max connected audio devices is within allowed range
mMaxConnectedAudioDevices = Math.min(Math.max(propertyOverlayedMaxConnectedAudioDevices,
MAX_CONNECTED_AUDIO_DEVICES_LOWER_BOUND), MAX_CONNECTED_AUDIO_DEVICES_UPPER_BOUND);
Log.i(TAG, "init(), maxConnectedAudioDevices, default="
+ configDefaultMaxConnectedAudioDevices + ", propertyOverlayed="
+ propertyOverlayedMaxConnectedAudioDevices + ", finalValue="
+ mMaxConnectedAudioDevices);
mA2dpOffloadEnabled =
SystemProperties.getBoolean(A2DP_OFFLOAD_SUPPORTED_PROPERTY, false)
&& !SystemProperties.getBoolean(A2DP_OFFLOAD_DISABLED_PROPERTY, false);
IntentFilter filter = new IntentFilter();
filter.setPriority(IntentFilter.SYSTEM_HIGH_PRIORITY);
filter.addAction(BluetoothHeadset.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothHeadsetClient.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothHearingAid.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothA2dpSink.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothHidDevice.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothHidHost.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothAvrcpController.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothPan.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothMap.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothMapClient.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothSap.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothPbapClient.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothLeAudio.ACTION_LE_AUDIO_CONNECTION_STATE_CHANGED);
mService.registerReceiver(mReceiver, filter);
mReceiverRegistered = true;
invalidateBluetoothCaches();
}
public void cleanup() {
mRemoteDevices = null;
mProfileConnectionState.clear();
if (mReceiverRegistered) {
mService.unregisterReceiver(mReceiver);
mReceiverRegistered = false;
}
mService = null;
mBondedDevices.clear();
mScanModeChanges.clear();
invalidateBluetoothCaches();
mAllowlistedPlayers.clear();
}
private static void invalidateGetProfileConnectionStateCache() {
BluetoothAdapter.invalidateGetProfileConnectionStateCache();
}
private static void invalidateIsOffloadedFilteringSupportedCache() {
BluetoothAdapter.invalidateIsOffloadedFilteringSupportedCache();
}
private static void invalidateBluetoothGetConnectionStateCache() {
BluetoothMap.invalidateBluetoothGetConnectionStateCache();
BluetoothSap.invalidateBluetoothGetConnectionStateCache();
}
private static void invalidateGetConnectionStateCache() {
BluetoothAdapter.invalidateGetAdapterConnectionStateCache();
}
private static void invalidateGetBondStateCache() {
BluetoothDevice.invalidateBluetoothGetBondStateCache();
}
private static void invalidateBluetoothCaches() {
invalidateGetProfileConnectionStateCache();
invalidateIsOffloadedFilteringSupportedCache();
invalidateGetConnectionStateCache();
invalidateGetBondStateCache();
invalidateBluetoothGetConnectionStateCache();
}
@Override
public Object clone() throws CloneNotSupportedException {
throw new CloneNotSupportedException();
}
/**
* @return the mName
*/
String getName() {
return mName;
}
/**
* Set the local adapter property - name
* @param name the name to set
*/
boolean setName(String name) {
synchronized (mObject) {
return mService.getNative()
.setAdapterProperty(
AbstractionLayer.BT_PROPERTY_BDNAME,
Utils.truncateStringForUtf8Storage(
name, BLUETOOTH_NAME_MAX_LENGTH_BYTES)
.getBytes());
}
}
boolean setIoCapability(int capability) {
synchronized (mObject) {
boolean result =
mService.getNative()
.setAdapterProperty(
AbstractionLayer.BT_PROPERTY_LOCAL_IO_CAPS,
Utils.intToByteArray(capability));
if (result) {
mLocalIOCapability = capability;
}
return result;
}
}
int getIoCapability() {
synchronized (mObject) {
return mLocalIOCapability;
}
}
/**
* @return the mScanMode
*/
int getScanMode() {
return mScanMode;
}
/**
* Set the local adapter property - scanMode
*
* @param scanMode the ScanMode to set, valid values are: {
* BluetoothAdapter.SCAN_MODE_NONE,
* BluetoothAdapter.SCAN_MODE_CONNECTABLE,
* BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE,
* }
*/
boolean setScanMode(int scanMode) {
addScanChangeLog(scanMode);
synchronized (mObject) {
return mService.getNative()
.setAdapterProperty(
AbstractionLayer.BT_PROPERTY_ADAPTER_SCAN_MODE,
Utils.intToByteArray(AdapterService.convertScanModeToHal(scanMode)));
}
}
private void addScanChangeLog(int scanMode) {
String time = Utils.getLocalTimeString();
String uidPid = Utils.getUidPidString();
String scanModeString = dumpScanMode(scanMode);
mScanModeChanges.add(time + " (" + uidPid + ") " + scanModeString);
}
/**
* @return the mUuids
*/
ParcelUuid[] getUuids() {
return mUuids;
}
/**
* @return the mAddress
*/
byte[] getAddress() {
return mAddress;
}
/**
* @param connectionState the mConnectionState to set
*/
void setConnectionState(int connectionState) {
mConnectionState = connectionState;
invalidateGetConnectionStateCache();
}
/**
* @return the mConnectionState
*/
int getConnectionState() {
return mConnectionState;
}
/**
* @param state the mState to set
*/
void setState(int state) {
debugLog("Setting state to " + BluetoothAdapter.nameForState(state));
mState = state;
}
/**
* @return the mState
*/
int getState() {
return mState;
}
/**
* @return the mNumOfAdvertisementInstancesSupported
*/
int getNumOfAdvertisementInstancesSupported() {
return mNumOfAdvertisementInstancesSupported;
}
/**
* @return the mRpaOffloadSupported
*/
boolean isRpaOffloadSupported() {
return mRpaOffloadSupported;
}
/**
* @return the mNumOfOffloadedIrkSupported
*/
int getNumOfOffloadedIrkSupported() {
return mNumOfOffloadedIrkSupported;
}
/**
* @return the mNumOfOffloadedScanFilterSupported
*/
int getNumOfOffloadedScanFilterSupported() {
return mNumOfOffloadedScanFilterSupported;
}
/**
* @return the mOffloadedScanResultStorageBytes
*/
int getOffloadedScanResultStorage() {
return mOffloadedScanResultStorageBytes;
}
/**
* @return tx/rx/idle activity and energy info
*/
boolean isActivityAndEnergyReportingSupported() {
return mIsActivityAndEnergyReporting;
}
/**
* @return the mIsLe2MPhySupported
*/
boolean isLe2MPhySupported() {
return mIsLe2MPhySupported;
}
/**
* @return the mIsLeCodedPhySupported
*/
boolean isLeCodedPhySupported() {
return mIsLeCodedPhySupported;
}
/**
* @return the mIsLeExtendedAdvertisingSupported
*/
boolean isLeExtendedAdvertisingSupported() {
return mIsLeExtendedAdvertisingSupported;
}
/**
* @return the mIsLePeriodicAdvertisingSupported
*/
boolean isLePeriodicAdvertisingSupported() {
return mIsLePeriodicAdvertisingSupported;
}
/**
* @return the mIsLePeriodicAdvertisingSyncTransferSenderSupported
*/
boolean isLePeriodicAdvertisingSyncTransferSenderSupported() {
return mIsLePeriodicAdvertisingSyncTransferSenderSupported;
}
/**
* @return the mIsLePeriodicAdvertisingSyncTransferRecipientSupported
*/
boolean isLePeriodicAdvertisingSyncTransferRecipientSupported() {
return mIsLePeriodicAdvertisingSyncTransferRecipientSupported;
}
/**
* @return the mIsLeConnectedIsochronousStreamCentralSupported
*/
boolean isLeConnectedIsochronousStreamCentralSupported() {
return mIsLeConnectedIsochronousStreamCentralSupported;
}
/**
* @return the mIsLeIsochronousBroadcasterSupported
*/
boolean isLeIsochronousBroadcasterSupported() {
return mIsLeIsochronousBroadcasterSupported;
}
/**
* @return the getLeMaximumAdvertisingDataLength
*/
int getLeMaximumAdvertisingDataLength() {
return mLeMaximumAdvertisingDataLength;
}
/**
* @return total number of trackable advertisements
*/
int getTotalNumOfTrackableAdvertisements() {
return mTotNumOfTrackableAdv;
}
/**
* @return the isOffloadedTransportDiscoveryDataScanSupported
*/
public boolean isOffloadedTransportDiscoveryDataScanSupported() {
return mIsOffloadedTransportDiscoveryDataScanSupported;
}
/**
* @return the maximum number of connected audio devices
*/
int getMaxConnectedAudioDevices() {
return mMaxConnectedAudioDevices;
}
/**
* @return A2DP offload support
*/
boolean isA2dpOffloadEnabled() {
return mA2dpOffloadEnabled;
}
/**
* @return Dynamic Audio Buffer support
*/
int getDynamicBufferSupport() {
if (!mA2dpOffloadEnabled) {
// TODO: Enable Dynamic Audio Buffer for A2DP software encoding when ready.
mIsDynamicAudioBufferSizeSupported =
BluetoothA2dp.DYNAMIC_BUFFER_SUPPORT_NONE;
} else {
if ((mDynamicAudioBufferSizeSupportedCodecsGroup1 != 0)
|| (mDynamicAudioBufferSizeSupportedCodecsGroup2 != 0)) {
mIsDynamicAudioBufferSizeSupported =
BluetoothA2dp.DYNAMIC_BUFFER_SUPPORT_A2DP_OFFLOAD;
} else {
mIsDynamicAudioBufferSizeSupported =
BluetoothA2dp.DYNAMIC_BUFFER_SUPPORT_NONE;
}
}
return mIsDynamicAudioBufferSizeSupported;
}
/**
* @return Dynamic Audio Buffer Capability
*/
BufferConstraints getBufferConstraints() {
return new BufferConstraints(mBufferConstraintList);
}
/**
* Set the dynamic audio buffer size
*
* @param codec the codecs to set
* @param size the size to set
*/
boolean setBufferLengthMillis(int codec, int size) {
return mService.getNative().setBufferLengthMillis(codec, size);
}
/**
* @return the mBondedDevices
*/
BluetoothDevice[] getBondedDevices() {
BluetoothDevice[] bondedDeviceList = new BluetoothDevice[0];
try {
bondedDeviceList = mBondedDevices.toArray(bondedDeviceList);
} catch (ArrayStoreException ee) {
errorLog("Error retrieving bonded device array");
}
infoLog("getBondedDevices: length=" + bondedDeviceList.length);
return bondedDeviceList;
}
// This function shall be invoked from BondStateMachine whenever the bond
// state changes.
@VisibleForTesting
void onBondStateChanged(BluetoothDevice device, int state) {
if (device == null) {
Log.w(TAG, "onBondStateChanged, device is null");
return;
}
try {
byte[] addrByte = Utils.getByteAddress(device);
DeviceProperties prop = mRemoteDevices.getDeviceProperties(device);
if (prop == null) {
prop = mRemoteDevices.addDeviceProperties(addrByte);
}
device = prop.getDevice();
prop.setBondState(state);
if (state == BluetoothDevice.BOND_BONDED) {
// add if not already in list
if (!mBondedDevices.contains(device)) {
debugLog("Adding bonded device:" + device);
mBondedDevices.add(device);
cleanupPrevBondRecordsFor(device);
}
} else if (state == BluetoothDevice.BOND_NONE) {
// remove device from list
if (mBondedDevices.remove(device)) {
debugLog("Removing bonded device:" + device);
} else {
debugLog("Failed to remove device: " + device);
}
}
invalidateGetBondStateCache();
} catch (Exception ee) {
Log.w(TAG, "onBondStateChanged: Exception ", ee);
}
}
void cleanupPrevBondRecordsFor(BluetoothDevice currentDevice) {
String currentAddress = currentDevice.getAddress();
String currentIdentityAddress = mService.getIdentityAddress(currentAddress);
debugLog("cleanupPrevBondRecordsFor: " + currentDevice);
if (currentIdentityAddress == null) {
return;
}
for (BluetoothDevice device : mBondedDevices) {
String address = device.getAddress();
String identityAddress = mService.getIdentityAddress(address);
if (currentIdentityAddress.equals(identityAddress) && !currentAddress.equals(address)) {
if (mService.getNative()
.removeBond(Utils.getBytesFromAddress(device.getAddress()))) {
mBondedDevices.remove(device);
infoLog("Removing old bond record: "
+ device
+ " for current device: "
+ currentDevice);
} else {
Log.e(TAG, "Unexpected error while removing old bond record:"
+ device
+ " for current device: "
+ currentDevice);
}
break;
}
}
}
int getDiscoverableTimeout() {
return mDiscoverableTimeout;
}
boolean setDiscoverableTimeout(int timeout) {
synchronized (mObject) {
return mService.getNative()
.setAdapterProperty(
AbstractionLayer.BT_PROPERTY_ADAPTER_DISCOVERABLE_TIMEOUT,
Utils.intToByteArray(timeout));
}
}
int getProfileConnectionState(int profile) {
synchronized (mObject) {
Pair<Integer, Integer> p = mProfileConnectionState.get(profile);
if (p != null) {
return p.first;
}
return BluetoothProfile.STATE_DISCONNECTED;
}
}
/**
* @return the mAllowlistedPlayers
*/
String[] getAllowlistedMediaPlayers() {
String[] AllowlistedPlayersList = new String[0];
try {
AllowlistedPlayersList = mAllowlistedPlayers.toArray(AllowlistedPlayersList);
} catch (ArrayStoreException ee) {
errorLog("Error retrieving Allowlisted Players array");
}
Log.d(TAG, "getAllowlistedMediaPlayers: numAllowlistedPlayers = "
+ AllowlistedPlayersList.length);
for (int i = 0; i < AllowlistedPlayersList.length; i++) {
Log.d(TAG, "players :" + AllowlistedPlayersList[i]);
}
return AllowlistedPlayersList;
}
long discoveryEndMillis() {
return mDiscoveryEndMs;
}
boolean isDiscovering() {
return mDiscovering;
}
@RequiresPermission(android.Manifest.permission.INTERACT_ACROSS_USERS)
private void sendConnectionStateChange(int profile, Intent connIntent) {
BluetoothDevice device = connIntent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
int prevState = connIntent.getIntExtra(BluetoothProfile.EXTRA_PREVIOUS_STATE, -1);
int state = connIntent.getIntExtra(BluetoothProfile.EXTRA_STATE, -1);
int metricId = mService.getMetricId(device);
if (state == BluetoothProfile.STATE_CONNECTING) {
BluetoothStatsLog.write(BluetoothStatsLog.BLUETOOTH_DEVICE_NAME_REPORTED,
metricId, device.getName());
MetricsLogger.getInstance().logSanitizedBluetoothDeviceName(metricId, device.getName());
}
Log.d(TAG, "PROFILE_CONNECTION_STATE_CHANGE: profile="
+ BluetoothProfile.getProfileName(profile) + ", device=" + device + ", "
+ prevState + " -> " + state);
BluetoothStatsLog.write(BluetoothStatsLog.BLUETOOTH_CONNECTION_STATE_CHANGED, state,
0 /* deprecated */, profile, mService.obfuscateAddress(device),
metricId, 0, -1);
if (!isNormalStateTransition(prevState, state)) {
Log.w(TAG, "PROFILE_CONNECTION_STATE_CHANGE: unexpected transition for profile="
+ BluetoothProfile.getProfileName(profile)
+ ", device=" + device + ", " + prevState + " -> " + state);
}
sendConnectionStateChange(device, profile, state, prevState);
}
@RequiresPermission(android.Manifest.permission.INTERACT_ACROSS_USERS)
void sendConnectionStateChange(BluetoothDevice device, int profile, int state, int prevState) {
if (!validateProfileConnectionState(state) || !validateProfileConnectionState(prevState)) {
// Previously, an invalid state was broadcast anyway,
// with the invalid state converted to -1 in the intent.
// Better to log an error and not send an intent with
// invalid contents or set mAdapterConnectionState to -1.
errorLog("sendConnectionStateChange: invalid state transition " + prevState + " -> "
+ state);
return;
}
synchronized (mObject) {
updateProfileConnectionState(profile, state, prevState);
if (updateCountersAndCheckForConnectionStateChange(state, prevState)) {
int newAdapterState = convertToAdapterState(state);
int prevAdapterState = convertToAdapterState(prevState);
setConnectionState(newAdapterState);
Intent intent = new Intent(BluetoothAdapter.ACTION_CONNECTION_STATE_CHANGED);
intent.putExtra(BluetoothDevice.EXTRA_DEVICE, device);
intent.putExtra(BluetoothAdapter.EXTRA_CONNECTION_STATE, newAdapterState);
intent.putExtra(BluetoothAdapter.EXTRA_PREVIOUS_CONNECTION_STATE, prevAdapterState);
intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
Log.d(TAG, "ADAPTER_CONNECTION_STATE_CHANGE: " + device + ": " + prevAdapterState
+ " -> " + newAdapterState);
if (!isNormalStateTransition(prevState, state)) {
Log.w(TAG, "ADAPTER_CONNECTION_STATE_CHANGE: unexpected transition for profile="
+ profile + ", device=" + device + ", " + prevState + " -> " + state);
}
mService.sendBroadcastAsUser(intent, UserHandle.ALL, BLUETOOTH_CONNECT,
Utils.getTempAllowlistBroadcastOptions());
}
}
}
private boolean validateProfileConnectionState(int state) {
return (state == BluetoothProfile.STATE_DISCONNECTED
|| state == BluetoothProfile.STATE_CONNECTING
|| state == BluetoothProfile.STATE_CONNECTED
|| state == BluetoothProfile.STATE_DISCONNECTING);
}
private static int convertToAdapterState(int state) {
switch (state) {
case BluetoothProfile.STATE_DISCONNECTED:
return BluetoothAdapter.STATE_DISCONNECTED;
case BluetoothProfile.STATE_DISCONNECTING:
return BluetoothAdapter.STATE_DISCONNECTING;
case BluetoothProfile.STATE_CONNECTED:
return BluetoothAdapter.STATE_CONNECTED;
case BluetoothProfile.STATE_CONNECTING:
return BluetoothAdapter.STATE_CONNECTING;
}
Log.e(TAG, "convertToAdapterState, unknow state " + state);
return -1;
}
private static boolean isNormalStateTransition(int prevState, int nextState) {
switch (prevState) {
case BluetoothProfile.STATE_DISCONNECTED:
return nextState == BluetoothProfile.STATE_CONNECTING;
case BluetoothProfile.STATE_CONNECTED:
return nextState == BluetoothProfile.STATE_DISCONNECTING;
case BluetoothProfile.STATE_DISCONNECTING:
case BluetoothProfile.STATE_CONNECTING:
return (nextState == BluetoothProfile.STATE_DISCONNECTED) || (nextState
== BluetoothProfile.STATE_CONNECTED);
default:
return false;
}
}
private boolean updateCountersAndCheckForConnectionStateChange(int state, int prevState) {
switch (prevState) {
case BluetoothProfile.STATE_CONNECTING:
if (mProfilesConnecting > 0) {
mProfilesConnecting--;
} else {
Log.e(TAG, "mProfilesConnecting " + mProfilesConnecting);
throw new IllegalStateException(
"Invalid state transition, " + prevState + " -> " + state);
}
break;
case BluetoothProfile.STATE_CONNECTED:
if (mProfilesConnected > 0) {
mProfilesConnected--;
} else {
Log.e(TAG, "mProfilesConnected " + mProfilesConnected);
throw new IllegalStateException(
"Invalid state transition, " + prevState + " -> " + state);
}
break;
case BluetoothProfile.STATE_DISCONNECTING:
if (mProfilesDisconnecting > 0) {
mProfilesDisconnecting--;
} else {
Log.e(TAG, "mProfilesDisconnecting " + mProfilesDisconnecting);
throw new IllegalStateException(
"Invalid state transition, " + prevState + " -> " + state);
}
break;
}
switch (state) {
case BluetoothProfile.STATE_CONNECTING:
mProfilesConnecting++;
return (mProfilesConnected == 0 && mProfilesConnecting == 1);
case BluetoothProfile.STATE_CONNECTED:
mProfilesConnected++;
return (mProfilesConnected == 1);
case BluetoothProfile.STATE_DISCONNECTING:
mProfilesDisconnecting++;
return (mProfilesConnected == 0 && mProfilesDisconnecting == 1);
case BluetoothProfile.STATE_DISCONNECTED:
return (mProfilesConnected == 0 && mProfilesConnecting == 0);
default:
return true;
}
}
private void updateProfileConnectionState(int profile, int newState, int oldState) {
// mProfileConnectionState is a hashmap -
// <Integer, Pair<Integer, Integer>>
// The key is the profile, the value is a pair. first element
// is the state and the second element is the number of devices
// in that state.
int numDev = 1;
int newHashState = newState;
boolean update = true;
// The following conditions are considered in this function:
// 1. If there is no record of profile and state - update
// 2. If a new device's state is current hash state - increment
// number of devices in the state.
// 3. If a state change has happened to Connected or Connecting
// (if current state is not connected), update.
// 4. If numDevices is 1 and that device state is being updated, update
// 5. If numDevices is > 1 and one of the devices is changing state,
// decrement numDevices but maintain oldState if it is Connected or
// Connecting
Pair<Integer, Integer> stateNumDev = mProfileConnectionState.get(profile);
if (stateNumDev != null) {
int currHashState = stateNumDev.first;
numDev = stateNumDev.second;
if (newState == currHashState) {
numDev++;
} else if (newState == BluetoothProfile.STATE_CONNECTED || (
newState == BluetoothProfile.STATE_CONNECTING
&& currHashState != BluetoothProfile.STATE_CONNECTED)) {
numDev = 1;
} else if (numDev == 1 && oldState == currHashState) {
update = true;
} else if (numDev > 1 && oldState == currHashState) {
numDev--;
if (currHashState == BluetoothProfile.STATE_CONNECTED
|| currHashState == BluetoothProfile.STATE_CONNECTING) {
newHashState = currHashState;
}
} else {
update = false;
}
}
if (update) {
mProfileConnectionState.put(profile, new Pair<Integer, Integer>(newHashState, numDev));
invalidateGetProfileConnectionStateCache();
}
}
void updateAllowlistedMediaPlayers(String playername) {
Log.d(TAG, "updateAllowlistedMediaPlayers ");
if (!mAllowlistedPlayers.contains(playername)) {
Log.d(TAG, "Adding to Allowlisted Players list:" + playername);
mAllowlistedPlayers.add(playername);
}
}
@RequiresPermission(android.Manifest.permission.INTERACT_ACROSS_USERS)
void adapterPropertyChangedCallback(int[] types, byte[][] values) {
Intent intent;
int type;
byte[] val;
for (int i = 0; i < types.length; i++) {
val = values[i];
type = types[i];
infoLog("adapterPropertyChangedCallback with type:" + type + " len:" + val.length);
synchronized (mObject) {
switch (type) {
case AbstractionLayer.BT_PROPERTY_BDNAME:
mName = new String(val);
intent = new Intent(BluetoothAdapter.ACTION_LOCAL_NAME_CHANGED);
intent.putExtra(BluetoothAdapter.EXTRA_LOCAL_NAME, mName);
intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
mService.sendBroadcastAsUser(intent, UserHandle.ALL,
BLUETOOTH_CONNECT, Utils.getTempAllowlistBroadcastOptions());
debugLog("Name is: " + mName);
break;
case AbstractionLayer.BT_PROPERTY_BDADDR:
mAddress = val;
String address = Utils.getAddressStringFromByte(mAddress);
intent = new Intent(BluetoothAdapter.ACTION_BLUETOOTH_ADDRESS_CHANGED);
intent.putExtra(BluetoothAdapter.EXTRA_BLUETOOTH_ADDRESS, address);
intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
mService.sendBroadcastAsUser(intent, UserHandle.ALL,
BLUETOOTH_CONNECT, Utils.getTempAllowlistBroadcastOptions());
break;
case AbstractionLayer.BT_PROPERTY_CLASS_OF_DEVICE:
if (val == null || val.length != 3) {
debugLog("Invalid BT CoD value from stack.");
return;
}
int bluetoothClass =
((int) val[0] << 16) + ((int) val[1] << 8) + (int) val[2];
if (bluetoothClass != 0) {
mBluetoothClass = new BluetoothClass(bluetoothClass);
}
debugLog("BT Class:" + mBluetoothClass);
break;
case AbstractionLayer.BT_PROPERTY_ADAPTER_SCAN_MODE:
int mode = Utils.byteArrayToInt(val, 0);
mScanMode = AdapterService.convertScanModeFromHal(mode);
intent = new Intent(BluetoothAdapter.ACTION_SCAN_MODE_CHANGED);
intent.putExtra(BluetoothAdapter.EXTRA_SCAN_MODE, mScanMode);
intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
Utils.sendBroadcast(mService, intent, BLUETOOTH_SCAN,
Utils.getTempAllowlistBroadcastOptions());
debugLog("Scan Mode:" + mScanMode);
break;
case AbstractionLayer.BT_PROPERTY_UUIDS:
mUuids = Utils.byteArrayToUuid(val);
break;
case AbstractionLayer.BT_PROPERTY_ADAPTER_BONDED_DEVICES:
int number = val.length / BD_ADDR_LEN;
byte[] addrByte = new byte[BD_ADDR_LEN];
for (int j = 0; j < number; j++) {
System.arraycopy(val, j * BD_ADDR_LEN, addrByte, 0, BD_ADDR_LEN);
onBondStateChanged(mAdapter.getRemoteDevice(
Utils.getAddressStringFromByte(addrByte)),
BluetoothDevice.BOND_BONDED);
}
break;
case AbstractionLayer.BT_PROPERTY_ADAPTER_DISCOVERABLE_TIMEOUT:
mDiscoverableTimeout = Utils.byteArrayToInt(val, 0);
debugLog("Discoverable Timeout:" + mDiscoverableTimeout);
break;
case AbstractionLayer.BT_PROPERTY_LOCAL_LE_FEATURES:
updateFeatureSupport(val);
mService.updateLeAudioProfileServiceState();
break;
case AbstractionLayer.BT_PROPERTY_DYNAMIC_AUDIO_BUFFER:
updateDynamicAudioBufferSupport(val);
break;
case AbstractionLayer.BT_PROPERTY_LOCAL_IO_CAPS:
mLocalIOCapability = Utils.byteArrayToInt(val);
debugLog("mLocalIOCapability set to " + mLocalIOCapability);
break;
case AbstractionLayer.BT_PROPERTY_WL_MEDIA_PLAYERS_LIST:
int pos = 0;
for (int j = 0; j < val.length; j++) {
if (val[j] != 0) {
continue;
}
int name_len = j - pos;
byte[] buf = new byte[name_len];
System.arraycopy(val, pos, buf, 0, name_len);
String player_name = new String(buf, 0, name_len);
Log.d(TAG, "player_name: " + player_name);
updateAllowlistedMediaPlayers(player_name);
pos += (name_len + 1);
}
break;
default:
errorLog("Property change not handled in Java land:" + type);
}
}
}
}
private void updateFeatureSupport(byte[] val) {
mVersSupported = ((0xFF & ((int) val[1])) << 8) + (0xFF & ((int) val[0]));
mNumOfAdvertisementInstancesSupported = (0xFF & ((int) val[3]));
mRpaOffloadSupported = ((0xFF & ((int) val[4])) != 0);
mNumOfOffloadedIrkSupported = (0xFF & ((int) val[5]));
mNumOfOffloadedScanFilterSupported = (0xFF & ((int) val[6]));
mIsActivityAndEnergyReporting = ((0xFF & ((int) val[7])) != 0);
mOffloadedScanResultStorageBytes = ((0xFF & ((int) val[9])) << 8) + (0xFF & ((int) val[8]));
mTotNumOfTrackableAdv = ((0xFF & ((int) val[11])) << 8) + (0xFF & ((int) val[10]));
mIsExtendedScanSupported = ((0xFF & ((int) val[12])) != 0);
mIsDebugLogSupported = ((0xFF & ((int) val[13])) != 0);
mIsLe2MPhySupported = ((0xFF & ((int) val[14])) != 0);
mIsLeCodedPhySupported = ((0xFF & ((int) val[15])) != 0);
mIsLeExtendedAdvertisingSupported = ((0xFF & ((int) val[16])) != 0);
mIsLePeriodicAdvertisingSupported = ((0xFF & ((int) val[17])) != 0);
mLeMaximumAdvertisingDataLength =
(0xFF & ((int) val[18])) + ((0xFF & ((int) val[19])) << 8);
mDynamicAudioBufferSizeSupportedCodecsGroup1 =
((0xFF & ((int) val[21])) << 8) + (0xFF & ((int) val[20]));
mDynamicAudioBufferSizeSupportedCodecsGroup2 =
((0xFF & ((int) val[23])) << 8) + (0xFF & ((int) val[22]));
mIsLePeriodicAdvertisingSyncTransferSenderSupported = ((0xFF & ((int) val[24])) != 0);
mIsLeConnectedIsochronousStreamCentralSupported = ((0xFF & ((int) val[25])) != 0);
mIsLeIsochronousBroadcasterSupported = ((0xFF & ((int) val[26])) != 0);
mIsLePeriodicAdvertisingSyncTransferRecipientSupported = ((0xFF & ((int) val[27])) != 0);
mIsOffloadedTransportDiscoveryDataScanSupported = ((0x01 & ((int) val[28])) != 0);
Log.d(TAG, "BT_PROPERTY_LOCAL_LE_FEATURES: update from BT controller"
+ " mNumOfAdvertisementInstancesSupported = "
+ mNumOfAdvertisementInstancesSupported + " mRpaOffloadSupported = "
+ mRpaOffloadSupported + " mNumOfOffloadedIrkSupported = "
+ mNumOfOffloadedIrkSupported + " mNumOfOffloadedScanFilterSupported = "
+ mNumOfOffloadedScanFilterSupported + " mOffloadedScanResultStorageBytes= "
+ mOffloadedScanResultStorageBytes + " mIsActivityAndEnergyReporting = "
+ mIsActivityAndEnergyReporting + " mVersSupported = " + mVersSupported
+ " mTotNumOfTrackableAdv = " + mTotNumOfTrackableAdv
+ " mIsExtendedScanSupported = " + mIsExtendedScanSupported
+ " mIsDebugLogSupported = " + mIsDebugLogSupported + " mIsLe2MPhySupported = "
+ mIsLe2MPhySupported + " mIsLeCodedPhySupported = " + mIsLeCodedPhySupported
+ " mIsLeExtendedAdvertisingSupported = " + mIsLeExtendedAdvertisingSupported
+ " mIsLePeriodicAdvertisingSupported = " + mIsLePeriodicAdvertisingSupported
+ " mLeMaximumAdvertisingDataLength = " + mLeMaximumAdvertisingDataLength
+ " mDynamicAudioBufferSizeSupportedCodecsGroup1 = "
+ mDynamicAudioBufferSizeSupportedCodecsGroup1
+ " mDynamicAudioBufferSizeSupportedCodecsGroup2 = "
+ mDynamicAudioBufferSizeSupportedCodecsGroup2
+ " mIsLePeriodicAdvertisingSyncTransferSenderSupported = "
+ mIsLePeriodicAdvertisingSyncTransferSenderSupported
+ " mIsLeConnectedIsochronousStreamCentralSupported = "
+ mIsLeConnectedIsochronousStreamCentralSupported
+ " mIsLeIsochronousBroadcasterSupported = "
+ mIsLeIsochronousBroadcasterSupported
+ " mIsLePeriodicAdvertisingSyncTransferRecipientSupported = "
+ mIsLePeriodicAdvertisingSyncTransferRecipientSupported
+ " mIsOffloadedTransportDiscoveryDataScanSupported = "
+ mIsOffloadedTransportDiscoveryDataScanSupported);
invalidateIsOffloadedFilteringSupportedCache();
}
private void updateDynamicAudioBufferSupport(byte[] val) {
// bufferConstraints is the table indicates the capability of all the codecs
// with buffer time. The raw is codec number, and the column is buffer type. There are 3
// buffer types - default/maximum/minimum.
// The maximum number of raw is BUFFER_CODEC_MAX_NUM(32).
// The maximum number of column is BUFFER_TYPE_MAX(3).
// The array element indicates the buffer time, the size is two octet.
mBufferConstraintList = new ArrayList<BufferConstraint>();
for (int i = 0; i < BufferConstraints.BUFFER_CODEC_MAX_NUM; i++) {
int defaultBufferTime = ((0xFF & ((int) val[i * 6 + 1])) << 8)
+ (0xFF & ((int) val[i * 6]));
int maximumBufferTime = ((0xFF & ((int) val[i * 6 + 3])) << 8)
+ (0xFF & ((int) val[i * 6 + 2]));
int minimumBufferTime = ((0xFF & ((int) val[i * 6 + 5])) << 8)
+ (0xFF & ((int) val[i * 6 + 4]));
BufferConstraint bufferConstraint = new BufferConstraint(defaultBufferTime,
maximumBufferTime, minimumBufferTime);
mBufferConstraintList.add(bufferConstraint);
}
}
void onBluetoothReady() {
debugLog("onBluetoothReady, state=" + BluetoothAdapter.nameForState(getState())
+ ", ScanMode=" + mScanMode);
synchronized (mObject) {
// Reset adapter and profile connection states
setConnectionState(BluetoothAdapter.STATE_DISCONNECTED);
mProfileConnectionState.clear();
invalidateGetProfileConnectionStateCache();
mProfilesConnected = 0;
mProfilesConnecting = 0;
mProfilesDisconnecting = 0;
// adapterPropertyChangedCallback has already been received. Set the scan mode.
setScanMode(BluetoothAdapter.SCAN_MODE_CONNECTABLE);
// This keeps NV up-to date on first-boot after flash.
setDiscoverableTimeout(mDiscoverableTimeout);
}
}
void onBleDisable() {
// Sequence BLE_ON to STATE_OFF - that is _complete_ OFF state.
debugLog("onBleDisable");
// Set the scan_mode to NONE (no incoming connections).
setScanMode(BluetoothAdapter.SCAN_MODE_NONE);
}
void discoveryStateChangeCallback(int state) {
infoLog("Callback:discoveryStateChangeCallback with state:" + state);
synchronized (mObject) {
Intent intent;
if (state == AbstractionLayer.BT_DISCOVERY_STOPPED) {
mDiscovering = false;
mService.clearDiscoveringPackages();
mDiscoveryEndMs = System.currentTimeMillis();
intent = new Intent(BluetoothAdapter.ACTION_DISCOVERY_FINISHED);
Utils.sendBroadcast(mService, intent, BLUETOOTH_SCAN,
getBroadcastOptionsForDiscoveryFinished());
} else if (state == AbstractionLayer.BT_DISCOVERY_STARTED) {
mDiscovering = true;
mDiscoveryEndMs = System.currentTimeMillis() + DEFAULT_DISCOVERY_TIMEOUT_MS;
intent = new Intent(BluetoothAdapter.ACTION_DISCOVERY_STARTED);
Utils.sendBroadcast(mService, intent, BLUETOOTH_SCAN,
Utils.getTempAllowlistBroadcastOptions());
}
}
}
/**
* @return broadcast options for ACTION_DISCOVERY_FINISHED broadcast
*/
private static @NonNull Bundle getBroadcastOptionsForDiscoveryFinished() {
final BroadcastOptions options = Utils.getTempBroadcastOptions();
if (SdkLevel.isAtLeastU()) {
options.setDeliveryGroupPolicy(BroadcastOptions.DELIVERY_GROUP_POLICY_MOST_RECENT);
options.setDeferralPolicy(BroadcastOptions.DEFERRAL_POLICY_UNTIL_ACTIVE);
}
return options.toBundle();
}
@RequiresPermission(android.Manifest.permission.BLUETOOTH_CONNECT)
protected void dump(FileDescriptor fd, PrintWriter writer, String[] args) {
writer.println(TAG);
writer.println(" " + "Name: " + getName());
writer.println(" " + "Address: " + Utils.getAddressStringFromByte(mAddress));
writer.println(" " + "ScanMode: " + dumpScanMode(getScanMode()));
writer.println(" " + "ConnectionState: " + dumpConnectionState(getConnectionState()));
writer.println(" " + "State: " + BluetoothAdapter.nameForState(getState()));
writer.println(" " + "MaxConnectedAudioDevices: " + getMaxConnectedAudioDevices());
writer.println(" " + "A2dpOffloadEnabled: " + mA2dpOffloadEnabled);
writer.println(" " + "Discovering: " + mDiscovering);
writer.println(" " + "DiscoveryEndMs: " + mDiscoveryEndMs);
writer.println(" " + "Bonded devices:");
StringBuilder sb = new StringBuilder();
for (BluetoothDevice device : mBondedDevices) {
String address = device.getAddress();
String identityAddress = mService.getIdentityAddress(address);
if (identityAddress.equals(address)) {
writer.println(" " + address
+ " [" + dumpDeviceType(device.getType()) + "] "
+ Utils.getName(device));
} else {
sb.append(" " + address + " => " + identityAddress
+ " [" + dumpDeviceType(device.getType()) + "] "
+ Utils.getName(device) + "\n");
}
}
writer.println(sb.toString());
writer.println(" " + "Scan Mode Changes:");
for (String log : mScanModeChanges) {
writer.println(" " + log);
}
}
private String dumpDeviceType(int deviceType) {
switch (deviceType) {
case BluetoothDevice.DEVICE_TYPE_UNKNOWN:
return " ???? ";
case BluetoothDevice.DEVICE_TYPE_CLASSIC:
return "BR/EDR";
case BluetoothDevice.DEVICE_TYPE_LE:
return " LE ";
case BluetoothDevice.DEVICE_TYPE_DUAL:
return " DUAL ";
default:
return "Invalid device type: " + deviceType;
}
}
private String dumpConnectionState(int state) {
switch (state) {
case BluetoothAdapter.STATE_DISCONNECTED:
return "STATE_DISCONNECTED";
case BluetoothAdapter.STATE_DISCONNECTING:
return "STATE_DISCONNECTING";
case BluetoothAdapter.STATE_CONNECTING:
return "STATE_CONNECTING";
case BluetoothAdapter.STATE_CONNECTED:
return "STATE_CONNECTED";
default:
return "Unknown Connection State " + state;
}
}
private String dumpScanMode(int scanMode) {
switch (scanMode) {
case BluetoothAdapter.SCAN_MODE_NONE:
return "SCAN_MODE_NONE";
case BluetoothAdapter.SCAN_MODE_CONNECTABLE:
return "SCAN_MODE_CONNECTABLE";
case BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE:
return "SCAN_MODE_CONNECTABLE_DISCOVERABLE";
default:
return "Unknown Scan Mode " + scanMode;
}
}
private static void infoLog(String msg) {
if (VDBG) {
Log.i(TAG, msg);
}
}
private static void debugLog(String msg) {
if (DBG) {
Log.d(TAG, msg);
}
}
private static void errorLog(String msg) {
Log.e(TAG, msg);
}
}