1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.bluetooth.le;
18 
19 import android.annotation.Nullable;
20 import android.bluetooth.BluetoothUuid;
21 import android.os.ParcelUuid;
22 import android.util.ArrayMap;
23 import android.util.Log;
24 import android.util.SparseArray;
25 
26 import java.util.ArrayList;
27 import java.util.Arrays;
28 import java.util.List;
29 import java.util.Map;
30 
31 /**
32  * Represents a scan record from Bluetooth LE scan.
33  */
34 public final class ScanRecord {
35 
36     private static final String TAG = "ScanRecord";
37 
38     // The following data type values are assigned by Bluetooth SIG.
39     // For more details refer to Bluetooth 4.1 specification, Volume 3, Part C, Section 18.
40     private static final int DATA_TYPE_FLAGS = 0x01;
41     private static final int DATA_TYPE_SERVICE_UUIDS_16_BIT_PARTIAL = 0x02;
42     private static final int DATA_TYPE_SERVICE_UUIDS_16_BIT_COMPLETE = 0x03;
43     private static final int DATA_TYPE_SERVICE_UUIDS_32_BIT_PARTIAL = 0x04;
44     private static final int DATA_TYPE_SERVICE_UUIDS_32_BIT_COMPLETE = 0x05;
45     private static final int DATA_TYPE_SERVICE_UUIDS_128_BIT_PARTIAL = 0x06;
46     private static final int DATA_TYPE_SERVICE_UUIDS_128_BIT_COMPLETE = 0x07;
47     private static final int DATA_TYPE_LOCAL_NAME_SHORT = 0x08;
48     private static final int DATA_TYPE_LOCAL_NAME_COMPLETE = 0x09;
49     private static final int DATA_TYPE_TX_POWER_LEVEL = 0x0A;
50     private static final int DATA_TYPE_SERVICE_DATA_16_BIT = 0x16;
51     private static final int DATA_TYPE_SERVICE_DATA_32_BIT = 0x20;
52     private static final int DATA_TYPE_SERVICE_DATA_128_BIT = 0x21;
53     private static final int DATA_TYPE_MANUFACTURER_SPECIFIC_DATA = 0xFF;
54 
55     // Flags of the advertising data.
56     private final int mAdvertiseFlags;
57 
58     @Nullable
59     private final List<ParcelUuid> mServiceUuids;
60 
61     private final SparseArray<byte[]> mManufacturerSpecificData;
62 
63     private final Map<ParcelUuid, byte[]> mServiceData;
64 
65     // Transmission power level(in dB).
66     private final int mTxPowerLevel;
67 
68     // Local name of the Bluetooth LE device.
69     private final String mDeviceName;
70 
71     // Raw bytes of scan record.
72     private final byte[] mBytes;
73 
74     /**
75      * Returns the advertising flags indicating the discoverable mode and capability of the device.
76      * Returns -1 if the flag field is not set.
77      */
getAdvertiseFlags()78     public int getAdvertiseFlags() {
79         return mAdvertiseFlags;
80     }
81 
82     /**
83      * Returns a list of service UUIDs within the advertisement that are used to identify the
84      * bluetooth GATT services.
85      */
getServiceUuids()86     public List<ParcelUuid> getServiceUuids() {
87         return mServiceUuids;
88     }
89 
90     /**
91      * Returns a sparse array of manufacturer identifier and its corresponding manufacturer specific
92      * data.
93      */
getManufacturerSpecificData()94     public SparseArray<byte[]> getManufacturerSpecificData() {
95         return mManufacturerSpecificData;
96     }
97 
98     /**
99      * Returns the manufacturer specific data associated with the manufacturer id. Returns
100      * {@code null} if the {@code manufacturerId} is not found.
101      */
102     @Nullable
getManufacturerSpecificData(int manufacturerId)103     public byte[] getManufacturerSpecificData(int manufacturerId) {
104         return mManufacturerSpecificData.get(manufacturerId);
105     }
106 
107     /**
108      * Returns a map of service UUID and its corresponding service data.
109      */
getServiceData()110     public Map<ParcelUuid, byte[]> getServiceData() {
111         return mServiceData;
112     }
113 
114     /**
115      * Returns the service data byte array associated with the {@code serviceUuid}. Returns
116      * {@code null} if the {@code serviceDataUuid} is not found.
117      */
118     @Nullable
getServiceData(ParcelUuid serviceDataUuid)119     public byte[] getServiceData(ParcelUuid serviceDataUuid) {
120         if (serviceDataUuid == null) {
121             return null;
122         }
123         return mServiceData.get(serviceDataUuid);
124     }
125 
126     /**
127      * Returns the transmission power level of the packet in dBm. Returns {@link Integer#MIN_VALUE}
128      * if the field is not set. This value can be used to calculate the path loss of a received
129      * packet using the following equation:
130      * <p>
131      * <code>pathloss = txPowerLevel - rssi</code>
132      */
getTxPowerLevel()133     public int getTxPowerLevel() {
134         return mTxPowerLevel;
135     }
136 
137     /**
138      * Returns the local name of the BLE device. The is a UTF-8 encoded string.
139      */
140     @Nullable
getDeviceName()141     public String getDeviceName() {
142         return mDeviceName;
143     }
144 
145     /**
146      * Returns raw bytes of scan record.
147      */
getBytes()148     public byte[] getBytes() {
149         return mBytes;
150     }
151 
ScanRecord(List<ParcelUuid> serviceUuids, SparseArray<byte[]> manufacturerData, Map<ParcelUuid, byte[]> serviceData, int advertiseFlags, int txPowerLevel, String localName, byte[] bytes)152     private ScanRecord(List<ParcelUuid> serviceUuids,
153             SparseArray<byte[]> manufacturerData,
154             Map<ParcelUuid, byte[]> serviceData,
155             int advertiseFlags, int txPowerLevel,
156             String localName, byte[] bytes) {
157         mServiceUuids = serviceUuids;
158         mManufacturerSpecificData = manufacturerData;
159         mServiceData = serviceData;
160         mDeviceName = localName;
161         mAdvertiseFlags = advertiseFlags;
162         mTxPowerLevel = txPowerLevel;
163         mBytes = bytes;
164     }
165 
166     /**
167      * Parse scan record bytes to {@link ScanRecord}.
168      * <p>
169      * The format is defined in Bluetooth 4.1 specification, Volume 3, Part C, Section 11 and 18.
170      * <p>
171      * All numerical multi-byte entities and values shall use little-endian <strong>byte</strong>
172      * order.
173      *
174      * @param scanRecord The scan record of Bluetooth LE advertisement and/or scan response.
175      * @hide
176      */
parseFromBytes(byte[] scanRecord)177     public static ScanRecord parseFromBytes(byte[] scanRecord) {
178         if (scanRecord == null) {
179             return null;
180         }
181 
182         int currentPos = 0;
183         int advertiseFlag = -1;
184         List<ParcelUuid> serviceUuids = new ArrayList<ParcelUuid>();
185         String localName = null;
186         int txPowerLevel = Integer.MIN_VALUE;
187 
188         SparseArray<byte[]> manufacturerData = new SparseArray<byte[]>();
189         Map<ParcelUuid, byte[]> serviceData = new ArrayMap<ParcelUuid, byte[]>();
190 
191         try {
192             while (currentPos < scanRecord.length) {
193                 // length is unsigned int.
194                 int length = scanRecord[currentPos++] & 0xFF;
195                 if (length == 0) {
196                     break;
197                 }
198                 // Note the length includes the length of the field type itself.
199                 int dataLength = length - 1;
200                 // fieldType is unsigned int.
201                 int fieldType = scanRecord[currentPos++] & 0xFF;
202                 switch (fieldType) {
203                     case DATA_TYPE_FLAGS:
204                         advertiseFlag = scanRecord[currentPos] & 0xFF;
205                         break;
206                     case DATA_TYPE_SERVICE_UUIDS_16_BIT_PARTIAL:
207                     case DATA_TYPE_SERVICE_UUIDS_16_BIT_COMPLETE:
208                         parseServiceUuid(scanRecord, currentPos,
209                                 dataLength, BluetoothUuid.UUID_BYTES_16_BIT, serviceUuids);
210                         break;
211                     case DATA_TYPE_SERVICE_UUIDS_32_BIT_PARTIAL:
212                     case DATA_TYPE_SERVICE_UUIDS_32_BIT_COMPLETE:
213                         parseServiceUuid(scanRecord, currentPos, dataLength,
214                                 BluetoothUuid.UUID_BYTES_32_BIT, serviceUuids);
215                         break;
216                     case DATA_TYPE_SERVICE_UUIDS_128_BIT_PARTIAL:
217                     case DATA_TYPE_SERVICE_UUIDS_128_BIT_COMPLETE:
218                         parseServiceUuid(scanRecord, currentPos, dataLength,
219                                 BluetoothUuid.UUID_BYTES_128_BIT, serviceUuids);
220                         break;
221                     case DATA_TYPE_LOCAL_NAME_SHORT:
222                     case DATA_TYPE_LOCAL_NAME_COMPLETE:
223                         localName = new String(
224                                 extractBytes(scanRecord, currentPos, dataLength));
225                         break;
226                     case DATA_TYPE_TX_POWER_LEVEL:
227                         txPowerLevel = scanRecord[currentPos];
228                         break;
229                     case DATA_TYPE_SERVICE_DATA_16_BIT:
230                     case DATA_TYPE_SERVICE_DATA_32_BIT:
231                     case DATA_TYPE_SERVICE_DATA_128_BIT:
232                         int serviceUuidLength = BluetoothUuid.UUID_BYTES_16_BIT;
233                         if (fieldType == DATA_TYPE_SERVICE_DATA_32_BIT) {
234                             serviceUuidLength = BluetoothUuid.UUID_BYTES_32_BIT;
235                         } else if (fieldType == DATA_TYPE_SERVICE_DATA_128_BIT) {
236                             serviceUuidLength = BluetoothUuid.UUID_BYTES_128_BIT;
237                         }
238 
239                         byte[] serviceDataUuidBytes = extractBytes(scanRecord, currentPos,
240                                 serviceUuidLength);
241                         ParcelUuid serviceDataUuid = BluetoothUuid.parseUuidFrom(
242                                 serviceDataUuidBytes);
243                         byte[] serviceDataArray = extractBytes(scanRecord,
244                                 currentPos + serviceUuidLength, dataLength - serviceUuidLength);
245                         serviceData.put(serviceDataUuid, serviceDataArray);
246                         break;
247                     case DATA_TYPE_MANUFACTURER_SPECIFIC_DATA:
248                         // The first two bytes of the manufacturer specific data are
249                         // manufacturer ids in little endian.
250                         int manufacturerId = ((scanRecord[currentPos + 1] & 0xFF) << 8)
251                                 + (scanRecord[currentPos] & 0xFF);
252                         byte[] manufacturerDataBytes = extractBytes(scanRecord, currentPos + 2,
253                                 dataLength - 2);
254                         manufacturerData.put(manufacturerId, manufacturerDataBytes);
255                         break;
256                     default:
257                         // Just ignore, we don't handle such data type.
258                         break;
259                 }
260                 currentPos += dataLength;
261             }
262 
263             if (serviceUuids.isEmpty()) {
264                 serviceUuids = null;
265             }
266             return new ScanRecord(serviceUuids, manufacturerData, serviceData,
267                     advertiseFlag, txPowerLevel, localName, scanRecord);
268         } catch (Exception e) {
269             Log.e(TAG, "unable to parse scan record: " + Arrays.toString(scanRecord));
270             // As the record is invalid, ignore all the parsed results for this packet
271             // and return an empty record with raw scanRecord bytes in results
272             return new ScanRecord(null, null, null, -1, Integer.MIN_VALUE, null, scanRecord);
273         }
274     }
275 
276     @Override
toString()277     public String toString() {
278         return "ScanRecord [mAdvertiseFlags=" + mAdvertiseFlags + ", mServiceUuids=" + mServiceUuids
279                 + ", mManufacturerSpecificData=" + BluetoothLeUtils.toString(
280                 mManufacturerSpecificData)
281                 + ", mServiceData=" + BluetoothLeUtils.toString(mServiceData)
282                 + ", mTxPowerLevel=" + mTxPowerLevel + ", mDeviceName=" + mDeviceName + "]";
283     }
284 
285     // Parse service UUIDs.
parseServiceUuid(byte[] scanRecord, int currentPos, int dataLength, int uuidLength, List<ParcelUuid> serviceUuids)286     private static int parseServiceUuid(byte[] scanRecord, int currentPos, int dataLength,
287             int uuidLength, List<ParcelUuid> serviceUuids) {
288         while (dataLength > 0) {
289             byte[] uuidBytes = extractBytes(scanRecord, currentPos,
290                     uuidLength);
291             serviceUuids.add(BluetoothUuid.parseUuidFrom(uuidBytes));
292             dataLength -= uuidLength;
293             currentPos += uuidLength;
294         }
295         return currentPos;
296     }
297 
298     // Helper method to extract bytes from byte array.
extractBytes(byte[] scanRecord, int start, int length)299     private static byte[] extractBytes(byte[] scanRecord, int start, int length) {
300         byte[] bytes = new byte[length];
301         System.arraycopy(scanRecord, start, bytes, 0, length);
302         return bytes;
303     }
304 }
305