Home | History | Annotate | Download | only in bluetooth
      1 /*
      2  * Copyright (C) 2009 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;
     18 
     19 import android.annotation.UnsupportedAppUsage;
     20 import android.os.Build;
     21 import android.os.ParcelUuid;
     22 
     23 import java.nio.ByteBuffer;
     24 import java.nio.ByteOrder;
     25 import java.util.Arrays;
     26 import java.util.HashSet;
     27 import java.util.UUID;
     28 
     29 /**
     30  * Static helper methods and constants to decode the ParcelUuid of remote devices.
     31  *
     32  * @hide
     33  */
     34 public final class BluetoothUuid {
     35 
     36     /* See Bluetooth Assigned Numbers document - SDP section, to get the values of UUIDs
     37      * for the various services.
     38      *
     39      * The following 128 bit values are calculated as:
     40      *  uuid * 2^96 + BASE_UUID
     41      */
     42     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
     43     public static final ParcelUuid AudioSink =
     44             ParcelUuid.fromString("0000110B-0000-1000-8000-00805F9B34FB");
     45     public static final ParcelUuid AudioSource =
     46             ParcelUuid.fromString("0000110A-0000-1000-8000-00805F9B34FB");
     47     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
     48     public static final ParcelUuid AdvAudioDist =
     49             ParcelUuid.fromString("0000110D-0000-1000-8000-00805F9B34FB");
     50     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
     51     public static final ParcelUuid HSP =
     52             ParcelUuid.fromString("00001108-0000-1000-8000-00805F9B34FB");
     53     public static final ParcelUuid HSP_AG =
     54             ParcelUuid.fromString("00001112-0000-1000-8000-00805F9B34FB");
     55     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
     56     public static final ParcelUuid Handsfree =
     57             ParcelUuid.fromString("0000111E-0000-1000-8000-00805F9B34FB");
     58     public static final ParcelUuid Handsfree_AG =
     59             ParcelUuid.fromString("0000111F-0000-1000-8000-00805F9B34FB");
     60     public static final ParcelUuid AvrcpController =
     61             ParcelUuid.fromString("0000110E-0000-1000-8000-00805F9B34FB");
     62     public static final ParcelUuid AvrcpTarget =
     63             ParcelUuid.fromString("0000110C-0000-1000-8000-00805F9B34FB");
     64     @UnsupportedAppUsage
     65     public static final ParcelUuid ObexObjectPush =
     66             ParcelUuid.fromString("00001105-0000-1000-8000-00805f9b34fb");
     67     public static final ParcelUuid Hid =
     68             ParcelUuid.fromString("00001124-0000-1000-8000-00805f9b34fb");
     69     @UnsupportedAppUsage
     70     public static final ParcelUuid Hogp =
     71             ParcelUuid.fromString("00001812-0000-1000-8000-00805f9b34fb");
     72     public static final ParcelUuid PANU =
     73             ParcelUuid.fromString("00001115-0000-1000-8000-00805F9B34FB");
     74     @UnsupportedAppUsage
     75     public static final ParcelUuid NAP =
     76             ParcelUuid.fromString("00001116-0000-1000-8000-00805F9B34FB");
     77     public static final ParcelUuid BNEP =
     78             ParcelUuid.fromString("0000000f-0000-1000-8000-00805F9B34FB");
     79     public static final ParcelUuid PBAP_PCE =
     80             ParcelUuid.fromString("0000112e-0000-1000-8000-00805F9B34FB");
     81     @UnsupportedAppUsage
     82     public static final ParcelUuid PBAP_PSE =
     83             ParcelUuid.fromString("0000112f-0000-1000-8000-00805F9B34FB");
     84     public static final ParcelUuid MAP =
     85             ParcelUuid.fromString("00001134-0000-1000-8000-00805F9B34FB");
     86     public static final ParcelUuid MNS =
     87             ParcelUuid.fromString("00001133-0000-1000-8000-00805F9B34FB");
     88     public static final ParcelUuid MAS =
     89             ParcelUuid.fromString("00001132-0000-1000-8000-00805F9B34FB");
     90     public static final ParcelUuid SAP =
     91             ParcelUuid.fromString("0000112D-0000-1000-8000-00805F9B34FB");
     92     public static final ParcelUuid HearingAid =
     93             ParcelUuid.fromString("0000FDF0-0000-1000-8000-00805f9b34fb");
     94 
     95     public static final ParcelUuid BASE_UUID =
     96             ParcelUuid.fromString("00000000-0000-1000-8000-00805F9B34FB");
     97 
     98     /** Length of bytes for 16 bit UUID */
     99     public static final int UUID_BYTES_16_BIT = 2;
    100     /** Length of bytes for 32 bit UUID */
    101     public static final int UUID_BYTES_32_BIT = 4;
    102     /** Length of bytes for 128 bit UUID */
    103     public static final int UUID_BYTES_128_BIT = 16;
    104 
    105     @UnsupportedAppUsage
    106     public static final ParcelUuid[] RESERVED_UUIDS = {
    107             AudioSink, AudioSource, AdvAudioDist, HSP, Handsfree, AvrcpController, AvrcpTarget,
    108             ObexObjectPush, PANU, NAP, MAP, MNS, MAS, SAP};
    109 
    110     @UnsupportedAppUsage
    111     public static boolean isAudioSource(ParcelUuid uuid) {
    112         return uuid.equals(AudioSource);
    113     }
    114 
    115     public static boolean isAudioSink(ParcelUuid uuid) {
    116         return uuid.equals(AudioSink);
    117     }
    118 
    119     @UnsupportedAppUsage
    120     public static boolean isAdvAudioDist(ParcelUuid uuid) {
    121         return uuid.equals(AdvAudioDist);
    122     }
    123 
    124     public static boolean isHandsfree(ParcelUuid uuid) {
    125         return uuid.equals(Handsfree);
    126     }
    127 
    128     public static boolean isHeadset(ParcelUuid uuid) {
    129         return uuid.equals(HSP);
    130     }
    131 
    132     public static boolean isAvrcpController(ParcelUuid uuid) {
    133         return uuid.equals(AvrcpController);
    134     }
    135 
    136     @UnsupportedAppUsage
    137     public static boolean isAvrcpTarget(ParcelUuid uuid) {
    138         return uuid.equals(AvrcpTarget);
    139     }
    140 
    141     public static boolean isInputDevice(ParcelUuid uuid) {
    142         return uuid.equals(Hid);
    143     }
    144 
    145     public static boolean isPanu(ParcelUuid uuid) {
    146         return uuid.equals(PANU);
    147     }
    148 
    149     public static boolean isNap(ParcelUuid uuid) {
    150         return uuid.equals(NAP);
    151     }
    152 
    153     public static boolean isBnep(ParcelUuid uuid) {
    154         return uuid.equals(BNEP);
    155     }
    156 
    157     public static boolean isMap(ParcelUuid uuid) {
    158         return uuid.equals(MAP);
    159     }
    160 
    161     public static boolean isMns(ParcelUuid uuid) {
    162         return uuid.equals(MNS);
    163     }
    164 
    165     public static boolean isMas(ParcelUuid uuid) {
    166         return uuid.equals(MAS);
    167     }
    168 
    169     public static boolean isSap(ParcelUuid uuid) {
    170         return uuid.equals(SAP);
    171     }
    172 
    173     /**
    174      * Returns true if ParcelUuid is present in uuidArray
    175      *
    176      * @param uuidArray - Array of ParcelUuids
    177      * @param uuid
    178      */
    179     @UnsupportedAppUsage
    180     public static boolean isUuidPresent(ParcelUuid[] uuidArray, ParcelUuid uuid) {
    181         if ((uuidArray == null || uuidArray.length == 0) && uuid == null) {
    182             return true;
    183         }
    184 
    185         if (uuidArray == null) {
    186             return false;
    187         }
    188 
    189         for (ParcelUuid element : uuidArray) {
    190             if (element.equals(uuid)) return true;
    191         }
    192         return false;
    193     }
    194 
    195     /**
    196      * Returns true if there any common ParcelUuids in uuidA and uuidB.
    197      *
    198      * @param uuidA - List of ParcelUuids
    199      * @param uuidB - List of ParcelUuids
    200      */
    201     @UnsupportedAppUsage
    202     public static boolean containsAnyUuid(ParcelUuid[] uuidA, ParcelUuid[] uuidB) {
    203         if (uuidA == null && uuidB == null) return true;
    204 
    205         if (uuidA == null) {
    206             return uuidB.length == 0;
    207         }
    208 
    209         if (uuidB == null) {
    210             return uuidA.length == 0;
    211         }
    212 
    213         HashSet<ParcelUuid> uuidSet = new HashSet<ParcelUuid>(Arrays.asList(uuidA));
    214         for (ParcelUuid uuid : uuidB) {
    215             if (uuidSet.contains(uuid)) return true;
    216         }
    217         return false;
    218     }
    219 
    220     /**
    221      * Returns true if all the ParcelUuids in ParcelUuidB are present in
    222      * ParcelUuidA
    223      *
    224      * @param uuidA - Array of ParcelUuidsA
    225      * @param uuidB - Array of ParcelUuidsB
    226      */
    227     public static boolean containsAllUuids(ParcelUuid[] uuidA, ParcelUuid[] uuidB) {
    228         if (uuidA == null && uuidB == null) return true;
    229 
    230         if (uuidA == null) {
    231             return uuidB.length == 0;
    232         }
    233 
    234         if (uuidB == null) return true;
    235 
    236         HashSet<ParcelUuid> uuidSet = new HashSet<ParcelUuid>(Arrays.asList(uuidA));
    237         for (ParcelUuid uuid : uuidB) {
    238             if (!uuidSet.contains(uuid)) return false;
    239         }
    240         return true;
    241     }
    242 
    243     /**
    244      * Extract the Service Identifier or the actual uuid from the Parcel Uuid.
    245      * For example, if 0000110B-0000-1000-8000-00805F9B34FB is the parcel Uuid,
    246      * this function will return 110B
    247      *
    248      * @param parcelUuid
    249      * @return the service identifier.
    250      */
    251     public static int getServiceIdentifierFromParcelUuid(ParcelUuid parcelUuid) {
    252         UUID uuid = parcelUuid.getUuid();
    253         long value = (uuid.getMostSignificantBits() & 0xFFFFFFFF00000000L) >>> 32;
    254         return (int) value;
    255     }
    256 
    257     /**
    258      * Parse UUID from bytes. The {@code uuidBytes} can represent a 16-bit, 32-bit or 128-bit UUID,
    259      * but the returned UUID is always in 128-bit format.
    260      * Note UUID is little endian in Bluetooth.
    261      *
    262      * @param uuidBytes Byte representation of uuid.
    263      * @return {@link ParcelUuid} parsed from bytes.
    264      * @throws IllegalArgumentException If the {@code uuidBytes} cannot be parsed.
    265      */
    266     public static ParcelUuid parseUuidFrom(byte[] uuidBytes) {
    267         if (uuidBytes == null) {
    268             throw new IllegalArgumentException("uuidBytes cannot be null");
    269         }
    270         int length = uuidBytes.length;
    271         if (length != UUID_BYTES_16_BIT && length != UUID_BYTES_32_BIT
    272                 && length != UUID_BYTES_128_BIT) {
    273             throw new IllegalArgumentException("uuidBytes length invalid - " + length);
    274         }
    275 
    276         // Construct a 128 bit UUID.
    277         if (length == UUID_BYTES_128_BIT) {
    278             ByteBuffer buf = ByteBuffer.wrap(uuidBytes).order(ByteOrder.LITTLE_ENDIAN);
    279             long msb = buf.getLong(8);
    280             long lsb = buf.getLong(0);
    281             return new ParcelUuid(new UUID(msb, lsb));
    282         }
    283 
    284         // For 16 bit and 32 bit UUID we need to convert them to 128 bit value.
    285         // 128_bit_value = uuid * 2^96 + BASE_UUID
    286         long shortUuid;
    287         if (length == UUID_BYTES_16_BIT) {
    288             shortUuid = uuidBytes[0] & 0xFF;
    289             shortUuid += (uuidBytes[1] & 0xFF) << 8;
    290         } else {
    291             shortUuid = uuidBytes[0] & 0xFF;
    292             shortUuid += (uuidBytes[1] & 0xFF) << 8;
    293             shortUuid += (uuidBytes[2] & 0xFF) << 16;
    294             shortUuid += (uuidBytes[3] & 0xFF) << 24;
    295         }
    296         long msb = BASE_UUID.getUuid().getMostSignificantBits() + (shortUuid << 32);
    297         long lsb = BASE_UUID.getUuid().getLeastSignificantBits();
    298         return new ParcelUuid(new UUID(msb, lsb));
    299     }
    300 
    301     /**
    302      * Parse UUID to bytes. The returned value is shortest representation, a 16-bit, 32-bit or
    303      * 128-bit UUID, Note returned value is little endian (Bluetooth).
    304      *
    305      * @param uuid uuid to parse.
    306      * @return shortest representation of {@code uuid} as bytes.
    307      * @throws IllegalArgumentException If the {@code uuid} is null.
    308      */
    309     public static byte[] uuidToBytes(ParcelUuid uuid) {
    310         if (uuid == null) {
    311             throw new IllegalArgumentException("uuid cannot be null");
    312         }
    313 
    314         if (is16BitUuid(uuid)) {
    315             byte[] uuidBytes = new byte[UUID_BYTES_16_BIT];
    316             int uuidVal = getServiceIdentifierFromParcelUuid(uuid);
    317             uuidBytes[0] = (byte) (uuidVal & 0xFF);
    318             uuidBytes[1] = (byte) ((uuidVal & 0xFF00) >> 8);
    319             return uuidBytes;
    320         }
    321 
    322         if (is32BitUuid(uuid)) {
    323             byte[] uuidBytes = new byte[UUID_BYTES_32_BIT];
    324             int uuidVal = getServiceIdentifierFromParcelUuid(uuid);
    325             uuidBytes[0] = (byte) (uuidVal & 0xFF);
    326             uuidBytes[1] = (byte) ((uuidVal & 0xFF00) >> 8);
    327             uuidBytes[2] = (byte) ((uuidVal & 0xFF0000) >> 16);
    328             uuidBytes[3] = (byte) ((uuidVal & 0xFF000000) >> 24);
    329             return uuidBytes;
    330         }
    331 
    332         // Construct a 128 bit UUID.
    333         long msb = uuid.getUuid().getMostSignificantBits();
    334         long lsb = uuid.getUuid().getLeastSignificantBits();
    335 
    336         byte[] uuidBytes = new byte[UUID_BYTES_128_BIT];
    337         ByteBuffer buf = ByteBuffer.wrap(uuidBytes).order(ByteOrder.LITTLE_ENDIAN);
    338         buf.putLong(8, msb);
    339         buf.putLong(0, lsb);
    340         return uuidBytes;
    341     }
    342 
    343     /**
    344      * Check whether the given parcelUuid can be converted to 16 bit bluetooth uuid.
    345      *
    346      * @param parcelUuid
    347      * @return true if the parcelUuid can be converted to 16 bit uuid, false otherwise.
    348      */
    349     @UnsupportedAppUsage
    350     public static boolean is16BitUuid(ParcelUuid parcelUuid) {
    351         UUID uuid = parcelUuid.getUuid();
    352         if (uuid.getLeastSignificantBits() != BASE_UUID.getUuid().getLeastSignificantBits()) {
    353             return false;
    354         }
    355         return ((uuid.getMostSignificantBits() & 0xFFFF0000FFFFFFFFL) == 0x1000L);
    356     }
    357 
    358 
    359     /**
    360      * Check whether the given parcelUuid can be converted to 32 bit bluetooth uuid.
    361      *
    362      * @param parcelUuid
    363      * @return true if the parcelUuid can be converted to 32 bit uuid, false otherwise.
    364      */
    365     @UnsupportedAppUsage
    366     public static boolean is32BitUuid(ParcelUuid parcelUuid) {
    367         UUID uuid = parcelUuid.getUuid();
    368         if (uuid.getLeastSignificantBits() != BASE_UUID.getUuid().getLeastSignificantBits()) {
    369             return false;
    370         }
    371         if (is16BitUuid(parcelUuid)) {
    372             return false;
    373         }
    374         return ((uuid.getMostSignificantBits() & 0xFFFFFFFFL) == 0x1000L);
    375     }
    376 }
    377