/* ******************************************************************************* * USB-MIDI class driver for USB Host Shield 2.0 Library * Copyright (c) 2012-2018 Yuuichi Akagawa * * Idea from LPK25 USB-MIDI to Serial MIDI converter * by Collin Cunningham - makezine.com, narbotic.com * * for use with USB Host Shield 2.0 from Circuitsathome.com * https://github.com/felis/USB_Host_Shield_2.0 ******************************************************************************* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see ******************************************************************************* */ #include "usbh_midi.h" ////////////////////////// // MIDI MESAGES // midi.org/techspecs/ ////////////////////////// // STATUS BYTES // 0x8n == noteOff // 0x9n == noteOn // 0xAn == afterTouch // 0xBn == controlChange // n == Channel(0x0-0xf) ////////////////////////// //DATA BYTE 1 // note# == (0-127) // or // control# == (0-119) ////////////////////////// // DATA BYTE 2 // velocity == (0-127) // or // controlVal == (0-127) /////////////////////////////////////////////////////////////////////////////// // USB-MIDI Event Packets // usb.org - Universal Serial Bus Device Class Definition for MIDI Devices 1.0 /////////////////////////////////////////////////////////////////////////////// //+-------------+-------------+-------------+-------------+ //| Byte 0 | Byte 1 | Byte 2 | Byte 3 | //+------+------+-------------+-------------+-------------+ //|Cable | Code | | | | //|Number|Index | MIDI_0 | MIDI_1 | MIDI_2 | //| |Number| | | | //|(4bit)|(4bit)| (8bit) | (8bit) | (8bit) | //+------+------+-------------+-------------+-------------+ // CN == 0x0-0xf //+-----+-----------+------------------------------------------------------------------- //| CIN |MIDI_x size|Description //+-----+-----------+------------------------------------------------------------------- //| 0x0 | 1, 2 or 3 |Miscellaneous function codes. Reserved for future extensions. //| 0x1 | 1, 2 or 3 |Cable events. Reserved for future expansion. //| 0x2 | 2 |Two-byte System Common messages like MTC, SongSelect, etc. //| 0x3 | 3 |Three-byte System Common messages like SPP, etc. //| 0x4 | 3 |SysEx starts or continues //| 0x5 | 1 |Single-byte System Common Message or SysEx ends with following single byte. //| 0x6 | 2 |SysEx ends with following two bytes. //| 0x7 | 3 |SysEx ends with following three bytes. //| 0x8 | 3 |Note-off //| 0x9 | 3 |Note-on //| 0xA | 3 |Poly-KeyPress //| 0xB | 3 |Control Change //| 0xC | 2 |Program Change //| 0xD | 2 |Channel Pressure //| 0xE | 3 |PitchBend Change //| 0xF | 1 |Single Byte //+-----+-----------+------------------------------------------------------------------- const uint8_t USBH_MIDI::epDataInIndex = 1; const uint8_t USBH_MIDI::epDataOutIndex = 2; const uint8_t USBH_MIDI::epDataInIndexVSP = 3; const uint8_t USBH_MIDI::epDataOutIndexVSP = 4; USBH_MIDI::USBH_MIDI(USB *p) : pUsb(p), bAddress(0), bNumEP(1), bPollEnable(false), isMidiFound(false), readPtr(0) { // initialize endpoint data structures for(uint8_t i=0; iRegisterDeviceClass(this); } } /* Connection initialization of an MIDI Device */ uint8_t USBH_MIDI::Init(uint8_t parent, uint8_t port, bool lowspeed) { uint8_t buf[sizeof (USB_DEVICE_DESCRIPTOR)]; USB_DEVICE_DESCRIPTOR * udd = reinterpret_cast(buf); uint8_t rcode; UsbDevice *p = NULL; EpInfo *oldep_ptr = NULL; uint8_t num_of_conf; // number of configurations USBTRACE("\rMIDI Init\r\n"); //for reconnect for(uint8_t i=epDataInIndex; i<=epDataOutIndex; i++) { epInfo[i].epAddr = (i==epDataInIndex) ? 0x81 : 0x01; epInfo[i].maxPktSize = 0; epInfo[i].bmSndToggle = 0; epInfo[i].bmRcvToggle = 0; } // get memory address of USB device address pool AddressPool &addrPool = pUsb->GetAddressPool(); // check if address has already been assigned to an instance if (bAddress) { return USB_ERROR_CLASS_INSTANCE_ALREADY_IN_USE; } // Get pointer to pseudo device with address 0 assigned p = addrPool.GetUsbDevicePtr(bAddress); if (!p) { return USB_ERROR_ADDRESS_NOT_FOUND_IN_POOL; } if (!p->epinfo) { return USB_ERROR_EPINFO_IS_NULL; } // Save old pointer to EP_RECORD of address 0 oldep_ptr = p->epinfo; // Temporary assign new pointer to epInfo to p->epinfo in order to avoid toggle inconsistence p->epinfo = epInfo; p->lowspeed = lowspeed; // Get device descriptor rcode = pUsb->getDevDescr( 0, 0, sizeof(USB_DEVICE_DESCRIPTOR), (uint8_t*)buf ); vid = udd->idVendor; pid = udd->idProduct; // Restore p->epinfo p->epinfo = oldep_ptr; if( rcode ){ goto FailGetDevDescr; } // Allocate new address according to device class bAddress = addrPool.AllocAddress(parent, false, port); if (!bAddress) { return USB_ERROR_OUT_OF_ADDRESS_SPACE_IN_POOL; } // Extract Max Packet Size from device descriptor epInfo[0].maxPktSize = udd->bMaxPacketSize0; // Assign new address to the device rcode = pUsb->setAddr( 0, 0, bAddress ); if (rcode) { p->lowspeed = false; addrPool.FreeAddress(bAddress); bAddress = 0; return rcode; }//if (rcode... USBTRACE2("Addr:", bAddress); p->lowspeed = false; //get pointer to assigned address record p = addrPool.GetUsbDevicePtr(bAddress); if (!p) { return USB_ERROR_ADDRESS_NOT_FOUND_IN_POOL; } p->lowspeed = lowspeed; num_of_conf = udd->bNumConfigurations; // Assign epInfo to epinfo pointer rcode = pUsb->setEpInfoEntry(bAddress, 1, epInfo); if (rcode) { USBTRACE("setEpInfoEntry failed"); goto FailSetDevTblEntry; } USBTRACE("VID:"), D_PrintHex(vid, 0x80); USBTRACE(" PID:"), D_PrintHex(pid, 0x80); USBTRACE2(" #Conf:", num_of_conf); //Setup for well known vendor/device specific configuration bTransferTypeMask = bmUSB_TRANSFER_TYPE; setupDeviceSpecific(); isMidiFound = false; for (uint8_t i=0; i 1) break; } // for USBTRACE2("\r\nNumEP:", bNumEP); if( bNumEP < 2 ){ //Device not found. rcode = 0xff; goto FailGetConfDescr; } if( !isMidiFound ){ //MIDI Device not found. Try last Bulk transfer device USBTRACE("MIDI not found. Attempts bulk device\r\n"); epInfo[epDataInIndex].epAddr = epInfo[epDataInIndexVSP].epAddr; epInfo[epDataInIndex].maxPktSize = epInfo[epDataInIndexVSP].maxPktSize; epInfo[epDataOutIndex].epAddr = epInfo[epDataOutIndexVSP].epAddr; epInfo[epDataOutIndex].maxPktSize = epInfo[epDataOutIndexVSP].maxPktSize; } // Assign epInfo to epinfo pointer rcode = pUsb->setEpInfoEntry(bAddress, 3, epInfo); USBTRACE2("Conf:", bConfNum); USBTRACE2("EPin :", (uint8_t)(epInfo[epDataInIndex].epAddr + 0x80)); USBTRACE2("EPout:", epInfo[epDataOutIndex].epAddr); // Set Configuration Value rcode = pUsb->setConf(bAddress, 0, bConfNum); if (rcode) { goto FailSetConfDescr; } bPollEnable = true; USBTRACE("Init done.\r\n"); return 0; FailGetDevDescr: FailSetDevTblEntry: FailGetConfDescr: FailSetConfDescr: Release(); return rcode; } /* get and parse config descriptor */ uint8_t USBH_MIDI::parseConfigDescr( uint8_t addr, uint8_t conf ) { uint8_t buf[ DESC_BUFF_SIZE ]; uint8_t* buf_ptr = buf; uint8_t rcode; uint8_t descr_length; uint8_t descr_type; uint16_t total_length; USB_ENDPOINT_DESCRIPTOR *epDesc; bool isMidi = false; // get configuration descriptor (get descriptor size only) rcode = pUsb->getConfDescr( addr, 0, 4, conf, buf ); if( rcode ){ return rcode; } total_length = buf[2] | ((int)buf[3] << 8); if( total_length > DESC_BUFF_SIZE ) { //check if total length is larger than buffer total_length = DESC_BUFF_SIZE; } // get configuration descriptor (all) rcode = pUsb->getConfDescr( addr, 0, total_length, conf, buf ); //get the whole descriptor if( rcode ){ return rcode; } //parsing descriptors while( buf_ptr < buf + total_length ) { descr_length = *( buf_ptr ); descr_type = *( buf_ptr + 1 ); switch( descr_type ) { case USB_DESCRIPTOR_CONFIGURATION : bConfNum = buf_ptr[5]; break; case USB_DESCRIPTOR_INTERFACE : USBTRACE("\r\nConf:"), D_PrintHex(bConfNum, 0x80); USBTRACE(" Int:"), D_PrintHex(buf_ptr[2], 0x80); USBTRACE(" Alt:"), D_PrintHex(buf_ptr[3], 0x80); USBTRACE(" EPs:"), D_PrintHex(buf_ptr[4], 0x80); USBTRACE(" IntCl:"), D_PrintHex(buf_ptr[5], 0x80); USBTRACE(" IntSubCl:"), D_PrintHex(buf_ptr[6], 0x80); USBTRACE("\r\n"); if( buf_ptr[5] == USB_CLASS_AUDIO && buf_ptr[6] == USB_SUBCLASS_MIDISTREAMING ) { //p[5]; bInterfaceClass = 1(Audio), p[6]; bInterfaceSubClass = 3(MIDI Streaming) isMidiFound = true; //MIDI device found. isMidi = true; USBTRACE("MIDI Device\r\n"); }else{ isMidi = false; USBTRACE("No MIDI Device\r\n"); } break; case USB_DESCRIPTOR_ENDPOINT : epDesc = (USB_ENDPOINT_DESCRIPTOR *)buf_ptr; USBTRACE("-EPAddr:"), D_PrintHex(epDesc->bEndpointAddress, 0x80); USBTRACE(" bmAttr:"), D_PrintHex(epDesc->bmAttributes, 0x80); USBTRACE2(" MaxPktSz:", (uint8_t)epDesc->wMaxPacketSize); if ((epDesc->bmAttributes & bTransferTypeMask) == USB_TRANSFER_TYPE_BULK) {//bulk uint8_t index; if( isMidi ) index = ((epDesc->bEndpointAddress & 0x80) == 0x80) ? epDataInIndex : epDataOutIndex; else index = ((epDesc->bEndpointAddress & 0x80) == 0x80) ? epDataInIndexVSP : epDataOutIndexVSP; epInfo[index].epAddr = (epDesc->bEndpointAddress & 0x0F); epInfo[index].maxPktSize = (uint8_t)epDesc->wMaxPacketSize; bNumEP ++; #ifdef DEBUG_USB_HOST PrintEndpointDescriptor(epDesc); #endif } break; default: break; }//switch( descr_type buf_ptr += descr_length; //advance buffer pointer }//while( buf_ptr <=... return 0; } /* Performs a cleanup after failed Init() attempt */ uint8_t USBH_MIDI::Release() { pUsb->GetAddressPool().FreeAddress(bAddress); bNumEP = 1; //must have to be reset to 1 bAddress = 0; bPollEnable = false; readPtr = 0; return 0; } /* Setup for well known vendor/device specific configuration */ void USBH_MIDI::setupDeviceSpecific() { // Novation if( vid == 0x1235 ) { // LaunchPad's endpoint attirbute is interrupt (0x20:S, 0x36:Mini, 0x51:Pro, 0x69:MK2, 0x7b:Launchkey25 MK2) if(pid == 0x20 || pid == 0x36 || pid == 0x51 || pid == 0x69 || pid == 0x7b ) { bTransferTypeMask = 2; } } } /* Receive data from MIDI device */ uint8_t USBH_MIDI::RecvData(uint16_t *bytes_rcvd, uint8_t *dataptr) { *bytes_rcvd = (uint16_t)epInfo[epDataInIndex].maxPktSize; uint8_t r = pUsb->inTransfer(bAddress, epInfo[epDataInIndex].epAddr, bytes_rcvd, dataptr); if( *bytes_rcvd < (MIDI_EVENT_PACKET_SIZE-4)){ dataptr[*bytes_rcvd] = '\0'; dataptr[(*bytes_rcvd)+1] = '\0'; } return r; } /* Receive data from MIDI device */ uint8_t USBH_MIDI::RecvData(uint8_t *outBuf, bool isRaw) { uint8_t rcode = 0; //return code uint16_t rcvd; if( bPollEnable == false ) return 0; //Checking unprocessed message in buffer. if( readPtr != 0 && readPtr < MIDI_EVENT_PACKET_SIZE ){ if(recvBuf[readPtr] == 0 && recvBuf[readPtr+1] == 0) { //no unprocessed message left in the buffer. }else{ goto RecvData_return_from_buffer; } } readPtr = 0; rcode = RecvData( &rcvd, recvBuf); if( rcode != 0 ) { return 0; } //if all data is zero, no valid data received. if( recvBuf[0] == 0 && recvBuf[1] == 0 && recvBuf[2] == 0 && recvBuf[3] == 0 ) { return 0; } RecvData_return_from_buffer: uint8_t m; uint8_t cin = recvBuf[readPtr]; if( isRaw == true ) { *(outBuf++) = cin; } readPtr++; *(outBuf++) = m = recvBuf[readPtr++]; *(outBuf++) = recvBuf[readPtr++]; *(outBuf++) = recvBuf[readPtr++]; return lookupMsgSize(m, cin); } /* Receive raw data from MIDI device */ uint8_t USBH_MIDI::RecvRawData(uint8_t *outBuf) { return RecvData(outBuf, true); } /* Send data to MIDI device */ uint8_t USBH_MIDI::SendData(uint8_t *dataptr, uint8_t nCable) { uint8_t buf[4]; uint8_t msg; msg = dataptr[0]; // SysEx long message ? if( msg == 0xf0 ) { return SendSysEx(dataptr, countSysExDataSize(dataptr), nCable); } buf[0] = (nCable << 4) | (msg >> 4); if( msg < 0xf0 ) msg = msg & 0xf0; //Building USB-MIDI Event Packets buf[1] = dataptr[0]; buf[2] = dataptr[1]; buf[3] = dataptr[2]; switch(lookupMsgSize(msg)) { //3 bytes message case 3 : if(msg == 0xf2) {//system common message(SPP) buf[0] = (nCable << 4) | 3; } break; //2 bytes message case 2 : if(msg == 0xf1 || msg == 0xf3) {//system common message(MTC/SongSelect) buf[0] = (nCable << 4) | 2; } buf[3] = 0; break; //1 byte message case 1 : default : buf[2] = 0; buf[3] = 0; break; } return pUsb->outTransfer(bAddress, epInfo[epDataOutIndex].epAddr, 4, buf); } #ifdef DEBUG_USB_HOST void USBH_MIDI::PrintEndpointDescriptor( const USB_ENDPOINT_DESCRIPTOR* ep_ptr ) { USBTRACE("Endpoint descriptor:\r\n"); USBTRACE2(" Length:\t", ep_ptr->bLength); USBTRACE2(" Type:\t\t", ep_ptr->bDescriptorType); USBTRACE2(" Address:\t", ep_ptr->bEndpointAddress); USBTRACE2(" Attributes:\t", ep_ptr->bmAttributes); USBTRACE2(" MaxPktSize:\t", ep_ptr->wMaxPacketSize); USBTRACE2(" Poll Intrv:\t", ep_ptr->bInterval); } #endif /* look up a MIDI message size from spec */ /*Return */ /* 0 : undefined message */ /* 0<: Vaild message size(1-3) */ uint8_t USBH_MIDI::lookupMsgSize(uint8_t midiMsg, uint8_t cin) { uint8_t msgSize = 0; //SysEx message? cin = cin & 0x0f; if( (cin & 0xc) == 4 ) { if( cin == 4 || cin == 7 ) return 3; if( cin == 6 ) return 2; if( cin == 5 ) return 1; } if( midiMsg < 0xf0 ) midiMsg &= 0xf0; switch(midiMsg) { //3 bytes messages case 0xf2 : //system common message(SPP) case 0x80 : //Note off case 0x90 : //Note on case 0xa0 : //Poly KeyPress case 0xb0 : //Control Change case 0xe0 : //PitchBend Change msgSize = 3; break; //2 bytes messages case 0xf1 : //system common message(MTC) case 0xf3 : //system common message(SongSelect) case 0xc0 : //Program Change case 0xd0 : //Channel Pressure msgSize = 2; break; //1 byte messages case 0xf8 : //system realtime message case 0xf9 : //system realtime message case 0xfa : //system realtime message case 0xfb : //system realtime message case 0xfc : //system realtime message case 0xfe : //system realtime message case 0xff : //system realtime message msgSize = 1; break; //undefine messages default : break; } return msgSize; } /* SysEx data size counter */ uint16_t USBH_MIDI::countSysExDataSize(uint8_t *dataptr) { uint16_t c = 1; if( *dataptr != 0xf0 ){ //not SysEx return 0; } //Search terminator(0xf7) while(*dataptr != 0xf7) { dataptr++; c++; //Limiter (default: 256 bytes) if(c > MIDI_MAX_SYSEX_SIZE){ c = 0; break; } } return c; } /* Send SysEx message to MIDI device */ uint8_t USBH_MIDI::SendSysEx(uint8_t *dataptr, uint16_t datasize, uint8_t nCable) { uint8_t buf[MIDI_EVENT_PACKET_SIZE]; uint8_t rc = 0; uint16_t n = datasize; uint16_t pktSize = (n*10/3+7)/10*4; //Calculate total USB MIDI packet size uint8_t wptr = 0; uint8_t maxpkt = epInfo[epDataInIndex].maxPktSize; if( maxpkt > MIDI_EVENT_PACKET_SIZE ) maxpkt = MIDI_EVENT_PACKET_SIZE; USBTRACE("SendSysEx:\r\t"); USBTRACE2(" Length:\t", datasize); USBTRACE2(" Total pktSize:\t", pktSize); while(n > 0) { //Byte 0 buf[wptr] = (nCable << 4) | 0x4; //x4 SysEx starts or continues switch ( n ) { case 1 : buf[wptr++] = (nCable << 4) | 0x5; //x5 SysEx ends with following single byte. buf[wptr++] = *(dataptr++); buf[wptr++] = 0x00; buf[wptr++] = 0x00; n = n - 1; break; case 2 : buf[wptr++] = (nCable << 4) | 0x6; //x6 SysEx ends with following two bytes. buf[wptr++] = *(dataptr++); buf[wptr++] = *(dataptr++); buf[wptr++] = 0x00; n = n - 2; break; case 3 : buf[wptr] = (nCable << 4) | 0x7; //x7 SysEx ends with following three bytes. // fall through default : wptr++; buf[wptr++] = *(dataptr++); buf[wptr++] = *(dataptr++); buf[wptr++] = *(dataptr++); n = n - 3; break; } if( wptr >= maxpkt || n == 0 ){ //Reach a maxPktSize or data end. USBTRACE2(" wptr:\t", wptr); if( (rc = pUsb->outTransfer(bAddress, epInfo[epDataOutIndex].epAddr, wptr, buf)) != 0 ){ break; } wptr = 0; //rewind data pointer } } return(rc); } /* Send raw data to MIDI device */ uint8_t USBH_MIDI::SendRawData(uint16_t bytes_send, uint8_t *dataptr) { return pUsb->outTransfer(bAddress, epInfo[epDataOutIndex].epAddr, bytes_send, dataptr); } uint8_t USBH_MIDI::extractSysExData(uint8_t *p, uint8_t *buf) { uint8_t rc = 0; uint8_t cin = *(p) & 0x0f; //SysEx message? if( (cin & 0xc) != 4 ) return rc; switch(cin) { case 4: case 7: *buf++ = *(p+1); *buf++ = *(p+2); *buf++ = *(p+3); rc = 3; break; case 6: *buf++ = *(p+1); *buf++ = *(p+2); rc = 2; break; case 5: *buf++ = *(p+1); rc = 1; break; default: break; } return(rc); }