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 an 14 * limitations under the License. 15 */ 16 17 package com.android.server.usb; 18 19 import android.content.Context; 20 import android.media.midi.MidiDeviceInfo; 21 import android.media.midi.MidiDeviceServer; 22 import android.media.midi.MidiDeviceStatus; 23 import android.media.midi.MidiManager; 24 import android.media.midi.MidiReceiver; 25 import android.media.midi.MidiSender; 26 import android.os.Bundle; 27 import android.system.ErrnoException; 28 import android.system.Os; 29 import android.system.OsConstants; 30 import android.system.StructPollfd; 31 import android.util.Log; 32 33 import com.android.internal.midi.MidiEventScheduler; 34 import com.android.internal.midi.MidiEventScheduler.MidiEvent; 35 36 import libcore.io.IoUtils; 37 38 import java.io.Closeable; 39 import java.io.FileDescriptor; 40 import java.io.FileInputStream; 41 import java.io.FileOutputStream; 42 import java.io.IOException; 43 44 public final class UsbMidiDevice implements Closeable { 45 private static final String TAG = "UsbMidiDevice"; 46 47 private final int mAlsaCard; 48 private final int mAlsaDevice; 49 private final int mSubdeviceCount; 50 private final InputReceiverProxy[] mInputPortReceivers; 51 52 private MidiDeviceServer mServer; 53 54 // event schedulers for each input port of the physical device 55 private MidiEventScheduler[] mEventSchedulers; 56 57 private static final int BUFFER_SIZE = 512; 58 59 private FileDescriptor[] mFileDescriptors; 60 61 // for polling multiple FileDescriptors for MIDI events 62 private StructPollfd[] mPollFDs; 63 // streams for reading from ALSA driver 64 private FileInputStream[] mInputStreams; 65 // streams for writing to ALSA driver 66 private FileOutputStream[] mOutputStreams; 67 68 private final Object mLock = new Object(); 69 private boolean mIsOpen; 70 71 // pipe file descriptor for signalling input thread to exit 72 // only accessed from JNI code 73 private int mPipeFD = -1; 74 75 private final MidiDeviceServer.Callback mCallback = new MidiDeviceServer.Callback() { 76 77 @Override 78 public void onDeviceStatusChanged(MidiDeviceServer server, MidiDeviceStatus status) { 79 MidiDeviceInfo deviceInfo = status.getDeviceInfo(); 80 int inputPorts = deviceInfo.getInputPortCount(); 81 int outputPorts = deviceInfo.getOutputPortCount(); 82 boolean hasOpenPorts = false; 83 84 for (int i = 0; i < inputPorts; i++) { 85 if (status.isInputPortOpen(i)) { 86 hasOpenPorts = true; 87 break; 88 } 89 } 90 91 if (!hasOpenPorts) { 92 for (int i = 0; i < outputPorts; i++) { 93 if (status.getOutputPortOpenCount(i) > 0) { 94 hasOpenPorts = true; 95 break; 96 } 97 } 98 } 99 100 synchronized (mLock) { 101 if (hasOpenPorts && !mIsOpen) { 102 openLocked(); 103 } else if (!hasOpenPorts && mIsOpen) { 104 closeLocked(); 105 } 106 } 107 } 108 109 @Override 110 public void onClose() { 111 } 112 }; 113 114 // This class acts as a proxy for our MidiEventScheduler receivers, which do not exist 115 // until the device has active clients 116 private final class InputReceiverProxy extends MidiReceiver { 117 private MidiReceiver mReceiver; 118 119 @Override onSend(byte[] msg, int offset, int count, long timestamp)120 public void onSend(byte[] msg, int offset, int count, long timestamp) throws IOException { 121 MidiReceiver receiver = mReceiver; 122 if (receiver != null) { 123 receiver.send(msg, offset, count, timestamp); 124 } 125 } 126 setReceiver(MidiReceiver receiver)127 public void setReceiver(MidiReceiver receiver) { 128 mReceiver = receiver; 129 } 130 131 @Override onFlush()132 public void onFlush() throws IOException { 133 MidiReceiver receiver = mReceiver; 134 if (receiver != null) { 135 receiver.flush(); 136 } 137 } 138 } 139 create(Context context, Bundle properties, int card, int device)140 public static UsbMidiDevice create(Context context, Bundle properties, int card, int device) { 141 // FIXME - support devices with different number of input and output ports 142 int subDeviceCount = nativeGetSubdeviceCount(card, device); 143 if (subDeviceCount <= 0) { 144 Log.e(TAG, "nativeGetSubdeviceCount failed"); 145 return null; 146 } 147 148 UsbMidiDevice midiDevice = new UsbMidiDevice(card, device, subDeviceCount); 149 if (!midiDevice.register(context, properties)) { 150 IoUtils.closeQuietly(midiDevice); 151 Log.e(TAG, "createDeviceServer failed"); 152 return null; 153 } 154 return midiDevice; 155 } 156 UsbMidiDevice(int card, int device, int subdeviceCount)157 private UsbMidiDevice(int card, int device, int subdeviceCount) { 158 mAlsaCard = card; 159 mAlsaDevice = device; 160 mSubdeviceCount = subdeviceCount; 161 162 // FIXME - support devices with different number of input and output ports 163 int inputPortCount = subdeviceCount; 164 mInputPortReceivers = new InputReceiverProxy[inputPortCount]; 165 for (int port = 0; port < inputPortCount; port++) { 166 mInputPortReceivers[port] = new InputReceiverProxy(); 167 } 168 } 169 openLocked()170 private boolean openLocked() { 171 // FIXME - support devices with different number of input and output ports 172 FileDescriptor[] fileDescriptors = nativeOpen(mAlsaCard, mAlsaDevice, mSubdeviceCount); 173 if (fileDescriptors == null) { 174 Log.e(TAG, "nativeOpen failed"); 175 return false; 176 } 177 178 mFileDescriptors = fileDescriptors; 179 int inputStreamCount = fileDescriptors.length; 180 // last file descriptor returned from nativeOpen() is only used for unblocking Os.poll() 181 // in our input thread 182 int outputStreamCount = fileDescriptors.length - 1; 183 184 mPollFDs = new StructPollfd[inputStreamCount]; 185 mInputStreams = new FileInputStream[inputStreamCount]; 186 for (int i = 0; i < inputStreamCount; i++) { 187 FileDescriptor fd = fileDescriptors[i]; 188 StructPollfd pollfd = new StructPollfd(); 189 pollfd.fd = fd; 190 pollfd.events = (short)OsConstants.POLLIN; 191 mPollFDs[i] = pollfd; 192 mInputStreams[i] = new FileInputStream(fd); 193 } 194 195 mOutputStreams = new FileOutputStream[outputStreamCount]; 196 mEventSchedulers = new MidiEventScheduler[outputStreamCount]; 197 for (int i = 0; i < outputStreamCount; i++) { 198 mOutputStreams[i] = new FileOutputStream(fileDescriptors[i]); 199 200 MidiEventScheduler scheduler = new MidiEventScheduler(); 201 mEventSchedulers[i] = scheduler; 202 mInputPortReceivers[i].setReceiver(scheduler.getReceiver()); 203 } 204 205 final MidiReceiver[] outputReceivers = mServer.getOutputPortReceivers(); 206 207 // Create input thread which will read from all output ports of the physical device 208 new Thread("UsbMidiDevice input thread") { 209 @Override 210 public void run() { 211 byte[] buffer = new byte[BUFFER_SIZE]; 212 try { 213 while (true) { 214 // Record time of event immediately after waking. 215 long timestamp = System.nanoTime(); 216 synchronized (mLock) { 217 if (!mIsOpen) break; 218 219 // look for a readable FileDescriptor 220 for (int index = 0; index < mPollFDs.length; index++) { 221 StructPollfd pfd = mPollFDs[index]; 222 if ((pfd.revents & (OsConstants.POLLERR 223 | OsConstants.POLLHUP)) != 0) { 224 break; 225 } else if ((pfd.revents & OsConstants.POLLIN) != 0) { 226 // clear readable flag 227 pfd.revents = 0; 228 229 if (index == mInputStreams.length - 1) { 230 // last file descriptor is used only for unblocking Os.poll() 231 break; 232 } 233 234 int count = mInputStreams[index].read(buffer); 235 outputReceivers[index].send(buffer, 0, count, timestamp); 236 } 237 } 238 } 239 240 // wait until we have a readable port or we are signalled to close 241 Os.poll(mPollFDs, -1 /* infinite timeout */); 242 } 243 } catch (IOException e) { 244 Log.d(TAG, "reader thread exiting"); 245 } catch (ErrnoException e) { 246 Log.d(TAG, "reader thread exiting"); 247 } 248 Log.d(TAG, "input thread exit"); 249 } 250 }.start(); 251 252 // Create output thread for each input port of the physical device 253 for (int port = 0; port < outputStreamCount; port++) { 254 final MidiEventScheduler eventSchedulerF = mEventSchedulers[port]; 255 final FileOutputStream outputStreamF = mOutputStreams[port]; 256 final int portF = port; 257 258 new Thread("UsbMidiDevice output thread " + port) { 259 @Override 260 public void run() { 261 while (true) { 262 MidiEvent event; 263 try { 264 event = (MidiEvent)eventSchedulerF.waitNextEvent(); 265 } catch (InterruptedException e) { 266 // try again 267 continue; 268 } 269 if (event == null) { 270 break; 271 } 272 try { 273 outputStreamF.write(event.data, 0, event.count); 274 } catch (IOException e) { 275 Log.e(TAG, "write failed for port " + portF); 276 } 277 eventSchedulerF.addEventToPool(event); 278 } 279 Log.d(TAG, "output thread exit"); 280 } 281 }.start(); 282 } 283 284 mIsOpen = true; 285 return true; 286 } 287 register(Context context, Bundle properties)288 private boolean register(Context context, Bundle properties) { 289 MidiManager midiManager = (MidiManager)context.getSystemService(Context.MIDI_SERVICE); 290 if (midiManager == null) { 291 Log.e(TAG, "No MidiManager in UsbMidiDevice.create()"); 292 return false; 293 } 294 295 mServer = midiManager.createDeviceServer(mInputPortReceivers, mSubdeviceCount, 296 null, null, properties, MidiDeviceInfo.TYPE_USB, mCallback); 297 if (mServer == null) { 298 return false; 299 } 300 301 return true; 302 } 303 304 @Override close()305 public void close() throws IOException { 306 synchronized (mLock) { 307 if (mIsOpen) { 308 closeLocked(); 309 } 310 } 311 312 if (mServer != null) { 313 IoUtils.closeQuietly(mServer); 314 } 315 } 316 closeLocked()317 private void closeLocked() { 318 for (int i = 0; i < mEventSchedulers.length; i++) { 319 mInputPortReceivers[i].setReceiver(null); 320 mEventSchedulers[i].close(); 321 } 322 mEventSchedulers = null; 323 324 for (int i = 0; i < mInputStreams.length; i++) { 325 IoUtils.closeQuietly(mInputStreams[i]); 326 } 327 mInputStreams = null; 328 329 for (int i = 0; i < mOutputStreams.length; i++) { 330 IoUtils.closeQuietly(mOutputStreams[i]); 331 } 332 mOutputStreams = null; 333 334 // nativeClose will close the file descriptors and signal the input thread to exit 335 nativeClose(mFileDescriptors); 336 mFileDescriptors = null; 337 338 mIsOpen = false; 339 } 340 nativeGetSubdeviceCount(int card, int device)341 private static native int nativeGetSubdeviceCount(int card, int device); nativeOpen(int card, int device, int subdeviceCount)342 private native FileDescriptor[] nativeOpen(int card, int device, int subdeviceCount); nativeClose(FileDescriptor[] fileDescriptors)343 private native void nativeClose(FileDescriptor[] fileDescriptors); 344 } 345