1 /* 2 * Copyright (C) 2015 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 com.android.bluetoothmidiservice; 18 19 import android.bluetooth.BluetoothDevice; 20 import android.bluetooth.BluetoothGatt; 21 import android.bluetooth.BluetoothGattCallback; 22 import android.bluetooth.BluetoothGattCharacteristic; 23 import android.bluetooth.BluetoothGattDescriptor; 24 import android.bluetooth.BluetoothGattService; 25 import android.bluetooth.BluetoothProfile; 26 import android.content.Context; 27 import android.media.midi.MidiDeviceInfo; 28 import android.media.midi.MidiDeviceServer; 29 import android.media.midi.MidiDeviceStatus; 30 import android.media.midi.MidiManager; 31 import android.media.midi.MidiReceiver; 32 import android.os.Bundle; 33 import android.os.IBinder; 34 import android.util.Log; 35 36 import com.android.internal.midi.MidiEventScheduler; 37 import com.android.internal.midi.MidiEventScheduler.MidiEvent; 38 39 import libcore.io.IoUtils; 40 41 import java.io.IOException; 42 import java.util.List; 43 import java.util.UUID; 44 45 /** 46 * Class used to implement a Bluetooth MIDI device. 47 */ 48 public final class BluetoothMidiDevice { 49 50 private static final String TAG = "BluetoothMidiDevice"; 51 private static final boolean DEBUG = false; 52 53 private static final int MAX_PACKET_SIZE = 20; 54 55 // Bluetooth MIDI Gatt service UUID 56 private static final UUID MIDI_SERVICE = UUID.fromString( 57 "03B80E5A-EDE8-4B33-A751-6CE34EC4C700"); 58 // Bluetooth MIDI Gatt characteristic UUID 59 private static final UUID MIDI_CHARACTERISTIC = UUID.fromString( 60 "7772E5DB-3868-4112-A1A9-F2669D106BF3"); 61 // Descriptor UUID for enabling characteristic changed notifications 62 private static final UUID CLIENT_CHARACTERISTIC_CONFIG = UUID.fromString( 63 "00002902-0000-1000-8000-00805f9b34fb"); 64 65 private final BluetoothDevice mBluetoothDevice; 66 private final BluetoothMidiService mService; 67 private final MidiManager mMidiManager; 68 private MidiReceiver mOutputReceiver; 69 private final MidiEventScheduler mEventScheduler = new MidiEventScheduler(); 70 71 private MidiDeviceServer mDeviceServer; 72 private BluetoothGatt mBluetoothGatt; 73 74 private BluetoothGattCharacteristic mCharacteristic; 75 76 // PacketReceiver for receiving formatted packets from our BluetoothPacketEncoder 77 private final PacketReceiver mPacketReceiver = new PacketReceiver(); 78 79 private final BluetoothPacketEncoder mPacketEncoder 80 = new BluetoothPacketEncoder(mPacketReceiver, MAX_PACKET_SIZE); 81 82 private final BluetoothPacketDecoder mPacketDecoder 83 = new BluetoothPacketDecoder(MAX_PACKET_SIZE); 84 85 private final MidiDeviceServer.Callback mDeviceServerCallback 86 = new MidiDeviceServer.Callback() { 87 @Override 88 public void onDeviceStatusChanged(MidiDeviceServer server, MidiDeviceStatus status) { 89 } 90 91 @Override 92 public void onClose() { 93 close(); 94 } 95 }; 96 97 private final BluetoothGattCallback mGattCallback = new BluetoothGattCallback() { 98 @Override 99 public void onConnectionStateChange(BluetoothGatt gatt, int status, 100 int newState) { 101 String intentAction; 102 if (newState == BluetoothProfile.STATE_CONNECTED) { 103 Log.d(TAG, "Connected to GATT server."); 104 Log.d(TAG, "Attempting to start service discovery:" + 105 mBluetoothGatt.discoverServices()); 106 } else if (newState == BluetoothProfile.STATE_DISCONNECTED) { 107 Log.i(TAG, "Disconnected from GATT server."); 108 close(); 109 } 110 } 111 112 @Override 113 public void onServicesDiscovered(BluetoothGatt gatt, int status) { 114 if (status == BluetoothGatt.GATT_SUCCESS) { 115 BluetoothGattService service = gatt.getService(MIDI_SERVICE); 116 if (service != null) { 117 Log.d(TAG, "found MIDI_SERVICE"); 118 BluetoothGattCharacteristic characteristic 119 = service.getCharacteristic(MIDI_CHARACTERISTIC); 120 if (characteristic != null) { 121 Log.d(TAG, "found MIDI_CHARACTERISTIC"); 122 mCharacteristic = characteristic; 123 124 // Request a lower Connection Interval for better latency. 125 boolean result = gatt.requestConnectionPriority( 126 BluetoothGatt.CONNECTION_PRIORITY_HIGH); 127 Log.d(TAG, "requestConnectionPriority(CONNECTION_PRIORITY_HIGH):" 128 + result); 129 130 // Specification says to read the characteristic first and then 131 // switch to receiving notifications 132 mBluetoothGatt.readCharacteristic(characteristic); 133 } 134 } 135 } else { 136 Log.e(TAG, "onServicesDiscovered received: " + status); 137 close(); 138 } 139 } 140 141 @Override 142 public void onCharacteristicRead(BluetoothGatt gatt, 143 BluetoothGattCharacteristic characteristic, 144 int status) { 145 Log.d(TAG, "onCharacteristicRead " + status); 146 147 // switch to receiving notifications after initial characteristic read 148 mBluetoothGatt.setCharacteristicNotification(characteristic, true); 149 150 // Use writeType that requests acknowledgement. 151 // This improves compatibility with various BLE-MIDI devices. 152 int originalWriteType = characteristic.getWriteType(); 153 characteristic.setWriteType(BluetoothGattCharacteristic.WRITE_TYPE_DEFAULT); 154 155 BluetoothGattDescriptor descriptor = characteristic.getDescriptor( 156 CLIENT_CHARACTERISTIC_CONFIG); 157 if (descriptor != null) { 158 descriptor.setValue(BluetoothGattDescriptor.ENABLE_NOTIFICATION_VALUE); 159 boolean result = mBluetoothGatt.writeDescriptor(descriptor); 160 Log.d(TAG, "writeDescriptor returned " + result); 161 } else { 162 Log.e(TAG, "No CLIENT_CHARACTERISTIC_CONFIG for device " + mBluetoothDevice); 163 } 164 165 characteristic.setWriteType(originalWriteType); 166 } 167 168 @Override 169 public void onCharacteristicWrite(BluetoothGatt gatt, 170 BluetoothGattCharacteristic characteristic, 171 int status) { 172 Log.d(TAG, "onCharacteristicWrite " + status); 173 mPacketEncoder.writeComplete(); 174 } 175 176 @Override 177 public void onCharacteristicChanged(BluetoothGatt gatt, 178 BluetoothGattCharacteristic characteristic) { 179 if (DEBUG) { 180 logByteArray("Received ", characteristic.getValue(), 0, 181 characteristic.getValue().length); 182 } 183 mPacketDecoder.decodePacket(characteristic.getValue(), mOutputReceiver); 184 } 185 }; 186 187 // This receives MIDI data that has already been passed through our MidiEventScheduler 188 // and has been normalized by our MidiFramer. 189 190 private class PacketReceiver implements PacketEncoder.PacketReceiver { 191 // buffers of every possible packet size 192 private final byte[][] mWriteBuffers; 193 194 public PacketReceiver() { 195 // Create buffers of every possible packet size 196 mWriteBuffers = new byte[MAX_PACKET_SIZE + 1][]; 197 for (int i = 0; i <= MAX_PACKET_SIZE; i++) { 198 mWriteBuffers[i] = new byte[i]; 199 } 200 } 201 202 @Override 203 public void writePacket(byte[] buffer, int count) { 204 if (mCharacteristic == null) { 205 Log.w(TAG, "not ready to send packet yet"); 206 return; 207 } 208 byte[] writeBuffer = mWriteBuffers[count]; 209 System.arraycopy(buffer, 0, writeBuffer, 0, count); 210 mCharacteristic.setValue(writeBuffer); 211 if (DEBUG) { 212 logByteArray("Sent ", mCharacteristic.getValue(), 0, 213 mCharacteristic.getValue().length); 214 } 215 mBluetoothGatt.writeCharacteristic(mCharacteristic); 216 } 217 } 218 219 public BluetoothMidiDevice(Context context, BluetoothDevice device, 220 BluetoothMidiService service) { 221 mBluetoothDevice = device; 222 mService = service; 223 224 mBluetoothGatt = mBluetoothDevice.connectGatt(context, false, mGattCallback); 225 226 mMidiManager = (MidiManager)context.getSystemService(Context.MIDI_SERVICE); 227 228 Bundle properties = new Bundle(); 229 properties.putString(MidiDeviceInfo.PROPERTY_NAME, mBluetoothGatt.getDevice().getName()); 230 properties.putParcelable(MidiDeviceInfo.PROPERTY_BLUETOOTH_DEVICE, 231 mBluetoothGatt.getDevice()); 232 233 MidiReceiver[] inputPortReceivers = new MidiReceiver[1]; 234 inputPortReceivers[0] = mEventScheduler.getReceiver(); 235 236 mDeviceServer = mMidiManager.createDeviceServer(inputPortReceivers, 1, 237 null, null, properties, MidiDeviceInfo.TYPE_BLUETOOTH, mDeviceServerCallback); 238 239 mOutputReceiver = mDeviceServer.getOutputPortReceivers()[0]; 240 241 // This thread waits for outgoing messages from our MidiEventScheduler 242 // And forwards them to our MidiFramer to be prepared to send via Bluetooth. 243 new Thread("BluetoothMidiDevice " + mBluetoothDevice) { 244 @Override 245 public void run() { 246 while (true) { 247 MidiEvent event; 248 try { 249 event = (MidiEvent)mEventScheduler.waitNextEvent(); 250 } catch (InterruptedException e) { 251 // try again 252 continue; 253 } 254 if (event == null) { 255 break; 256 } 257 try { 258 mPacketEncoder.send(event.data, 0, event.count, 259 event.getTimestamp()); 260 } catch (IOException e) { 261 Log.e(TAG, "mPacketAccumulator.send failed", e); 262 } 263 mEventScheduler.addEventToPool(event); 264 } 265 Log.d(TAG, "BluetoothMidiDevice thread exit"); 266 } 267 }.start(); 268 } 269 270 private void close() { 271 synchronized (mBluetoothDevice) { 272 mEventScheduler.close(); 273 mService.deviceClosed(mBluetoothDevice); 274 275 if (mDeviceServer != null) { 276 IoUtils.closeQuietly(mDeviceServer); 277 mDeviceServer = null; 278 } 279 if (mBluetoothGatt != null) { 280 mBluetoothGatt.close(); 281 mBluetoothGatt = null; 282 } 283 } 284 } 285 286 public IBinder getBinder() { 287 return mDeviceServer.asBinder(); 288 } 289 290 private static void logByteArray(String prefix, byte[] value, int offset, int count) { 291 StringBuilder builder = new StringBuilder(prefix); 292 for (int i = offset; i < count; i++) { 293 builder.append(String.format("0x%02X", value[i])); 294 if (i != value.length - 1) { 295 builder.append(", "); 296 } 297 } 298 Log.d(TAG, builder.toString()); 299 } 300 } 301