1*90c8c64dSAndroid Build Coastguard Worker /* 2*90c8c64dSAndroid Build Coastguard Worker * Copyright (C) 2015 The Android Open Source Project 3*90c8c64dSAndroid Build Coastguard Worker * 4*90c8c64dSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*90c8c64dSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*90c8c64dSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*90c8c64dSAndroid Build Coastguard Worker * 8*90c8c64dSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*90c8c64dSAndroid Build Coastguard Worker * 10*90c8c64dSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*90c8c64dSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*90c8c64dSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*90c8c64dSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*90c8c64dSAndroid Build Coastguard Worker * limitations under the License. 15*90c8c64dSAndroid Build Coastguard Worker */ 16*90c8c64dSAndroid Build Coastguard Worker 17*90c8c64dSAndroid Build Coastguard Worker package com.example.android.common.midi; 18*90c8c64dSAndroid Build Coastguard Worker 19*90c8c64dSAndroid Build Coastguard Worker import android.media.midi.MidiReceiver; 20*90c8c64dSAndroid Build Coastguard Worker import android.util.Log; 21*90c8c64dSAndroid Build Coastguard Worker 22*90c8c64dSAndroid Build Coastguard Worker import java.io.IOException; 23*90c8c64dSAndroid Build Coastguard Worker 24*90c8c64dSAndroid Build Coastguard Worker /** 25*90c8c64dSAndroid Build Coastguard Worker * Convert stream of arbitrary MIDI bytes into discrete messages. 26*90c8c64dSAndroid Build Coastguard Worker * 27*90c8c64dSAndroid Build Coastguard Worker * Parses the incoming bytes and then posts individual messages to the receiver 28*90c8c64dSAndroid Build Coastguard Worker * specified in the constructor. Short messages of 1-3 bytes will be complete. 29*90c8c64dSAndroid Build Coastguard Worker * System Exclusive messages may be posted in pieces. 30*90c8c64dSAndroid Build Coastguard Worker * 31*90c8c64dSAndroid Build Coastguard Worker * Resolves Running Status and interleaved System Real-Time messages. 32*90c8c64dSAndroid Build Coastguard Worker */ 33*90c8c64dSAndroid Build Coastguard Worker public class MidiFramer extends MidiReceiver { 34*90c8c64dSAndroid Build Coastguard Worker private MidiReceiver mReceiver; 35*90c8c64dSAndroid Build Coastguard Worker private byte[] mBuffer = new byte[3]; 36*90c8c64dSAndroid Build Coastguard Worker private int mCount; 37*90c8c64dSAndroid Build Coastguard Worker private byte mRunningStatus; 38*90c8c64dSAndroid Build Coastguard Worker private int mNeeded; 39*90c8c64dSAndroid Build Coastguard Worker private boolean mInSysEx; 40*90c8c64dSAndroid Build Coastguard Worker MidiFramer(MidiReceiver receiver)41*90c8c64dSAndroid Build Coastguard Worker public MidiFramer(MidiReceiver receiver) { 42*90c8c64dSAndroid Build Coastguard Worker mReceiver = receiver; 43*90c8c64dSAndroid Build Coastguard Worker } 44*90c8c64dSAndroid Build Coastguard Worker 45*90c8c64dSAndroid Build Coastguard Worker /* 46*90c8c64dSAndroid Build Coastguard Worker * @see android.midi.MidiReceiver#onSend(byte[], int, int, long) 47*90c8c64dSAndroid Build Coastguard Worker */ 48*90c8c64dSAndroid Build Coastguard Worker @Override onSend(byte[] data, int offset, int count, long timestamp)49*90c8c64dSAndroid Build Coastguard Worker public void onSend(byte[] data, int offset, int count, long timestamp) 50*90c8c64dSAndroid Build Coastguard Worker throws IOException { 51*90c8c64dSAndroid Build Coastguard Worker int sysExStartOffset = (mInSysEx ? offset : -1); 52*90c8c64dSAndroid Build Coastguard Worker 53*90c8c64dSAndroid Build Coastguard Worker for (int i = 0; i < count; i++) { 54*90c8c64dSAndroid Build Coastguard Worker final byte currentByte = data[offset]; 55*90c8c64dSAndroid Build Coastguard Worker final int currentInt = currentByte & 0xFF; 56*90c8c64dSAndroid Build Coastguard Worker if (currentInt >= 0x80) { // status byte? 57*90c8c64dSAndroid Build Coastguard Worker if (currentInt < 0xF0) { // channel message? 58*90c8c64dSAndroid Build Coastguard Worker mRunningStatus = currentByte; 59*90c8c64dSAndroid Build Coastguard Worker mCount = 1; 60*90c8c64dSAndroid Build Coastguard Worker mNeeded = MidiConstants.getBytesPerMessage(currentByte) - 1; 61*90c8c64dSAndroid Build Coastguard Worker } else if (currentInt < 0xF8) { // system common? 62*90c8c64dSAndroid Build Coastguard Worker if (currentInt == 0xF0 /* SysEx Start */) { 63*90c8c64dSAndroid Build Coastguard Worker // Log.i(TAG, "SysEx Start"); 64*90c8c64dSAndroid Build Coastguard Worker mInSysEx = true; 65*90c8c64dSAndroid Build Coastguard Worker sysExStartOffset = offset; 66*90c8c64dSAndroid Build Coastguard Worker } else if (currentInt == 0xF7 /* SysEx End */) { 67*90c8c64dSAndroid Build Coastguard Worker // Log.i(TAG, "SysEx End"); 68*90c8c64dSAndroid Build Coastguard Worker if (mInSysEx) { 69*90c8c64dSAndroid Build Coastguard Worker mReceiver.send(data, sysExStartOffset, 70*90c8c64dSAndroid Build Coastguard Worker offset - sysExStartOffset + 1, timestamp); 71*90c8c64dSAndroid Build Coastguard Worker mInSysEx = false; 72*90c8c64dSAndroid Build Coastguard Worker sysExStartOffset = -1; 73*90c8c64dSAndroid Build Coastguard Worker } 74*90c8c64dSAndroid Build Coastguard Worker } else { 75*90c8c64dSAndroid Build Coastguard Worker mBuffer[0] = currentByte; 76*90c8c64dSAndroid Build Coastguard Worker mRunningStatus = 0; 77*90c8c64dSAndroid Build Coastguard Worker mCount = 1; 78*90c8c64dSAndroid Build Coastguard Worker mNeeded = MidiConstants.getBytesPerMessage(currentByte) - 1; 79*90c8c64dSAndroid Build Coastguard Worker } 80*90c8c64dSAndroid Build Coastguard Worker } else { // real-time? 81*90c8c64dSAndroid Build Coastguard Worker // Single byte message interleaved with other data. 82*90c8c64dSAndroid Build Coastguard Worker if (mInSysEx) { 83*90c8c64dSAndroid Build Coastguard Worker mReceiver.send(data, sysExStartOffset, 84*90c8c64dSAndroid Build Coastguard Worker offset - sysExStartOffset, timestamp); 85*90c8c64dSAndroid Build Coastguard Worker sysExStartOffset = offset + 1; 86*90c8c64dSAndroid Build Coastguard Worker } 87*90c8c64dSAndroid Build Coastguard Worker mReceiver.send(data, offset, 1, timestamp); 88*90c8c64dSAndroid Build Coastguard Worker } 89*90c8c64dSAndroid Build Coastguard Worker } else { // data byte 90*90c8c64dSAndroid Build Coastguard Worker if (!mInSysEx) { 91*90c8c64dSAndroid Build Coastguard Worker mBuffer[mCount++] = currentByte; 92*90c8c64dSAndroid Build Coastguard Worker if (--mNeeded == 0) { 93*90c8c64dSAndroid Build Coastguard Worker if (mRunningStatus != 0) { 94*90c8c64dSAndroid Build Coastguard Worker mBuffer[0] = mRunningStatus; 95*90c8c64dSAndroid Build Coastguard Worker } 96*90c8c64dSAndroid Build Coastguard Worker mReceiver.send(mBuffer, 0, mCount, timestamp); 97*90c8c64dSAndroid Build Coastguard Worker mNeeded = MidiConstants.getBytesPerMessage(mBuffer[0]) - 1; 98*90c8c64dSAndroid Build Coastguard Worker mCount = 1; 99*90c8c64dSAndroid Build Coastguard Worker } 100*90c8c64dSAndroid Build Coastguard Worker } 101*90c8c64dSAndroid Build Coastguard Worker } 102*90c8c64dSAndroid Build Coastguard Worker ++offset; 103*90c8c64dSAndroid Build Coastguard Worker } 104*90c8c64dSAndroid Build Coastguard Worker 105*90c8c64dSAndroid Build Coastguard Worker // send any accumulatedSysEx data 106*90c8c64dSAndroid Build Coastguard Worker if (sysExStartOffset >= 0 && sysExStartOffset < offset) { 107*90c8c64dSAndroid Build Coastguard Worker mReceiver.send(data, sysExStartOffset, 108*90c8c64dSAndroid Build Coastguard Worker offset - sysExStartOffset, timestamp); 109*90c8c64dSAndroid Build Coastguard Worker } 110*90c8c64dSAndroid Build Coastguard Worker } 111*90c8c64dSAndroid Build Coastguard Worker 112*90c8c64dSAndroid Build Coastguard Worker } 113