1ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh/*
2ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu * Copyright (C) 2012 The Android Open Source Project
3ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu *
4ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu * Licensed under the Apache License, Version 2.0 (the "License");
5ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu * you may not use this file except in compliance with the License.
6ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu * You may obtain a copy of the License at
7ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu *
8ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu *      http://www.apache.org/licenses/LICENSE-2.0
9ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu *
10ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu * Unless required by applicable law or agreed to in writing, software
11ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu * distributed under the License is distributed on an "AS IS" BASIS,
12ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu * See the License for the specific language governing permissions and
14ede67c26e7b2564ea35db6d9b3027a269c150e13Zhihai Xu * limitations under the License.
15ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh */
16ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
17ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshpackage com.android.bluetooth.btservice;
18ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
19ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport android.bluetooth.BluetoothAdapter;
20ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport android.bluetooth.BluetoothDevice;
21ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport android.bluetooth.BluetoothProfile;
22ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport android.content.Context;
23ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport android.content.Intent;
24ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport android.os.ParcelUuid;
25bd704c741b8c523ad747214f6f0520ac3e2caf8fZhihai Xuimport android.os.UserHandle;
26ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport android.util.Log;
27ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport android.util.Pair;
28ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
29ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport com.android.bluetooth.Utils;
30ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport com.android.bluetooth.btservice.RemoteDevices.DeviceProperties;
31ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
32ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshimport java.util.HashMap;
332f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajanimport java.util.ArrayList;
34004550da492fb000ab08b6324558c1e4cbd383d4Srinu Jellaimport java.util.concurrent.CopyOnWriteArrayList;
35ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
36ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganeshclass AdapterProperties {
3734e323baafb5563c3874f95847ec544faf6923f8Matthew Xie    private static final boolean DBG = true;
3834e323baafb5563c3874f95847ec544faf6923f8Matthew Xie    private static final boolean VDBG = false;
39ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private static final String TAG = "BluetoothAdapterProperties";
40ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
41ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private static final int BD_ADDR_LEN = 6; // 6 bytes
42ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private String mName;
43ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private byte[] mAddress;
44ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private int mBluetoothClass;
45ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private int mScanMode;
46ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private int mDiscoverableTimeout;
47ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private ParcelUuid[] mUuids;
48004550da492fb000ab08b6324558c1e4cbd383d4Srinu Jella    private CopyOnWriteArrayList<BluetoothDevice> mBondedDevices = new CopyOnWriteArrayList<BluetoothDevice>();
49ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
50ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private int mProfilesConnecting, mProfilesConnected, mProfilesDisconnecting;
51ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private HashMap<Integer, Pair<Integer, Integer>> mProfileConnectionState;
52ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
53ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
54ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private int mConnectionState = BluetoothAdapter.STATE_DISCONNECTED;
55ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private int mState = BluetoothAdapter.STATE_OFF;
56ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
57ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private AdapterService mService;
58ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private boolean mDiscovering;
5931ba132491053bc86d419a7d51fc04af3299c076fredc    private RemoteDevices mRemoteDevices;
602f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan    private BluetoothAdapter mAdapter;
61c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    //TODO - all hw capabilities to be exposed as a class
62c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    private int mNumOfAdvertisementInstancesSupported;
63c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    private boolean mRpaOffloadSupported;
64c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    private int mNumOfOffloadedIrkSupported;
65c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    private int mNumOfOffloadedScanFilterSupported;
66c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    private int mOffloadedScanResultStorageBytes;
67d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham    private boolean mIsActivityAndEnergyReporting;
68ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
69ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    // Lock for all getters and setters.
70ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    // If finer grained locking is needer, more locks
71ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    // can be added here.
72ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private Object mObject = new Object();
73ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
7474ae04c73312403e89db0f8e9bd9601d403b4783fredc    public AdapterProperties(AdapterService service) {
75ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        mService = service;
762f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan        mAdapter = BluetoothAdapter.getDefaultAdapter();
77ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
7831ba132491053bc86d419a7d51fc04af3299c076fredc    public void init(RemoteDevices remoteDevices) {
7931ba132491053bc86d419a7d51fc04af3299c076fredc        if (mProfileConnectionState ==null) {
8031ba132491053bc86d419a7d51fc04af3299c076fredc            mProfileConnectionState = new HashMap<Integer, Pair<Integer, Integer>>();
816654f5c903de510a70f9e72cd5ad7837b615d93ffredc        } else {
8231ba132491053bc86d419a7d51fc04af3299c076fredc            mProfileConnectionState.clear();
836654f5c903de510a70f9e72cd5ad7837b615d93ffredc        }
8431ba132491053bc86d419a7d51fc04af3299c076fredc        mRemoteDevices = remoteDevices;
85ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
86ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
8731ba132491053bc86d419a7d51fc04af3299c076fredc    public void cleanup() {
8874ae04c73312403e89db0f8e9bd9601d403b4783fredc        mRemoteDevices = null;
8931ba132491053bc86d419a7d51fc04af3299c076fredc        if (mProfileConnectionState != null) {
9031ba132491053bc86d419a7d51fc04af3299c076fredc            mProfileConnectionState.clear();
9131ba132491053bc86d419a7d51fc04af3299c076fredc            mProfileConnectionState = null;
9231ba132491053bc86d419a7d51fc04af3299c076fredc        }
9331ba132491053bc86d419a7d51fc04af3299c076fredc        mService = null;
942f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan        if (!mBondedDevices.isEmpty())
952f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            mBondedDevices.clear();
966654f5c903de510a70f9e72cd5ad7837b615d93ffredc    }
976654f5c903de510a70f9e72cd5ad7837b615d93ffredc
983a91ac3eefe23a22aa0abc1a89ac253a2b0dbae9Nick Kralevich    @Override
993a91ac3eefe23a22aa0abc1a89ac253a2b0dbae9Nick Kralevich    public Object clone() throws CloneNotSupportedException {
100ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        throw new CloneNotSupportedException();
101ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
102ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
103ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
104ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @return the mName
105ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
106ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    String getName() {
107ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
108ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mName;
109ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
110ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
111ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
112ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
113ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * Set the local adapter property - name
114ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @param name the name to set
115ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
116ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    boolean setName(String name) {
117ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
118ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mService.setAdapterPropertyNative(
119ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    AbstractionLayer.BT_PROPERTY_BDNAME, name.getBytes());
120ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
121ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
122ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
123ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
124ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @return the mClass
125ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
126ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    int getBluetoothClass() {
127ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
128ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mBluetoothClass;
129ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
130ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
131ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
132ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
133ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @return the mScanMode
134ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
135ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    int getScanMode() {
136ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
137ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mScanMode;
138ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
139ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
140ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
141ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
142ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * Set the local adapter property - scanMode
143ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     *
144ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @param scanMode the ScanMode to set
145ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
146ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    boolean setScanMode(int scanMode) {
147ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
148ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mService.setAdapterPropertyNative(
149ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    AbstractionLayer.BT_PROPERTY_ADAPTER_SCAN_MODE, Utils.intToByteArray(scanMode));
150ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
151ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
152ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
153ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
154ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @return the mUuids
155ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
156ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    ParcelUuid[] getUuids() {
157ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
158ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mUuids;
159ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
160ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
161ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
162ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
163ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * Set local adapter UUIDs.
164ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     *
165ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @param uuids the uuids to be set.
166ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
167ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    boolean setUuids(ParcelUuid[] uuids) {
168ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
169ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mService.setAdapterPropertyNative(
170ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    AbstractionLayer.BT_PROPERTY_UUIDS, Utils.uuidsToByteArray(uuids));
171ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
172ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
173ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
174ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
175ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @return the mAddress
176ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
177ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    byte[] getAddress() {
178ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
179ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mAddress;
180ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
181ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
182ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
183ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
184ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @param mConnectionState the mConnectionState to set
185ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
186ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    void setConnectionState(int mConnectionState) {
187ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
188ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            this.mConnectionState = mConnectionState;
189ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
190ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
191ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
192ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
193ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @return the mConnectionState
194ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
195ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    int getConnectionState() {
196ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
197ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mConnectionState;
198ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
199ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
200ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
201ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
202ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @param mState the mState to set
203ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
204ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    void setState(int mState) {
205ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
206fd1da115cbf09b7dd9bca3c7d3a4fb816a835dc5Matthew Xie            debugLog("Setting state to " + mState);
207ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            this.mState = mState;
208ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
209ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
210ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
211ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
212ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @return the mState
213ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
214ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    int getState() {
21550f6cb42218fd50ed2532884d1212e1c9a74c7b2Zhihai Xu        /* remove the lock to work around a platform deadlock problem */
21650f6cb42218fd50ed2532884d1212e1c9a74c7b2Zhihai Xu        /* and also for read access, it is safe to remove the lock to save CPU power */
21750f6cb42218fd50ed2532884d1212e1c9a74c7b2Zhihai Xu        return mState;
218ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
219ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
220ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    /**
221c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     * @return the mNumOfAdvertisementInstancesSupported
222c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     */
223c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    int getNumOfAdvertisementInstancesSupported() {
224c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham        return mNumOfAdvertisementInstancesSupported;
225c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    }
226c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham
227c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    /**
228c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     * @return the mRpaOffloadSupported
229c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     */
230c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    boolean isRpaOffloadSupported() {
231c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham        return mRpaOffloadSupported;
232c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    }
233c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham
234c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    /**
235c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     * @return the mNumOfOffloadedIrkSupported
236c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     */
237c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    int getNumOfOffloadedIrkSupported() {
238c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham        return mNumOfOffloadedIrkSupported;
239c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    }
240c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham
241c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    /**
242c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     * @return the mNumOfOffloadedScanFilterSupported
243c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     */
244c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    int getNumOfOffloadedScanFilterSupported() {
245c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham        return mNumOfOffloadedScanFilterSupported;
246c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    }
247c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham
248c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    /**
249c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     * @return the mOffloadedScanResultStorageBytes
250c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham     */
251c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    int getOffloadedScanResultStorage() {
252c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham        return mOffloadedScanResultStorageBytes;
253c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    }
254c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham
255c145fd074dac20e6c9836ee6e900538f2cc7dd54Prerepa Viswanadham    /**
256d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham     * @return tx/rx/idle activity and energy info
257d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham     */
258d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham    boolean isActivityAndEnergyReportingSupported() {
259d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham        return mIsActivityAndEnergyReporting;
260d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham    }
261d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham    /**
262ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     * @return the mBondedDevices
263ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh     */
264ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    BluetoothDevice[] getBondedDevices() {
2652f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan        BluetoothDevice[] bondedDeviceList = new BluetoothDevice[0];
266ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
2672f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            if(mBondedDevices.isEmpty())
2682f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan                return (new BluetoothDevice[0]);
2692f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan
2702f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            try {
2712f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan                bondedDeviceList = mBondedDevices.toArray(bondedDeviceList);
27220de1a6601e8aea7bbc9b9a8219cc616ffddae25Andre Eisenbach                infoLog("getBondedDevices: length="+bondedDeviceList.length);
2732f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan                return bondedDeviceList;
2742f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            } catch(ArrayStoreException ee) {
2752f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan                errorLog("Error retrieving bonded device array");
2762f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan                return (new BluetoothDevice[0]);
2772f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            }
2782f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan        }
2792f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan    }
280179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy    // This function shall be invoked from BondStateMachine whenever the bond
281179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy    // state changes.
282179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy    void onBondStateChanged(BluetoothDevice device, int state)
283179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy    {
2842f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan        if(device == null)
2852f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            return;
2862f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan        try {
2872f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            byte[] addrByte = Utils.getByteAddress(device);
2882f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            DeviceProperties prop = mRemoteDevices.getDeviceProperties(device);
2892f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            if (prop == null)
2902f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan                prop = mRemoteDevices.addDeviceProperties(addrByte);
291179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy            prop.setBondState(state);
292179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy
293179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy            if (state == BluetoothDevice.BOND_BONDED) {
294179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                // add if not already in list
295179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                if(!mBondedDevices.contains(device)) {
296179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                    debugLog("Adding bonded device:" +  device);
297179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                    mBondedDevices.add(device);
298179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                }
299179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy            } else if (state == BluetoothDevice.BOND_NONE) {
300179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                // remove device from list
301179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                if (mBondedDevices.remove(device))
302179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                    debugLog("Removing bonded device:" +  device);
303179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                else
304179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                    debugLog("Failed to remove device: " + device);
3052f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan            }
3062f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan        }
3072f9a79b2d545e1a8b50a0018456892a9934db99aRavi Nagarajan        catch(Exception ee) {
308179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy            Log.e(TAG, "Exception in onBondStateChanged : ", ee);
309ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
310ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
311ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
312ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    int getDiscoverableTimeout() {
313ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
314ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mDiscoverableTimeout;
315ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
316ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
317ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
318ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    boolean setDiscoverableTimeout(int timeout) {
319ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
320ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mService.setAdapterPropertyNative(
321ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    AbstractionLayer.BT_PROPERTY_ADAPTER_DISCOVERABLE_TIMEOUT,
322ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    Utils.intToByteArray(timeout));
323ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
324ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
325ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
326ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    int getProfileConnectionState(int profile) {
327ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
328ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            Pair<Integer, Integer> p = mProfileConnectionState.get(profile);
329ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            if (p != null) return p.first;
330ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return BluetoothProfile.STATE_DISCONNECTED;
331ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
332ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
333ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
334ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    boolean isDiscovering() {
335ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
336ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return mDiscovering;
337ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
338ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
339ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
340ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    void sendConnectionStateChange(BluetoothDevice device, int profile, int state, int prevState) {
341ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        if (!validateProfileConnectionState(state) ||
342ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                !validateProfileConnectionState(prevState)) {
343ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            // Previously, an invalid state was broadcast anyway,
344ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            // with the invalid state converted to -1 in the intent.
345ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            // Better to log an error and not send an intent with
346ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            // invalid contents or set mAdapterConnectionState to -1.
347ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            errorLog("Error in sendConnectionStateChange: "
348ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    + "prevState " + prevState + " state " + state);
349ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            return;
350ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
351ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
352ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
353ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            updateProfileConnectionState(profile, state, prevState);
354ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
355ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            if (updateCountersAndCheckForConnectionStateChange(state, prevState)) {
356ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                setConnectionState(state);
357ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
358ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                Intent intent = new Intent(BluetoothAdapter.ACTION_CONNECTION_STATE_CHANGED);
359ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                intent.putExtra(BluetoothDevice.EXTRA_DEVICE, device);
360ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                intent.putExtra(BluetoothAdapter.EXTRA_CONNECTION_STATE,
361ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        convertToAdapterState(state));
362ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                intent.putExtra(BluetoothAdapter.EXTRA_PREVIOUS_CONNECTION_STATE,
363ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        convertToAdapterState(prevState));
364ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
365bd704c741b8c523ad747214f6f0520ac3e2caf8fZhihai Xu                mService.sendBroadcastAsUser(intent, UserHandle.ALL,
366bd704c741b8c523ad747214f6f0520ac3e2caf8fZhihai Xu                        mService.BLUETOOTH_PERM);
367fd1da115cbf09b7dd9bca3c7d3a4fb816a835dc5Matthew Xie                Log.d(TAG, "CONNECTION_STATE_CHANGE: " + device + ": "
368ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        + prevState + " -> " + state);
369ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            }
370ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
371ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
372ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
373ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private boolean validateProfileConnectionState(int state) {
374ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        return (state == BluetoothProfile.STATE_DISCONNECTED ||
375ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                state == BluetoothProfile.STATE_CONNECTING ||
376ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                state == BluetoothProfile.STATE_CONNECTED ||
377ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                state == BluetoothProfile.STATE_DISCONNECTING);
378ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
379ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
380ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
381ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private int convertToAdapterState(int state) {
382ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        switch (state) {
383ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_DISCONNECTED:
384ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                return BluetoothAdapter.STATE_DISCONNECTED;
385ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_DISCONNECTING:
386ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                return BluetoothAdapter.STATE_DISCONNECTING;
387ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_CONNECTED:
388ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                return BluetoothAdapter.STATE_CONNECTED;
389ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_CONNECTING:
390ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                return BluetoothAdapter.STATE_CONNECTING;
391ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
392ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        Log.e(TAG, "Error in convertToAdapterState");
393ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        return -1;
394ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
395ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
396ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private boolean updateCountersAndCheckForConnectionStateChange(int state, int prevState) {
397ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        switch (prevState) {
398ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_CONNECTING:
399ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                mProfilesConnecting--;
400ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                break;
401ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
402ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_CONNECTED:
403ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                mProfilesConnected--;
404ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                break;
405ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
406ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_DISCONNECTING:
407ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                mProfilesDisconnecting--;
408ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                break;
409ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
410ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
411ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        switch (state) {
412ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_CONNECTING:
413ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                mProfilesConnecting++;
414ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                return (mProfilesConnected == 0 && mProfilesConnecting == 1);
415ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
416ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_CONNECTED:
417ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                mProfilesConnected++;
418ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                return (mProfilesConnected == 1);
419ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
420ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_DISCONNECTING:
421ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                mProfilesDisconnecting++;
422ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                return (mProfilesConnected == 0 && mProfilesDisconnecting == 1);
423ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
424ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            case BluetoothProfile.STATE_DISCONNECTED:
425ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                return (mProfilesConnected == 0 && mProfilesConnecting == 0);
426ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
427ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            default:
428ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                return true;
429ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
430ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
431ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
432ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private void updateProfileConnectionState(int profile, int newState, int oldState) {
433ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // mProfileConnectionState is a hashmap -
434ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // <Integer, Pair<Integer, Integer>>
435ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // The key is the profile, the value is a pair. first element
436ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // is the state and the second element is the number of devices
437ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // in that state.
438ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        int numDev = 1;
439ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        int newHashState = newState;
440ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        boolean update = true;
441ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
442ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // The following conditions are considered in this function:
443ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // 1. If there is no record of profile and state - update
444ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // 2. If a new device's state is current hash state - increment
445ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        //    number of devices in the state.
446ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // 3. If a state change has happened to Connected or Connecting
447ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        //    (if current state is not connected), update.
448ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // 4. If numDevices is 1 and that device state is being updated, update
449ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        // 5. If numDevices is > 1 and one of the devices is changing state,
450ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        //    decrement numDevices but maintain oldState if it is Connected or
451ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        //    Connecting
452ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        Pair<Integer, Integer> stateNumDev = mProfileConnectionState.get(profile);
453ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        if (stateNumDev != null) {
454ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            int currHashState = stateNumDev.first;
455ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            numDev = stateNumDev.second;
456ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
457ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            if (newState == currHashState) {
458ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                numDev ++;
459ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            } else if (newState == BluetoothProfile.STATE_CONNECTED ||
460ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                   (newState == BluetoothProfile.STATE_CONNECTING &&
461ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    currHashState != BluetoothProfile.STATE_CONNECTED)) {
462ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                 numDev = 1;
463ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            } else if (numDev == 1 && oldState == currHashState) {
464ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                 update = true;
465ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            } else if (numDev > 1 && oldState == currHashState) {
466ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                 numDev --;
467ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
468ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                 if (currHashState == BluetoothProfile.STATE_CONNECTED ||
469ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                     currHashState == BluetoothProfile.STATE_CONNECTING) {
470ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    newHashState = currHashState;
471ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                 }
472ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            } else {
473ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                 update = false;
474ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            }
475ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
476ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
477ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        if (update) {
478ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            mProfileConnectionState.put(profile, new Pair<Integer, Integer>(newHashState,
479ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    numDev));
480ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
481ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
482ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
483ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    void adapterPropertyChangedCallback(int[] types, byte[][] values) {
484ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        Intent intent;
485ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        int type;
486ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        byte[] val;
487ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        for (int i = 0; i < types.length; i++) {
488ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            val = values[i];
489ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            type = types[i];
490ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            infoLog("adapterPropertyChangedCallback with type:" + type + " len:" + val.length);
491ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            synchronized (mObject) {
492ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                switch (type) {
493ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    case AbstractionLayer.BT_PROPERTY_BDNAME:
494ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        mName = new String(val);
495ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        intent = new Intent(BluetoothAdapter.ACTION_LOCAL_NAME_CHANGED);
496ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        intent.putExtra(BluetoothAdapter.EXTRA_LOCAL_NAME, mName);
497ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
49831be0d2814d1d3c640160aeafba17feafbaa8bb6Zhihai Xu                        mService.sendBroadcastAsUser(intent, UserHandle.ALL,
49931be0d2814d1d3c640160aeafba17feafbaa8bb6Zhihai Xu                                 mService.BLUETOOTH_PERM);
500ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        debugLog("Name is: " + mName);
501ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        break;
502ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    case AbstractionLayer.BT_PROPERTY_BDADDR:
503ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        mAddress = val;
504ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        debugLog("Address is:" + Utils.getAddressStringFromByte(mAddress));
505ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        break;
506ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    case AbstractionLayer.BT_PROPERTY_CLASS_OF_DEVICE:
507ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        mBluetoothClass = Utils.byteArrayToInt(val, 0);
508ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        debugLog("BT Class:" + mBluetoothClass);
509ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        break;
510ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    case AbstractionLayer.BT_PROPERTY_ADAPTER_SCAN_MODE:
511ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        int mode = Utils.byteArrayToInt(val, 0);
512ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        mScanMode = mService.convertScanModeFromHal(mode);
513ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        intent = new Intent(BluetoothAdapter.ACTION_SCAN_MODE_CHANGED);
514ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        intent.putExtra(BluetoothAdapter.EXTRA_SCAN_MODE, mScanMode);
515ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
51674ae04c73312403e89db0f8e9bd9601d403b4783fredc                        mService.sendBroadcast(intent, mService.BLUETOOTH_PERM);
517ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        debugLog("Scan Mode:" + mScanMode);
51831ba132491053bc86d419a7d51fc04af3299c076fredc                        if (mBluetoothDisabling) {
51931ba132491053bc86d419a7d51fc04af3299c076fredc                            mBluetoothDisabling=false;
52031ba132491053bc86d419a7d51fc04af3299c076fredc                            mService.startBluetoothDisable();
52131ba132491053bc86d419a7d51fc04af3299c076fredc                        }
522ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        break;
523ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    case AbstractionLayer.BT_PROPERTY_UUIDS:
524ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        mUuids = Utils.byteArrayToUuid(val);
525ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        break;
526ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    case AbstractionLayer.BT_PROPERTY_ADAPTER_BONDED_DEVICES:
527ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        int number = val.length/BD_ADDR_LEN;
528ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        byte[] addrByte = new byte[BD_ADDR_LEN];
529ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        for (int j = 0; j < number; j++) {
530ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                            System.arraycopy(val, j * BD_ADDR_LEN, addrByte, 0, BD_ADDR_LEN);
531179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                            onBondStateChanged(mAdapter.getRemoteDevice(
532179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                                               Utils.getAddressStringFromByte(addrByte)),
533179fd2512d1a93bbd2a1a3803f5764cb417fe4c9Kausik Sinnaswamy                                               BluetoothDevice.BOND_BONDED);
534ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        }
535ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        break;
536ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    case AbstractionLayer.BT_PROPERTY_ADAPTER_DISCOVERABLE_TIMEOUT:
537ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        mDiscoverableTimeout = Utils.byteArrayToInt(val, 0);
538ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        debugLog("Discoverable Timeout:" + mDiscoverableTimeout);
539ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        break;
540d0f1d36e653e1b91337743b92472ba4ea97c538bGanesh Ganapathi Batta
541d0f1d36e653e1b91337743b92472ba4ea97c538bGanesh Ganapathi Batta                    case AbstractionLayer.BT_PROPERTY_LOCAL_LE_FEATURES:
542d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham                        updateFeatureSupport(val);
543d0f1d36e653e1b91337743b92472ba4ea97c538bGanesh Ganapathi Batta                        break;
544d0f1d36e653e1b91337743b92472ba4ea97c538bGanesh Ganapathi Batta
545ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                    default:
546ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                        errorLog("Property change not handled in Java land:" + type);
547ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                }
548ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            }
549ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
550ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
551ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
552d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham    void updateFeatureSupport(byte[] val) {
553d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham        mNumOfAdvertisementInstancesSupported = (0xFF & ((int)val[1]));
554d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham        mRpaOffloadSupported = ((0xFF & ((int)val[2]))!= 0);
555d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham        mNumOfOffloadedIrkSupported =  (0xFF & ((int)val[3]));
556d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham        mNumOfOffloadedScanFilterSupported = (0xFF & ((int)val[4]));
557f2f2e89a583d764066c4a84b3d5de5201310202bPrerepa Viswanadham        mOffloadedScanResultStorageBytes = ((0xFF & ((int)val[6])) << 8)
558f2f2e89a583d764066c4a84b3d5de5201310202bPrerepa Viswanadham                            + (0xFF & ((int)val[5]));
559f2f2e89a583d764066c4a84b3d5de5201310202bPrerepa Viswanadham        mIsActivityAndEnergyReporting = ((0xFF & ((int)val[7])) != 0);
560d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham
561d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham        Log.d(TAG, "BT_PROPERTY_LOCAL_LE_FEATURES: update from BT controller"
562f2f2e89a583d764066c4a84b3d5de5201310202bPrerepa Viswanadham                + " mNumOfAdvertisementInstancesSupported = "
563f2f2e89a583d764066c4a84b3d5de5201310202bPrerepa Viswanadham                + mNumOfAdvertisementInstancesSupported
564d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham                + " mRpaOffloadSupported = " + mRpaOffloadSupported
565f2f2e89a583d764066c4a84b3d5de5201310202bPrerepa Viswanadham                + " mNumOfOffloadedIrkSupported = "
566f2f2e89a583d764066c4a84b3d5de5201310202bPrerepa Viswanadham                + mNumOfOffloadedIrkSupported
567d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham                + " mNumOfOffloadedScanFilterSupported = "
568d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham                + mNumOfOffloadedScanFilterSupported
569f2f2e89a583d764066c4a84b3d5de5201310202bPrerepa Viswanadham                + " mOffloadedScanResultStorageBytes= "
570f2f2e89a583d764066c4a84b3d5de5201310202bPrerepa Viswanadham                + mOffloadedScanResultStorageBytes
571d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham                + " mIsActivityAndEnergyReporting = "
572d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham                + mIsActivityAndEnergyReporting);
573d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham    }
574d981566d5912bfd66fc265508e5b0c43dce76f82Prerepa Viswanadham
57544abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh    void onBluetoothReady() {
57631ba132491053bc86d419a7d51fc04af3299c076fredc        Log.d(TAG, "ScanMode =  " + mScanMode );
57731ba132491053bc86d419a7d51fc04af3299c076fredc        Log.d(TAG, "State =  " + getState() );
57831ba132491053bc86d419a7d51fc04af3299c076fredc
57944abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh        // When BT is being turned on, all adapter properties will be sent in 1
58044abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh        // callback. At this stage, set the scan mode.
58144abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh        synchronized (mObject) {
58244abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh            if (getState() == BluetoothAdapter.STATE_TURNING_ON &&
58344abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh                    mScanMode == BluetoothAdapter.SCAN_MODE_NONE) {
58419da573973557408b1b7398a2c3a167d8da9527dSrikanth Uppala                    /* mDiscoverableTimeout is part of the
58519da573973557408b1b7398a2c3a167d8da9527dSrikanth Uppala                       adapterPropertyChangedCallback received before
58619da573973557408b1b7398a2c3a167d8da9527dSrikanth Uppala                       onBluetoothReady */
58719da573973557408b1b7398a2c3a167d8da9527dSrikanth Uppala                    if (mDiscoverableTimeout != 0)
58819da573973557408b1b7398a2c3a167d8da9527dSrikanth Uppala                      setScanMode(AbstractionLayer.BT_SCAN_MODE_CONNECTABLE);
58919da573973557408b1b7398a2c3a167d8da9527dSrikanth Uppala                    else /* if timeout == never (0) at startup */
59019da573973557408b1b7398a2c3a167d8da9527dSrikanth Uppala                      setScanMode(AbstractionLayer.BT_SCAN_MODE_CONNECTABLE_DISCOVERABLE);
59119da573973557408b1b7398a2c3a167d8da9527dSrikanth Uppala                    /* though not always required, this keeps NV up-to date on first-boot after flash */
59219da573973557408b1b7398a2c3a167d8da9527dSrikanth Uppala                    setDiscoverableTimeout(mDiscoverableTimeout);
59344abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh            }
59444abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh        }
59544abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh    }
59644abcd2791995793a5de09906b0ce6ed51bb6080Jaikumar Ganesh
59731ba132491053bc86d419a7d51fc04af3299c076fredc    private boolean mBluetoothDisabling=false;
59831ba132491053bc86d419a7d51fc04af3299c076fredc
599d570893c5ec3bc9fd1860aa0355c550d3a51f8ddKausik Sinnaswamy    void onBluetoothDisable() {
600d570893c5ec3bc9fd1860aa0355c550d3a51f8ddKausik Sinnaswamy        // When BT disable is invoked, set the scan_mode to NONE
601d570893c5ec3bc9fd1860aa0355c550d3a51f8ddKausik Sinnaswamy        // so no incoming connections are possible
60231ba132491053bc86d419a7d51fc04af3299c076fredc
60331ba132491053bc86d419a7d51fc04af3299c076fredc        //Set flag to indicate we are disabling. When property change of scan mode done
60431ba132491053bc86d419a7d51fc04af3299c076fredc        //continue with disable sequence
60531ba132491053bc86d419a7d51fc04af3299c076fredc        debugLog("onBluetoothDisable()");
60631ba132491053bc86d419a7d51fc04af3299c076fredc        mBluetoothDisabling = true;
607d570893c5ec3bc9fd1860aa0355c550d3a51f8ddKausik Sinnaswamy        if (getState() == BluetoothAdapter.STATE_TURNING_OFF) {
608d570893c5ec3bc9fd1860aa0355c550d3a51f8ddKausik Sinnaswamy            setScanMode(AbstractionLayer.BT_SCAN_MODE_NONE);
609d570893c5ec3bc9fd1860aa0355c550d3a51f8ddKausik Sinnaswamy        }
610d570893c5ec3bc9fd1860aa0355c550d3a51f8ddKausik Sinnaswamy    }
611ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    void discoveryStateChangeCallback(int state) {
612ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        infoLog("Callback:discoveryStateChangeCallback with state:" + state);
613ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        synchronized (mObject) {
614ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            Intent intent;
615ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            if (state == AbstractionLayer.BT_DISCOVERY_STOPPED) {
616ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                mDiscovering = false;
617ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                intent = new Intent(BluetoothAdapter.ACTION_DISCOVERY_FINISHED);
61874ae04c73312403e89db0f8e9bd9601d403b4783fredc                mService.sendBroadcast(intent, mService.BLUETOOTH_PERM);
619ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            } else if (state == AbstractionLayer.BT_DISCOVERY_STARTED) {
620ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                mDiscovering = true;
621ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh                intent = new Intent(BluetoothAdapter.ACTION_DISCOVERY_STARTED);
62274ae04c73312403e89db0f8e9bd9601d403b4783fredc                mService.sendBroadcast(intent, mService.BLUETOOTH_PERM);
623ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh            }
624ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        }
625ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
626ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
627ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private void infoLog(String msg) {
62820de1a6601e8aea7bbc9b9a8219cc616ffddae25Andre Eisenbach        if (VDBG) Log.i(TAG, msg);
629ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
630ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
631ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private void debugLog(String msg) {
632ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        if (DBG) Log.d(TAG, msg);
633ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
634ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh
635ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    private void errorLog(String msg) {
636ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh        Log.e(TAG, msg);
637ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh    }
638ff4f17bf64978d0738c66e1b6dd70be8664efc24Jaikumar Ganesh}
639