mirror of https://github.com/ARMmbed/mbed-os.git
				
				
				
			
		
			
				
	
	
		
			485 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
			
		
		
	
	
			485 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
/*
 | 
						|
 * Copyright (c) 2018-2019, Arm Limited and affiliates.
 | 
						|
 * SPDX-License-Identifier: Apache-2.0
 | 
						|
 *
 | 
						|
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
 * you may not use this file except in compliance with the License.
 | 
						|
 * You may obtain a copy of the License at
 | 
						|
 *
 | 
						|
 *     http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 *
 | 
						|
 * Unless required by applicable law or agreed to in writing, software
 | 
						|
 * distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
 * See the License for the specific language governing permissions and
 | 
						|
 * limitations under the License.
 | 
						|
 */
 | 
						|
 | 
						|
#ifndef MIDIMESSAGE_H
 | 
						|
#define MIDIMESSAGE_H
 | 
						|
 | 
						|
#include <stdint.h>
 | 
						|
 | 
						|
#define MAX_MIDI_MESSAGE_SIZE 256 // Max message size. SysEx can be up to 65536 but 256 should be fine for most usage
 | 
						|
 | 
						|
// MIDI Message Format
 | 
						|
//
 | 
						|
// [ msg(4) | channel(4) ] [ 0 | n(7) ] [ 0 | m(7) ]
 | 
						|
//
 | 
						|
// MIDI Data Messages (Channel Specific)
 | 
						|
//
 | 
						|
// Message               msg n          m
 | 
						|
// ---------------------------------------------
 | 
						|
// Note Off              0x8 Key        Velocity
 | 
						|
// Note On               0x9 Key        Velocity
 | 
						|
// Polyphonic Aftertouch 0xA Key        Pressure
 | 
						|
// Control Change        0xB Controller Value
 | 
						|
// Program Change        0xC Program    -
 | 
						|
// Channel Aftertouch    0xD Pressure   -
 | 
						|
// Pitch Wheel           0xE LSB        MSB
 | 
						|
 | 
						|
#define CABLE_NUM (0<<4)
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
 * \defgroup drivers_MIDIMessage MIDIMessage class
 | 
						|
 * \ingroup drivers-internal-api-usb
 | 
						|
 * @{
 | 
						|
 */
 | 
						|
 | 
						|
/** A MIDI message container */
 | 
						|
class MIDIMessage {
 | 
						|
public:
 | 
						|
 | 
						|
    MIDIMessage() : data(new uint8_t[MAX_MIDI_MESSAGE_SIZE + 1]), length(0) {}
 | 
						|
 | 
						|
    MIDIMessage(uint8_t *buf) : data(new uint8_t[MAX_MIDI_MESSAGE_SIZE + 1]), length(0)
 | 
						|
    {
 | 
						|
        for (int i = 0; i < 4; i++) {
 | 
						|
            data[i] = buf[i];
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Copy constructor
 | 
						|
     */
 | 
						|
    MIDIMessage(const MIDIMessage &other)
 | 
						|
    {
 | 
						|
        *this = other;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Assignment operator
 | 
						|
     */
 | 
						|
    MIDIMessage &operator=(const MIDIMessage &other)
 | 
						|
    {
 | 
						|
        length = other.length;
 | 
						|
        for (int i = 0; i < length; i++) {
 | 
						|
            data[i] = other.data[i];
 | 
						|
        }
 | 
						|
 | 
						|
        return *this;
 | 
						|
    }
 | 
						|
 | 
						|
    ~MIDIMessage()
 | 
						|
    {
 | 
						|
        delete[] data;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Set this MIDIMessage to a raw MIDI message
 | 
						|
     *
 | 
						|
     * @param buf is a true MIDI message (not USBMidi message)
 | 
						|
     * @param buf_len size of message
 | 
						|
     */
 | 
						|
    void from_raw(uint8_t *buf, int buf_len)
 | 
						|
    {
 | 
						|
        length = buf_len + 1;
 | 
						|
        if (length > MAX_MIDI_MESSAGE_SIZE) {
 | 
						|
            // Message is too big
 | 
						|
            length = 0;
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        // first byte keeped for retro-compatibility
 | 
						|
        data[0] = CABLE_NUM | 0x08;
 | 
						|
 | 
						|
        for (int i = 0; i < buf_len; i++) {
 | 
						|
            data[i + 1] = buf[i];
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    // create messages
 | 
						|
 | 
						|
    /** Create a NoteOff message
 | 
						|
     * @param key Key ID
 | 
						|
     * @param velocity Key velocity (0-127, default = 127)
 | 
						|
     * @param channel Key channel (0-15, default 0)
 | 
						|
     * @returns A MIDIMessage
 | 
						|
     */
 | 
						|
    static MIDIMessage NoteOff(int key, int velocity = 127, int channel = 0)
 | 
						|
    {
 | 
						|
        MIDIMessage msg;
 | 
						|
        msg.data[0] = CABLE_NUM | 0x08;
 | 
						|
        msg.data[1] = 0x80 | (channel & 0x0F);
 | 
						|
        msg.data[2] = key & 0x7F;
 | 
						|
        msg.data[3] = velocity & 0x7F;
 | 
						|
        msg.length = 4;
 | 
						|
        return msg;
 | 
						|
    }
 | 
						|
 | 
						|
    /** Create a NoteOn message
 | 
						|
     * @param key Key ID
 | 
						|
     * @param velocity Key velocity (0-127, default = 127)
 | 
						|
     * @param channel Key channel (0-15, default 0)
 | 
						|
     * @returns A MIDIMessage
 | 
						|
     */
 | 
						|
    static MIDIMessage NoteOn(int key, int velocity = 127, int channel = 0)
 | 
						|
    {
 | 
						|
        MIDIMessage msg;
 | 
						|
        msg.data[0] = CABLE_NUM | 0x09;
 | 
						|
        msg.data[1] = 0x90 | (channel & 0x0F);
 | 
						|
        msg.data[2] = key & 0x7F;
 | 
						|
        msg.data[3] = velocity & 0x7F;
 | 
						|
        msg.length = 4;
 | 
						|
        return msg;
 | 
						|
    }
 | 
						|
 | 
						|
    /** Create a PolyPhonic Aftertouch message
 | 
						|
     * @param key Key ID
 | 
						|
     * @param pressure Aftertouch pressure (0-127)
 | 
						|
     * @param channel Key channel (0-15, default 0)
 | 
						|
     * @returns A MIDIMessage
 | 
						|
     */
 | 
						|
    static MIDIMessage PolyphonicAftertouch(int key, int pressure, int channel = 0)
 | 
						|
    {
 | 
						|
        MIDIMessage msg;
 | 
						|
        msg.data[0] = CABLE_NUM | 0x0A;
 | 
						|
        msg.data[1] = 0xA0 | (channel & 0x0F);
 | 
						|
        msg.data[2] = key & 0x7F;
 | 
						|
        msg.data[3] = pressure & 0x7F;
 | 
						|
        msg.length = 4;
 | 
						|
        return msg;
 | 
						|
    }
 | 
						|
 | 
						|
    /** Create a Control Change message
 | 
						|
     * @param control Controller ID
 | 
						|
     * @param value Controller value (0-127)
 | 
						|
     * @param channel Controller channel (0-15, default 0)
 | 
						|
     * @returns A MIDIMessage
 | 
						|
     */
 | 
						|
    static MIDIMessage ControlChange(int control, int value, int channel = 0)
 | 
						|
    {
 | 
						|
        MIDIMessage msg;
 | 
						|
        msg.data[0] = CABLE_NUM | 0x0B;
 | 
						|
        msg.data[1] = 0xB0 | (channel & 0x0F);
 | 
						|
        msg.data[2] = control & 0x7F;
 | 
						|
        msg.data[3] = value & 0x7F;
 | 
						|
        msg.length = 4;
 | 
						|
        return msg;
 | 
						|
    }
 | 
						|
 | 
						|
    /** Create a Program Change message
 | 
						|
     * @param program Program ID
 | 
						|
     * @param channel Channel (0-15, default 0)
 | 
						|
     * @returns A MIDIMessage
 | 
						|
     */
 | 
						|
    static MIDIMessage ProgramChange(int program, int channel = 0)
 | 
						|
    {
 | 
						|
        MIDIMessage msg;
 | 
						|
        msg.data[0] = CABLE_NUM | 0x0C;
 | 
						|
        msg.data[1] = 0xC0 | (channel & 0x0F);
 | 
						|
        msg.data[2] = program & 0x7F;
 | 
						|
        msg.data[3] = 0x00;
 | 
						|
        msg.length = 4;
 | 
						|
        return msg;
 | 
						|
    }
 | 
						|
 | 
						|
    /** Create a Channel Aftertouch message
 | 
						|
     * @param pressure Pressure
 | 
						|
     * @param channel Key channel (0-15, default 0)
 | 
						|
     * @returns A MIDIMessage
 | 
						|
     */
 | 
						|
    static MIDIMessage ChannelAftertouch(int pressure, int channel = 0)
 | 
						|
    {
 | 
						|
        MIDIMessage msg;
 | 
						|
        msg.data[0] = CABLE_NUM | 0x0D;
 | 
						|
        msg.data[1] = 0xD0 | (channel & 0x0F);
 | 
						|
        msg.data[2] = pressure & 0x7F;
 | 
						|
        msg.data[3] = 0x00;
 | 
						|
        msg.length = 4;
 | 
						|
        return msg;
 | 
						|
    }
 | 
						|
 | 
						|
    /** Create a Pitch Wheel message
 | 
						|
     * @param pitch Pitch (-8192 - 8191, default = 0)
 | 
						|
     * @param channel Channel (0-15, default 0)
 | 
						|
     * @returns A MIDIMessage
 | 
						|
     */
 | 
						|
    static MIDIMessage PitchWheel(int pitch = 0, int channel = 0)
 | 
						|
    {
 | 
						|
        MIDIMessage msg;
 | 
						|
        int p = pitch + 8192;    // 0 - 16383, 8192 is center
 | 
						|
        msg.data[0] = CABLE_NUM | 0x0E;
 | 
						|
        msg.data[1] = 0xE0 | (channel & 0x0F);
 | 
						|
        msg.data[2] = p & 0x7F;
 | 
						|
        msg.data[3] = (p >> 7) & 0x7F;
 | 
						|
        msg.length = 4;
 | 
						|
        return msg;
 | 
						|
    }
 | 
						|
 | 
						|
    /** Create an All Notes Off message
 | 
						|
     * @param channel Channel (0-15, default 0)
 | 
						|
     * @returns A MIDIMessage
 | 
						|
     */
 | 
						|
    static MIDIMessage AllNotesOff(int channel = 0)
 | 
						|
    {
 | 
						|
        return ControlChange(123, 0, channel);
 | 
						|
    }
 | 
						|
 | 
						|
    /** Create a SysEx message
 | 
						|
    * @param data SysEx data (including 0xF0 .. 0xF7)
 | 
						|
    * @param len SysEx data length
 | 
						|
    * @returns A MIDIMessage
 | 
						|
    */
 | 
						|
    static MIDIMessage SysEx(uint8_t *data, int len)
 | 
						|
    {
 | 
						|
        MIDIMessage msg;
 | 
						|
        msg.from_raw(data, len);
 | 
						|
        return msg;
 | 
						|
    }
 | 
						|
 | 
						|
    // decode messages
 | 
						|
 | 
						|
    /** MIDI Message Types */
 | 
						|
    enum MIDIMessageType {
 | 
						|
        ErrorType,
 | 
						|
        NoteOffType,
 | 
						|
        NoteOnType,
 | 
						|
        PolyphonicAftertouchType,
 | 
						|
        ControlChangeType,
 | 
						|
        ProgramChangeType,
 | 
						|
        ChannelAftertouchType,
 | 
						|
        PitchWheelType,
 | 
						|
        ResetAllControllersType,
 | 
						|
        AllNotesOffType,
 | 
						|
        SysExType
 | 
						|
    };
 | 
						|
 | 
						|
    /** Read the message type
 | 
						|
     *
 | 
						|
     * @returns MIDIMessageType
 | 
						|
     */
 | 
						|
    MIDIMessageType type()
 | 
						|
    {
 | 
						|
        MIDIMessageType message_type;
 | 
						|
        uint8_t min_size;
 | 
						|
        switch ((data[1] >> 4) & 0xF) {
 | 
						|
            case 0x8:
 | 
						|
                // message, channel
 | 
						|
                // key
 | 
						|
                // velocity
 | 
						|
                min_size = 3;
 | 
						|
                message_type = NoteOffType;
 | 
						|
                break;
 | 
						|
            case 0x9:
 | 
						|
                // message, channel
 | 
						|
                // key
 | 
						|
                // velocity
 | 
						|
                min_size = 3;
 | 
						|
                message_type = NoteOnType;
 | 
						|
                break;
 | 
						|
            case 0xA:
 | 
						|
                // message, channel
 | 
						|
                // key
 | 
						|
                // pressure
 | 
						|
                min_size = 3;
 | 
						|
                message_type = PolyphonicAftertouchType;
 | 
						|
                break;
 | 
						|
            case 0xB:
 | 
						|
                // message, channel
 | 
						|
                // controller
 | 
						|
                min_size = 2;
 | 
						|
                if ((data[2] & 0x7F) < 120) { // standard controllers
 | 
						|
                    message_type = ControlChangeType;
 | 
						|
                } else if ((data[2] & 0x7F) == 121) {
 | 
						|
                    message_type = ResetAllControllersType;
 | 
						|
                } else if ((data[2] & 0x7F) == 123) {
 | 
						|
                    message_type = AllNotesOffType;
 | 
						|
                } else {
 | 
						|
                    message_type = ErrorType; // unsupported atm
 | 
						|
                }
 | 
						|
                break;
 | 
						|
            case 0xC:
 | 
						|
                // message, channel
 | 
						|
                // program
 | 
						|
                min_size = 2;
 | 
						|
                message_type = ProgramChangeType;
 | 
						|
                break;
 | 
						|
            case 0xD:
 | 
						|
                // message, channel
 | 
						|
                // pressure
 | 
						|
                min_size = 2;
 | 
						|
                message_type = ChannelAftertouchType;
 | 
						|
                break;
 | 
						|
            case 0xE:
 | 
						|
                // message, channel
 | 
						|
                // pitch lsb
 | 
						|
                // pitch msb
 | 
						|
                min_size = 3;
 | 
						|
                message_type = PitchWheelType;
 | 
						|
                break;
 | 
						|
            case 0xF:
 | 
						|
                min_size = 2;
 | 
						|
                message_type = SysExType;
 | 
						|
                break;
 | 
						|
            default:
 | 
						|
                message_type = ErrorType;
 | 
						|
                break;
 | 
						|
        }
 | 
						|
 | 
						|
 | 
						|
        if (length < min_size) {
 | 
						|
            // too small to be a valid message
 | 
						|
            message_type = ErrorType;
 | 
						|
        }
 | 
						|
        return message_type;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Read the channel number
 | 
						|
     *
 | 
						|
     * @return channel number or -1 on error
 | 
						|
     */
 | 
						|
 | 
						|
    int channel()
 | 
						|
    {
 | 
						|
        return (data[1] & 0x0F);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Read the key ID
 | 
						|
     *
 | 
						|
     * @return key ID or -1 on error
 | 
						|
     */
 | 
						|
    int key()
 | 
						|
    {
 | 
						|
        MIDIMessageType msg_type = type();
 | 
						|
        if ((msg_type != NoteOffType) &&
 | 
						|
                (msg_type != NoteOnType) &&
 | 
						|
                (msg_type != PolyphonicAftertouchType)) {
 | 
						|
            return -1;
 | 
						|
        }
 | 
						|
 | 
						|
        return data[2] & 0x7F;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Read the velocity
 | 
						|
     *
 | 
						|
     * @return velocity or -1 on error
 | 
						|
     */
 | 
						|
    int velocity()
 | 
						|
    {
 | 
						|
        MIDIMessageType msg_type = type();
 | 
						|
        if ((msg_type != NoteOffType) &&
 | 
						|
                (msg_type != NoteOnType)) {
 | 
						|
            return -1;
 | 
						|
        }
 | 
						|
 | 
						|
        return data[3] & 0x7F;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Read the controller value
 | 
						|
     *
 | 
						|
     * @return controller value or -1 on error
 | 
						|
     */
 | 
						|
    int value()
 | 
						|
    {
 | 
						|
        MIDIMessageType msg_type = type();
 | 
						|
        if ((msg_type != ControlChangeType) &&
 | 
						|
                (msg_type != ResetAllControllersType) &&
 | 
						|
                (msg_type != AllNotesOffType)) {
 | 
						|
            return -1;
 | 
						|
        }
 | 
						|
 | 
						|
        return data[3] & 0x7F;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Read the aftertouch pressure
 | 
						|
     *
 | 
						|
     * @return aftertouch pressure or -1 on error
 | 
						|
     */
 | 
						|
    int pressure()
 | 
						|
    {
 | 
						|
        MIDIMessageType msg_type = type();
 | 
						|
        if ((msg_type != PolyphonicAftertouchType) &&
 | 
						|
                (msg_type != ChannelAftertouchType)) {
 | 
						|
            return -1;
 | 
						|
        }
 | 
						|
 | 
						|
        if (type() == PolyphonicAftertouchType) {
 | 
						|
            return data[3] & 0x7F;
 | 
						|
        } else {
 | 
						|
            return data[2] & 0x7F;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Read the controller number
 | 
						|
     *
 | 
						|
     * @return controller number or -1 on error
 | 
						|
     */
 | 
						|
    int controller()
 | 
						|
    {
 | 
						|
        MIDIMessageType msg_type = type();
 | 
						|
        if ((msg_type != ControlChangeType) &&
 | 
						|
                (msg_type != ResetAllControllersType) &&
 | 
						|
                (msg_type != AllNotesOffType)) {
 | 
						|
            return -1;
 | 
						|
        }
 | 
						|
 | 
						|
        return data[2] & 0x7F;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Read the program number
 | 
						|
     *
 | 
						|
     * @return program number or -1 on error
 | 
						|
     */
 | 
						|
    int program()
 | 
						|
    {
 | 
						|
        MIDIMessageType msg_type = type();
 | 
						|
        if (msg_type != ProgramChangeType) {
 | 
						|
            return -1;
 | 
						|
        }
 | 
						|
 | 
						|
        return data[2] & 0x7F;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Read the pitch value
 | 
						|
     *
 | 
						|
     * @return pitch value or -1 on error
 | 
						|
     */
 | 
						|
    int pitch()
 | 
						|
    {
 | 
						|
        MIDIMessageType msg_type = type();
 | 
						|
        if (msg_type != PitchWheelType) {
 | 
						|
            return -1;
 | 
						|
        }
 | 
						|
 | 
						|
        int p = ((data[3] & 0x7F) << 7) | (data[2] & 0x7F);
 | 
						|
        return p - 8192; // 0 - 16383, 8192 is center
 | 
						|
    }
 | 
						|
 | 
						|
    uint8_t *data;
 | 
						|
    uint16_t length;
 | 
						|
};
 | 
						|
 | 
						|
/** @}*/
 | 
						|
 | 
						|
#endif
 |