/* * Copyright (C) 2012 The Android Open Source Project * * 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 android.bluetooth.BluetoothAdapter; import android.bluetooth.BluetoothDevice; import android.bluetooth.BluetoothProfile; import android.content.Context; import android.content.Intent; import android.os.ParcelUuid; import android.os.UserHandle; import android.util.Log; import android.util.Pair; import com.android.bluetooth.Utils; import com.android.bluetooth.btservice.RemoteDevices.DeviceProperties; import java.util.HashMap; import java.util.ArrayList; class AdapterProperties { private static final boolean DBG = true; private static final boolean VDBG = false; private static final String TAG = "BluetoothAdapterProperties"; private static final int BD_ADDR_LEN = 6; // 6 bytes private String mName; private byte[] mAddress; private int mBluetoothClass; private int mScanMode; private int mDiscoverableTimeout; private ParcelUuid[] mUuids; private ArrayList mBondedDevices = new ArrayList(); private int mProfilesConnecting, mProfilesConnected, mProfilesDisconnecting; private HashMap> mProfileConnectionState; private int mConnectionState = BluetoothAdapter.STATE_DISCONNECTED; private int mState = BluetoothAdapter.STATE_OFF; private AdapterService mService; private boolean mDiscovering; private RemoteDevices mRemoteDevices; private BluetoothAdapter mAdapter; // Lock for all getters and setters. // If finer grained locking is needer, more locks // can be added here. private Object mObject = new Object(); public AdapterProperties(AdapterService service) { mService = service; mAdapter = BluetoothAdapter.getDefaultAdapter(); } public void init(RemoteDevices remoteDevices) { if (mProfileConnectionState ==null) { mProfileConnectionState = new HashMap>(); } else { mProfileConnectionState.clear(); } mRemoteDevices = remoteDevices; } public void cleanup() { mRemoteDevices = null; if (mProfileConnectionState != null) { mProfileConnectionState.clear(); mProfileConnectionState = null; } mService = null; if (!mBondedDevices.isEmpty()) mBondedDevices.clear(); } public Object Clone() throws CloneNotSupportedException { throw new CloneNotSupportedException(); } /** * @return the mName */ String getName() { synchronized (mObject) { return mName; } } /** * Set the local adapter property - name * @param name the name to set */ boolean setName(String name) { synchronized (mObject) { return mService.setAdapterPropertyNative( AbstractionLayer.BT_PROPERTY_BDNAME, name.getBytes()); } } /** * @return the mClass */ int getBluetoothClass() { synchronized (mObject) { return mBluetoothClass; } } /** * @return the mScanMode */ int getScanMode() { synchronized (mObject) { return mScanMode; } } /** * Set the local adapter property - scanMode * * @param scanMode the ScanMode to set */ boolean setScanMode(int scanMode) { synchronized (mObject) { return mService.setAdapterPropertyNative( AbstractionLayer.BT_PROPERTY_ADAPTER_SCAN_MODE, Utils.intToByteArray(scanMode)); } } /** * @return the mUuids */ ParcelUuid[] getUuids() { synchronized (mObject) { return mUuids; } } /** * Set local adapter UUIDs. * * @param uuids the uuids to be set. */ boolean setUuids(ParcelUuid[] uuids) { synchronized (mObject) { return mService.setAdapterPropertyNative( AbstractionLayer.BT_PROPERTY_UUIDS, Utils.uuidsToByteArray(uuids)); } } /** * @return the mAddress */ byte[] getAddress() { synchronized (mObject) { return mAddress; } } /** * @param mConnectionState the mConnectionState to set */ void setConnectionState(int mConnectionState) { synchronized (mObject) { this.mConnectionState = mConnectionState; } } /** * @return the mConnectionState */ int getConnectionState() { synchronized (mObject) { return mConnectionState; } } /** * @param mState the mState to set */ void setState(int mState) { synchronized (mObject) { debugLog("Setting state to " + mState); this.mState = mState; } } /** * @return the mState */ int getState() { synchronized (mObject) { if (VDBG) debugLog("State = " + mState); return mState; } } /** * @return the mBondedDevices */ BluetoothDevice[] getBondedDevices() { BluetoothDevice[] bondedDeviceList = new BluetoothDevice[0]; synchronized (mObject) { if(mBondedDevices.isEmpty()) return (new BluetoothDevice[0]); try { bondedDeviceList = mBondedDevices.toArray(bondedDeviceList); debugLog("getBondedDevices: length="+bondedDeviceList.length); return bondedDeviceList; } catch(ArrayStoreException ee) { errorLog("Error retrieving bonded device array"); return (new BluetoothDevice[0]); } } } // This function shall be invoked from BondStateMachine whenever the bond // state changes. void onBondStateChanged(BluetoothDevice device, int state) { if(device == null) return; try { byte[] addrByte = Utils.getByteAddress(device); DeviceProperties prop = mRemoteDevices.getDeviceProperties(device); if (prop == null) prop = mRemoteDevices.addDeviceProperties(addrByte); 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); } } 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); } } catch(Exception ee) { Log.e(TAG, "Exception in onBondStateChanged : ", ee); } } int getDiscoverableTimeout() { synchronized (mObject) { return mDiscoverableTimeout; } } boolean setDiscoverableTimeout(int timeout) { synchronized (mObject) { return mService.setAdapterPropertyNative( AbstractionLayer.BT_PROPERTY_ADAPTER_DISCOVERABLE_TIMEOUT, Utils.intToByteArray(timeout)); } } int getProfileConnectionState(int profile) { synchronized (mObject) { Pair p = mProfileConnectionState.get(profile); if (p != null) return p.first; return BluetoothProfile.STATE_DISCONNECTED; } } boolean isDiscovering() { synchronized (mObject) { return mDiscovering; } } 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("Error in sendConnectionStateChange: " + "prevState " + prevState + " state " + state); return; } synchronized (mObject) { updateProfileConnectionState(profile, state, prevState); if (updateCountersAndCheckForConnectionStateChange(state, prevState)) { setConnectionState(state); Intent intent = new Intent(BluetoothAdapter.ACTION_CONNECTION_STATE_CHANGED); intent.putExtra(BluetoothDevice.EXTRA_DEVICE, device); intent.putExtra(BluetoothAdapter.EXTRA_CONNECTION_STATE, convertToAdapterState(state)); intent.putExtra(BluetoothAdapter.EXTRA_PREVIOUS_CONNECTION_STATE, convertToAdapterState(prevState)); intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT); mService.sendBroadcastAsUser(intent, UserHandle.ALL, mService.BLUETOOTH_PERM); Log.d(TAG, "CONNECTION_STATE_CHANGE: " + device + ": " + prevState + " -> " + state); } } } private boolean validateProfileConnectionState(int state) { return (state == BluetoothProfile.STATE_DISCONNECTED || state == BluetoothProfile.STATE_CONNECTING || state == BluetoothProfile.STATE_CONNECTED || state == BluetoothProfile.STATE_DISCONNECTING); } private 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, "Error in convertToAdapterState"); return -1; } private boolean updateCountersAndCheckForConnectionStateChange(int state, int prevState) { switch (prevState) { case BluetoothProfile.STATE_CONNECTING: mProfilesConnecting--; break; case BluetoothProfile.STATE_CONNECTED: mProfilesConnected--; break; case BluetoothProfile.STATE_DISCONNECTING: mProfilesDisconnecting--; 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 - // > // 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 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(newHashState, numDev)); } } 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, mService.BLUETOOTH_PERM); debugLog("Name is: " + mName); break; case AbstractionLayer.BT_PROPERTY_BDADDR: mAddress = val; debugLog("Address is:" + Utils.getAddressStringFromByte(mAddress)); break; case AbstractionLayer.BT_PROPERTY_CLASS_OF_DEVICE: mBluetoothClass = Utils.byteArrayToInt(val, 0); debugLog("BT Class:" + mBluetoothClass); break; case AbstractionLayer.BT_PROPERTY_ADAPTER_SCAN_MODE: int mode = Utils.byteArrayToInt(val, 0); mScanMode = mService.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); mService.sendBroadcast(intent, mService.BLUETOOTH_PERM); debugLog("Scan Mode:" + mScanMode); if (mBluetoothDisabling) { mBluetoothDisabling=false; mService.startBluetoothDisable(); } 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; default: errorLog("Property change not handled in Java land:" + type); } } } } void onBluetoothReady() { Log.d(TAG, "ScanMode = " + mScanMode ); Log.d(TAG, "State = " + getState() ); // When BT is being turned on, all adapter properties will be sent in 1 // callback. At this stage, set the scan mode. synchronized (mObject) { if (getState() == BluetoothAdapter.STATE_TURNING_ON && mScanMode == BluetoothAdapter.SCAN_MODE_NONE) { /* mDiscoverableTimeout is part of the adapterPropertyChangedCallback received before onBluetoothReady */ if (mDiscoverableTimeout != 0) setScanMode(AbstractionLayer.BT_SCAN_MODE_CONNECTABLE); else /* if timeout == never (0) at startup */ setScanMode(AbstractionLayer.BT_SCAN_MODE_CONNECTABLE_DISCOVERABLE); /* though not always required, this keeps NV up-to date on first-boot after flash */ setDiscoverableTimeout(mDiscoverableTimeout); } } } private boolean mBluetoothDisabling=false; void onBluetoothDisable() { // When BT disable is invoked, set the scan_mode to NONE // so no incoming connections are possible //Set flag to indicate we are disabling. When property change of scan mode done //continue with disable sequence debugLog("onBluetoothDisable()"); mBluetoothDisabling = true; if (getState() == BluetoothAdapter.STATE_TURNING_OFF) { setScanMode(AbstractionLayer.BT_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; intent = new Intent(BluetoothAdapter.ACTION_DISCOVERY_FINISHED); mService.sendBroadcast(intent, mService.BLUETOOTH_PERM); } else if (state == AbstractionLayer.BT_DISCOVERY_STARTED) { mDiscovering = true; intent = new Intent(BluetoothAdapter.ACTION_DISCOVERY_STARTED); mService.sendBroadcast(intent, mService.BLUETOOTH_PERM); } } } private void infoLog(String msg) { if (DBG) Log.i(TAG, msg); } private void debugLog(String msg) { if (DBG) Log.d(TAG, msg); } private void errorLog(String msg) { Log.e(TAG, msg); } }