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