diff options
Diffstat (limited to 'tinyusb/src/class/midi')
| -rwxr-xr-x | tinyusb/src/class/midi/midi.h | 212 | ||||
| -rwxr-xr-x | tinyusb/src/class/midi/midi_device.c | 538 | ||||
| -rwxr-xr-x | tinyusb/src/class/midi/midi_device.h | 173 | 
3 files changed, 923 insertions, 0 deletions
| diff --git a/tinyusb/src/class/midi/midi.h b/tinyusb/src/class/midi/midi.h new file mode 100755 index 00000000..74dc4174 --- /dev/null +++ b/tinyusb/src/class/midi/midi.h @@ -0,0 +1,212 @@ +/*  + * The MIT License (MIT) + * + * Copyright (c) 2019 Ha Thach (tinyusb.org) + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + * + * This file is part of the TinyUSB stack. + */ + +/** \ingroup group_class + *  \defgroup ClassDriver_CDC Communication Device Class (CDC) + *            Currently only Abstract Control Model subclass is supported + *  @{ */ + +#ifndef _TUSB_MIDI_H__ +#define _TUSB_MIDI_H__ + +#include "common/tusb_common.h" + +#ifdef __cplusplus + extern "C" { +#endif + +//--------------------------------------------------------------------+ +// Class Specific Descriptor +//--------------------------------------------------------------------+ + +typedef enum +{ +  MIDI_CS_INTERFACE_HEADER    = 0x01, +  MIDI_CS_INTERFACE_IN_JACK   = 0x02, +  MIDI_CS_INTERFACE_OUT_JACK  = 0x03, +  MIDI_CS_INTERFACE_ELEMENT   = 0x04, +} midi_cs_interface_subtype_t; + +typedef enum +{ +  MIDI_CS_ENDPOINT_GENERAL = 0x01 +} midi_cs_endpoint_subtype_t; + +typedef enum +{ +  MIDI_JACK_EMBEDDED = 0x01, +  MIDI_JACK_EXTERNAL = 0x02 +} midi_jack_type_t; + +typedef enum +{ +  MIDI_CIN_MISC              = 0, +  MIDI_CIN_CABLE_EVENT       = 1, +  MIDI_CIN_SYSCOM_2BYTE      = 2, // 2 byte system common message e.g MTC, SongSelect +  MIDI_CIN_SYSCOM_3BYTE      = 3, // 3 byte system common message e.g SPP +  MIDI_CIN_SYSEX_START       = 4, // SysEx starts or continue +  MIDI_CIN_SYSEX_END_1BYTE   = 5, // SysEx ends with 1 data, or 1 byte system common message +  MIDI_CIN_SYSEX_END_2BYTE   = 6, // SysEx ends with 2 data +  MIDI_CIN_SYSEX_END_3BYTE   = 7, // SysEx ends with 3 data +  MIDI_CIN_NOTE_ON           = 8, +  MIDI_CIN_NOTE_OFF          = 9, +  MIDI_CIN_POLY_KEYPRESS     = 10, +  MIDI_CIN_CONTROL_CHANGE    = 11, +  MIDI_CIN_PROGRAM_CHANGE    = 12, +  MIDI_CIN_CHANNEL_PRESSURE  = 13, +  MIDI_CIN_PITCH_BEND_CHANGE = 14, +  MIDI_CIN_1BYTE_DATA = 15 +} midi_code_index_number_t; + +// MIDI 1.0 status byte +enum +{ +  //------------- System Exclusive -------------// +  MIDI_STATUS_SYSEX_START                    = 0xF0, +  MIDI_STATUS_SYSEX_END                      = 0xF7, + +  //------------- System Common -------------// +  MIDI_STATUS_SYSCOM_TIME_CODE_QUARTER_FRAME = 0xF1, +  MIDI_STATUS_SYSCOM_SONG_POSITION_POINTER   = 0xF2, +  MIDI_STATUS_SYSCOM_SONG_SELECT             = 0xF3, +  // F4, F5 is undefined +  MIDI_STATUS_SYSCOM_TUNE_REQUEST            = 0xF6, + +  //------------- System RealTime  -------------// +  MIDI_STATUS_SYSREAL_TIMING_CLOCK           = 0xF8, +  // 0xF9 is undefined +  MIDI_STATUS_SYSREAL_START                  = 0xFA, +  MIDI_STATUS_SYSREAL_CONTINUE               = 0xFB, +  MIDI_STATUS_SYSREAL_STOP                   = 0xFC, +  // 0xFD is undefined +  MIDI_STATUS_SYSREAL_ACTIVE_SENSING         = 0xFE, +  MIDI_STATUS_SYSREAL_SYSTEM_RESET           = 0xFF, +}; + +/// MIDI Interface Header Descriptor +typedef struct TU_ATTR_PACKED +{ +  uint8_t bLength            ; ///< Size of this descriptor in bytes. +  uint8_t bDescriptorType    ; ///< Descriptor Type, must be Class-Specific +  uint8_t bDescriptorSubType ; ///< Descriptor SubType +  uint16_t bcdMSC            ; ///< MidiStreaming SubClass release number in Binary-Coded Decimal +  uint16_t wTotalLength      ; +} midi_desc_header_t; + +/// MIDI In Jack Descriptor +typedef struct TU_ATTR_PACKED +{ +  uint8_t bLength            ; ///< Size of this descriptor in bytes. +  uint8_t bDescriptorType    ; ///< Descriptor Type, must be Class-Specific +  uint8_t bDescriptorSubType ; ///< Descriptor SubType +  uint8_t bJackType          ; ///< Embedded or External +  uint8_t bJackID            ; ///< Unique ID for MIDI IN Jack +  uint8_t iJack              ; ///< string descriptor +} midi_desc_in_jack_t; + + +/// MIDI Out Jack Descriptor with single pin +typedef struct TU_ATTR_PACKED +{ +  uint8_t bLength            ; ///< Size of this descriptor in bytes. +  uint8_t bDescriptorType    ; ///< Descriptor Type, must be Class-Specific +  uint8_t bDescriptorSubType ; ///< Descriptor SubType +  uint8_t bJackType          ; ///< Embedded or External +  uint8_t bJackID            ; ///< Unique ID for MIDI IN Jack +  uint8_t bNrInputPins; + +  uint8_t baSourceID; +  uint8_t baSourcePin; + +  uint8_t iJack              ; ///< string descriptor +} midi_desc_out_jack_t ; + +/// MIDI Out Jack Descriptor with multiple pins +#define midi_desc_out_jack_n_t(input_num) \ +  struct TU_ATTR_PACKED { \ +    uint8_t bLength            ; \ +    uint8_t bDescriptorType    ; \ +    uint8_t bDescriptorSubType ; \ +    uint8_t bJackType          ; \ +    uint8_t bJackID            ; \ +    uint8_t bNrInputPins       ; \ +    struct TU_ATTR_PACKED {      \ +        uint8_t baSourceID;      \ +        uint8_t baSourcePin;     \ +    } pins[input_num];           \ +   uint8_t iJack              ;  \ +  } + +/// MIDI Element Descriptor +typedef struct TU_ATTR_PACKED +{ +  uint8_t bLength            ; ///< Size of this descriptor in bytes. +  uint8_t bDescriptorType    ; ///< Descriptor Type, must be Class-Specific +  uint8_t bDescriptorSubType ; ///< Descriptor SubType +  uint8_t bElementID; + +  uint8_t bNrInputPins; +  uint8_t baSourceID; +  uint8_t baSourcePin; + +  uint8_t bNrOutputPins; +  uint8_t bInTerminalLink; +  uint8_t bOutTerminalLink; +  uint8_t bElCapsSize; + +  uint16_t bmElementCaps; +  uint8_t  iElement; +} midi_desc_element_t; + +/// MIDI Element Descriptor with multiple pins +#define midi_desc_element_n_t(input_num) \ +  struct TU_ATTR_PACKED {       \ +    uint8_t bLength;            \ +    uint8_t bDescriptorType;    \ +    uint8_t bDescriptorSubType; \ +    uint8_t bElementID;         \ +    uint8_t bNrInputPins;       \ +    struct TU_ATTR_PACKED {     \ +        uint8_t baSourceID;     \ +        uint8_t baSourcePin;    \ +    } pins[input_num];          \ +    uint8_t bNrOutputPins;      \ +    uint8_t bInTerminalLink;    \ +    uint8_t bOutTerminalLink;   \ +    uint8_t bElCapsSize;        \ +    uint16_t bmElementCaps;     \ +    uint8_t  iElement;          \ + } + +/** @} */ + +#ifdef __cplusplus + } +#endif + +#endif + +/** @} */ diff --git a/tinyusb/src/class/midi/midi_device.c b/tinyusb/src/class/midi/midi_device.c new file mode 100755 index 00000000..eb85b668 --- /dev/null +++ b/tinyusb/src/class/midi/midi_device.c @@ -0,0 +1,538 @@ +/*  + * The MIT License (MIT) + * + * Copyright (c) 2019 Ha Thach (tinyusb.org) + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + * + * This file is part of the TinyUSB stack. + */ + +#include "tusb_option.h" + +#if (TUSB_OPT_DEVICE_ENABLED && CFG_TUD_MIDI) + +//--------------------------------------------------------------------+ +// INCLUDE +//--------------------------------------------------------------------+ +#include "device/usbd.h" +#include "device/usbd_pvt.h" + +#include "midi_device.h" + +//--------------------------------------------------------------------+ +// MACRO CONSTANT TYPEDEF +//--------------------------------------------------------------------+ + +typedef struct +{ +  uint8_t buffer[4]; +  uint8_t index; +  uint8_t total; +}midid_stream_t; + +typedef struct +{ +  uint8_t itf_num; +  uint8_t ep_in; +  uint8_t ep_out; + +  // For Stream read()/write() API +  // Messages are always 4 bytes long, queue them for reading and writing so the +  // callers can use the Stream interface with single-byte read/write calls. +  midid_stream_t stream_write; +  midid_stream_t stream_read; + +  /*------------- From this point, data is not cleared by bus reset -------------*/ +  // FIFO +  tu_fifo_t rx_ff; +  tu_fifo_t tx_ff; +  uint8_t rx_ff_buf[CFG_TUD_MIDI_RX_BUFSIZE]; +  uint8_t tx_ff_buf[CFG_TUD_MIDI_TX_BUFSIZE]; + +  #if CFG_FIFO_MUTEX +  osal_mutex_def_t rx_ff_mutex; +  osal_mutex_def_t tx_ff_mutex; +  #endif + +  // Endpoint Transfer buffer +  CFG_TUSB_MEM_ALIGN uint8_t epout_buf[CFG_TUD_MIDI_EP_BUFSIZE]; +  CFG_TUSB_MEM_ALIGN uint8_t epin_buf[CFG_TUD_MIDI_EP_BUFSIZE]; + +} midid_interface_t; + +#define ITF_MEM_RESET_SIZE   offsetof(midid_interface_t, rx_ff) + +//--------------------------------------------------------------------+ +// INTERNAL OBJECT & FUNCTION DECLARATION +//--------------------------------------------------------------------+ +CFG_TUSB_MEM_SECTION midid_interface_t _midid_itf[CFG_TUD_MIDI]; + +bool tud_midi_n_mounted (uint8_t itf) +{ +  midid_interface_t* midi = &_midid_itf[itf]; +  return midi->ep_in && midi->ep_out; +} + +static void _prep_out_transaction (midid_interface_t* p_midi) +{ +  uint8_t const rhport = TUD_OPT_RHPORT; +  uint16_t available = tu_fifo_remaining(&p_midi->rx_ff); + +  // Prepare for incoming data but only allow what we can store in the ring buffer. +  // TODO Actually we can still carry out the transfer, keeping count of received bytes +  // and slowly move it to the FIFO when read(). +  // This pre-check reduces endpoint claiming +  TU_VERIFY(available >= sizeof(p_midi->epout_buf), ); + +  // claim endpoint +  TU_VERIFY(usbd_edpt_claim(rhport, p_midi->ep_out), ); + +  // fifo can be changed before endpoint is claimed +  available = tu_fifo_remaining(&p_midi->rx_ff); + +  if ( available >= sizeof(p_midi->epout_buf) )  { +    usbd_edpt_xfer(rhport, p_midi->ep_out, p_midi->epout_buf, sizeof(p_midi->epout_buf)); +  }else +  { +    // Release endpoint since we don't make any transfer +    usbd_edpt_release(rhport, p_midi->ep_out); +  } +} + +//--------------------------------------------------------------------+ +// READ API +//--------------------------------------------------------------------+ +uint32_t tud_midi_n_available(uint8_t itf, uint8_t cable_num) +{ +  (void) cable_num; + +  midid_interface_t* midi = &_midid_itf[itf]; +  midid_stream_t const* stream = &midi->stream_read; + +  // when using with packet API stream total & index are both zero +  return tu_fifo_count(&midi->rx_ff) + (stream->total - stream->index); +} + +uint32_t tud_midi_n_stream_read(uint8_t itf, uint8_t cable_num, void* buffer, uint32_t bufsize) +{ +  (void) cable_num; +  TU_VERIFY(bufsize, 0); + +  uint8_t* buf8 = (uint8_t*) buffer; + +  midid_interface_t* midi = &_midid_itf[itf]; +  midid_stream_t* stream = &midi->stream_read; + +  uint32_t total_read = 0; +  while( bufsize ) +  { +    // Get new packet from fifo, then set packet expected bytes +    if ( stream->total == 0 ) +    { +      // return if there is no more data from fifo +      if ( !tud_midi_n_packet_read(itf, stream->buffer) ) return total_read; + +      uint8_t const code_index = stream->buffer[0] & 0x0f; + +      // MIDI 1.0 Table 4-1: Code Index Number Classifications +      switch(code_index) +      { +        case MIDI_CIN_MISC: +        case MIDI_CIN_CABLE_EVENT: +          // These are reserved and unused, possibly issue somewhere, skip this packet +          return 0; +        break; + +        case MIDI_CIN_SYSEX_END_1BYTE: +        case MIDI_CIN_1BYTE_DATA: +          stream->total = 1; +        break; + +        case MIDI_CIN_SYSCOM_2BYTE     : +        case MIDI_CIN_SYSEX_END_2BYTE  : +        case MIDI_CIN_PROGRAM_CHANGE   : +        case MIDI_CIN_CHANNEL_PRESSURE : +          stream->total = 2; +        break; + +        default: +          stream->total = 3; +        break; +      } +    } + +    // Copy data up to bufsize +    uint32_t const count = tu_min32(stream->total - stream->index, bufsize); + +    // Skip the header (1st byte) in the buffer +    memcpy(buf8, stream->buffer + 1 + stream->index, count); + +    total_read += count; +    stream->index += count; +    buf8 += count; +    bufsize -= count; + +    // complete current event packet, reset stream +    if ( stream->total == stream->index ) +    { +      stream->index = 0; +      stream->total = 0; +    } +  } + +  return total_read; +} + +bool tud_midi_n_packet_read (uint8_t itf, uint8_t packet[4]) +{ +  midid_interface_t* midi = &_midid_itf[itf]; +  TU_VERIFY(midi->ep_out); + +  uint32_t const num_read = tu_fifo_read_n(&midi->rx_ff, packet, 4); +  _prep_out_transaction(midi); +  return (num_read == 4); +} + +//--------------------------------------------------------------------+ +// WRITE API +//--------------------------------------------------------------------+ + +static uint32_t write_flush(midid_interface_t* midi) +{ +  // No data to send +  if ( !tu_fifo_count(&midi->tx_ff) ) return 0; + +  uint8_t const rhport = TUD_OPT_RHPORT; + +  // skip if previous transfer not complete +  TU_VERIFY( usbd_edpt_claim(rhport, midi->ep_in), 0 ); + +  uint16_t count = tu_fifo_read_n(&midi->tx_ff, midi->epin_buf, CFG_TUD_MIDI_EP_BUFSIZE); + +  if (count) +  { +    TU_ASSERT( usbd_edpt_xfer(rhport, midi->ep_in, midi->epin_buf, count), 0 ); +    return count; +  }else +  { +    // Release endpoint since we don't make any transfer +    usbd_edpt_release(rhport, midi->ep_in); +    return 0; +  } +} + +uint32_t tud_midi_n_stream_write(uint8_t itf, uint8_t cable_num, uint8_t const* buffer, uint32_t bufsize) +{ +  midid_interface_t* midi = &_midid_itf[itf]; +  TU_VERIFY(midi->ep_in, 0); + +  midid_stream_t* stream = &midi->stream_write; + +  uint32_t i = 0; +  while ( (i < bufsize) && (tu_fifo_remaining(&midi->tx_ff) >= 4) ) +  { +    uint8_t const data = buffer[i]; +    i++; + +    if ( stream->index == 0 ) +    { +      //------------- New event packet -------------// + +      uint8_t const msg = data >> 4; + +      stream->index = 2; +      stream->buffer[1] = data; + +      // Check to see if we're still in a SysEx transmit. +      if ( stream->buffer[0] == MIDI_CIN_SYSEX_START ) +      { +        if ( data == MIDI_STATUS_SYSEX_END ) +        { +          stream->buffer[0] = MIDI_CIN_SYSEX_END_1BYTE; +          stream->total = 2; +        } +        else +        { +          stream->total = 4; +        } +      } +      else if ( (msg >= 0x8 && msg <= 0xB) || msg == 0xE ) +      { +        // Channel Voice Messages +        stream->buffer[0] = (cable_num << 4) | msg; +        stream->total = 4; +      } +      else if ( msg == 0xf ) +      { +        // System message +        if ( data == MIDI_STATUS_SYSEX_START ) +        { +          stream->buffer[0] = MIDI_CIN_SYSEX_START; +          stream->total = 4; +        } +        else if ( data == MIDI_STATUS_SYSCOM_TIME_CODE_QUARTER_FRAME || data == MIDI_STATUS_SYSCOM_SONG_SELECT ) +        { +          stream->buffer[0] = MIDI_CIN_SYSCOM_2BYTE; +          stream->total = 3; +        } +        else if ( data == MIDI_STATUS_SYSCOM_SONG_POSITION_POINTER ) +        { +          stream->buffer[0] = MIDI_CIN_SYSCOM_3BYTE; +          stream->total = 4; +        } +        else +        { +          stream->buffer[0] = MIDI_CIN_SYSEX_END_1BYTE; +          stream->total = 2; +        } +      } +      else +      { +        // Pack individual bytes if we don't support packing them into words. +        stream->buffer[0] = cable_num << 4 | 0xf; +        stream->buffer[2] = 0; +        stream->buffer[3] = 0; +        stream->index = 2; +        stream->total = 2; +      } +    } +    else +    { +      //------------- On-going (buffering) packet -------------// + +      TU_ASSERT(stream->index < 4, i); +      stream->buffer[stream->index] = data; +      stream->index++; + +      // See if this byte ends a SysEx. +      if ( stream->buffer[0] == MIDI_CIN_SYSEX_START && data == MIDI_STATUS_SYSEX_END ) +      { +        stream->buffer[0] = MIDI_CIN_SYSEX_START + (stream->index - 1); +        stream->total = stream->index; +      } +    } + +    // Send out packet +    if ( stream->index == stream->total ) +    { +      // zeroes unused bytes +      for(uint8_t idx = stream->total; idx < 4; idx++) stream->buffer[idx] = 0; + +      uint16_t const count = tu_fifo_write_n(&midi->tx_ff, stream->buffer, 4); + +      // complete current event packet, reset stream +      stream->index = stream->total = 0; + +      // FIFO overflown, since we already check fifo remaining. It is probably race condition +      TU_ASSERT(count == 4, i); +    } +  } + +  write_flush(midi); + +  return i; +} + +bool tud_midi_n_packet_write (uint8_t itf, uint8_t const packet[4]) +{ +  midid_interface_t* midi = &_midid_itf[itf]; +  TU_VERIFY(midi->ep_in); + +  if (tu_fifo_remaining(&midi->tx_ff) < 4) return false; + +  tu_fifo_write_n(&midi->tx_ff, packet, 4); +  write_flush(midi); + +  return true; +} + +//--------------------------------------------------------------------+ +// USBD Driver API +//--------------------------------------------------------------------+ +void midid_init(void) +{ +  tu_memclr(_midid_itf, sizeof(_midid_itf)); + +  for(uint8_t i=0; i<CFG_TUD_MIDI; i++) +  { +    midid_interface_t* midi = &_midid_itf[i]; + +    // config fifo +    tu_fifo_config(&midi->rx_ff, midi->rx_ff_buf, CFG_TUD_MIDI_RX_BUFSIZE, 1, false); // true, true +    tu_fifo_config(&midi->tx_ff, midi->tx_ff_buf, CFG_TUD_MIDI_TX_BUFSIZE, 1, false); // OBVS. + +    #if CFG_FIFO_MUTEX +    tu_fifo_config_mutex(&midi->rx_ff, NULL, osal_mutex_create(&midi->rx_ff_mutex)); +    tu_fifo_config_mutex(&midi->tx_ff, osal_mutex_create(&midi->tx_ff_mutex), NULL); +    #endif +  } +} + +void midid_reset(uint8_t rhport) +{ +  (void) rhport; + +  for(uint8_t i=0; i<CFG_TUD_MIDI; i++) +  { +    midid_interface_t* midi = &_midid_itf[i]; +    tu_memclr(midi, ITF_MEM_RESET_SIZE); +    tu_fifo_clear(&midi->rx_ff); +    tu_fifo_clear(&midi->tx_ff); +  } +} + +uint16_t midid_open(uint8_t rhport, tusb_desc_interface_t const * desc_itf, uint16_t max_len) +{ +  // 1st Interface is Audio Control v1 +  TU_VERIFY(TUSB_CLASS_AUDIO               == desc_itf->bInterfaceClass    && +            AUDIO_SUBCLASS_CONTROL         == desc_itf->bInterfaceSubClass && +            AUDIO_FUNC_PROTOCOL_CODE_UNDEF == desc_itf->bInterfaceProtocol, 0); + +  uint16_t drv_len = tu_desc_len(desc_itf); +  uint8_t const * p_desc = tu_desc_next(desc_itf); + +  // Skip Class Specific descriptors +  while ( TUSB_DESC_CS_INTERFACE == tu_desc_type(p_desc) && drv_len <= max_len ) +  { +    drv_len += tu_desc_len(p_desc); +    p_desc   = tu_desc_next(p_desc); +  } + +  // 2nd Interface is MIDI Streaming +  TU_VERIFY(TUSB_DESC_INTERFACE == tu_desc_type(p_desc), 0); +  tusb_desc_interface_t const * desc_midi = (tusb_desc_interface_t const *) p_desc; + +  TU_VERIFY(TUSB_CLASS_AUDIO               == desc_midi->bInterfaceClass    && +            AUDIO_SUBCLASS_MIDI_STREAMING  == desc_midi->bInterfaceSubClass && +            AUDIO_FUNC_PROTOCOL_CODE_UNDEF == desc_midi->bInterfaceProtocol, 0); + +  // Find available interface +  midid_interface_t * p_midi = NULL; +  for(uint8_t i=0; i<CFG_TUD_MIDI; i++) +  { +    if ( _midid_itf[i].ep_in == 0 && _midid_itf[i].ep_out == 0 ) +    { +      p_midi = &_midid_itf[i]; +      break; +    } +  } + +  p_midi->itf_num = desc_midi->bInterfaceNumber; +  (void) p_midi->itf_num; + +  // next descriptor +  drv_len += tu_desc_len(p_desc); +  p_desc   = tu_desc_next(p_desc); + +  // Find and open endpoint descriptors +  uint8_t found_endpoints = 0; +  while ( (found_endpoints < desc_midi->bNumEndpoints) && (drv_len <= max_len)  ) +  { +    if ( TUSB_DESC_ENDPOINT == tu_desc_type(p_desc) ) +    { +      TU_ASSERT(usbd_edpt_open(rhport, (tusb_desc_endpoint_t const *) p_desc), 0); +      uint8_t ep_addr = ((tusb_desc_endpoint_t const *) p_desc)->bEndpointAddress; + +      if (tu_edpt_dir(ep_addr) == TUSB_DIR_IN) +      { +        p_midi->ep_in = ep_addr; +      } else { +        p_midi->ep_out = ep_addr; +      } + +      // Class Specific MIDI Stream endpoint descriptor +      drv_len += tu_desc_len(p_desc); +      p_desc   = tu_desc_next(p_desc); + +      found_endpoints += 1; +    } + +    drv_len += tu_desc_len(p_desc); +    p_desc   = tu_desc_next(p_desc); +  } + +  // Prepare for incoming data +  _prep_out_transaction(p_midi); + +  return drv_len; +} + +// Invoked when a control transfer occurred on an interface of this class +// Driver response accordingly to the request and the transfer stage (setup/data/ack) +// return false to stall control endpoint (e.g unsupported request) +bool midid_control_xfer_cb(uint8_t rhport, uint8_t stage, tusb_control_request_t const * request) +{ +  (void) rhport; +  (void) stage; +  (void) request; + +  // driver doesn't support any request yet +  return false; +} + +bool midid_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes) +{ +  (void) result; +  (void) rhport; + +  uint8_t itf; +  midid_interface_t* p_midi; + +  // Identify which interface to use +  for (itf = 0; itf < CFG_TUD_MIDI; itf++) +  { +    p_midi = &_midid_itf[itf]; +    if ( ( ep_addr == p_midi->ep_out ) || ( ep_addr == p_midi->ep_in ) ) break; +  } +  TU_ASSERT(itf < CFG_TUD_MIDI); + +  // receive new data +  if ( ep_addr == p_midi->ep_out ) +  { +    tu_fifo_write_n(&p_midi->rx_ff, p_midi->epout_buf, xferred_bytes); + +    // invoke receive callback if available +    if (tud_midi_rx_cb) tud_midi_rx_cb(itf); + +    // prepare for next +    // TODO for now ep_out is not used by public API therefore there is no race condition, +    // and does not need to claim like ep_in +    _prep_out_transaction(p_midi); +  } +  else if ( ep_addr == p_midi->ep_in ) +  { +    if (0 == write_flush(p_midi)) +    { +      // If there is no data left, a ZLP should be sent if +      // xferred_bytes is multiple of EP size and not zero +      if ( !tu_fifo_count(&p_midi->tx_ff) && xferred_bytes && (0 == (xferred_bytes % CFG_TUD_MIDI_EP_BUFSIZE)) ) +      { +        if ( usbd_edpt_claim(rhport, p_midi->ep_in) ) +        { +          usbd_edpt_xfer(rhport, p_midi->ep_in, NULL, 0); +        } +      } +    } +  } + +  return true; +} + +#endif diff --git a/tinyusb/src/class/midi/midi_device.h b/tinyusb/src/class/midi/midi_device.h new file mode 100755 index 00000000..211edc8d --- /dev/null +++ b/tinyusb/src/class/midi/midi_device.h @@ -0,0 +1,173 @@ +/*  + * The MIT License (MIT) + * + * Copyright (c) 2019 Ha Thach (tinyusb.org) + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + * + * This file is part of the TinyUSB stack. + */ + +#ifndef _TUSB_MIDI_DEVICE_H_ +#define _TUSB_MIDI_DEVICE_H_ + +#include "class/audio/audio.h" +#include "midi.h" + +//--------------------------------------------------------------------+ +// Class Driver Configuration +//--------------------------------------------------------------------+ + +#if !defined(CFG_TUD_MIDI_EP_BUFSIZE) && defined(CFG_TUD_MIDI_EPSIZE) +  #warning CFG_TUD_MIDI_EPSIZE is renamed to CFG_TUD_MIDI_EP_BUFSIZE, please update to use the new name +  #define CFG_TUD_MIDI_EP_BUFSIZE    CFG_TUD_MIDI_EPSIZE +#endif + +#ifndef CFG_TUD_MIDI_EP_BUFSIZE +  #define CFG_TUD_MIDI_EP_BUFSIZE     (TUD_OPT_HIGH_SPEED ? 512 : 64) +#endif + +#ifdef __cplusplus + extern "C" { +#endif + +/** \addtogroup MIDI_Serial Serial + *  @{ + *  \defgroup   MIDI_Serial_Device Device + *  @{ */ + +//--------------------------------------------------------------------+ +// Application API (Multiple Interfaces) +// CFG_TUD_MIDI > 1 +//--------------------------------------------------------------------+ + +// Check if midi interface is mounted +bool     tud_midi_n_mounted      (uint8_t itf); + +// Get the number of bytes available for reading +uint32_t tud_midi_n_available    (uint8_t itf, uint8_t cable_num); + +// Read byte stream              (legacy) +uint32_t tud_midi_n_stream_read  (uint8_t itf, uint8_t cable_num, void* buffer, uint32_t bufsize); + +// Write byte Stream             (legacy) +uint32_t tud_midi_n_stream_write (uint8_t itf, uint8_t cable_num, uint8_t const* buffer, uint32_t bufsize); + +// Read event packet             (4 bytes) +bool     tud_midi_n_packet_read  (uint8_t itf, uint8_t packet[4]); + +// Write event packet            (4 bytes) +bool     tud_midi_n_packet_write (uint8_t itf, uint8_t const packet[4]); + +//--------------------------------------------------------------------+ +// Application API (Single Interface) +//--------------------------------------------------------------------+ +static inline bool     tud_midi_mounted      (void); +static inline uint32_t tud_midi_available    (void); + +static inline uint32_t tud_midi_stream_read  (void* buffer, uint32_t bufsize); +static inline uint32_t tud_midi_stream_write (uint8_t cable_num, uint8_t const* buffer, uint32_t bufsize); + +static inline bool     tud_midi_packet_read  (uint8_t packet[4]); +static inline bool     tud_midi_packet_write (uint8_t const packet[4]); + +//------------- Deprecated API name  -------------// +// TODO remove after 0.10.0 release + +TU_ATTR_DEPRECATED("tud_midi_read() is renamed to tud_midi_stream_read()") +static inline uint32_t tud_midi_read (void* buffer, uint32_t bufsize) +{ +  return tud_midi_stream_read(buffer, bufsize); +} + +TU_ATTR_DEPRECATED("tud_midi_write() is renamed to tud_midi_stream_write()") +static inline uint32_t tud_midi_write(uint8_t cable_num, uint8_t const* buffer, uint32_t bufsize) +{ +  return tud_midi_stream_write(cable_num, buffer, bufsize); +} + + +TU_ATTR_DEPRECATED("tud_midi_send() is renamed to tud_midi_packet_write()") +static inline bool tud_midi_send(uint8_t packet[4]) +{ +  return tud_midi_packet_write(packet); +} + +TU_ATTR_DEPRECATED("tud_midi_receive() is renamed to tud_midi_packet_read()") +static inline bool tud_midi_receive(uint8_t packet[4]) +{ +  return tud_midi_packet_read(packet); +} + +//--------------------------------------------------------------------+ +// Application Callback API (weak is optional) +//--------------------------------------------------------------------+ +TU_ATTR_WEAK void tud_midi_rx_cb(uint8_t itf); + +//--------------------------------------------------------------------+ +// Inline Functions +//--------------------------------------------------------------------+ + +static inline bool tud_midi_mounted (void) +{ +  return tud_midi_n_mounted(0); +} + +static inline uint32_t tud_midi_available (void) +{ +  return tud_midi_n_available(0, 0); +} + +static inline uint32_t tud_midi_stream_read (void* buffer, uint32_t bufsize) +{ +  return tud_midi_n_stream_read(0, 0, buffer, bufsize); +} + +static inline uint32_t tud_midi_stream_write (uint8_t cable_num, uint8_t const* buffer, uint32_t bufsize) +{ +  return tud_midi_n_stream_write(0, cable_num, buffer, bufsize); +} + +static inline bool tud_midi_packet_read (uint8_t packet[4]) +{ +  return tud_midi_n_packet_read(0, packet); +} + +static inline bool tud_midi_packet_write (uint8_t const packet[4]) +{ +  return tud_midi_n_packet_write(0, packet); +} + +//--------------------------------------------------------------------+ +// Internal Class Driver API +//--------------------------------------------------------------------+ +void     midid_init            (void); +void     midid_reset           (uint8_t rhport); +uint16_t midid_open            (uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t max_len); +bool     midid_control_xfer_cb (uint8_t rhport, uint8_t stage, tusb_control_request_t const * request); +bool     midid_xfer_cb         (uint8_t rhport, uint8_t edpt_addr, xfer_result_t result, uint32_t xferred_bytes); + +#ifdef __cplusplus + } +#endif + +#endif /* _TUSB_MIDI_DEVICE_H_ */ + +/** @} */ +/** @} */ | 
