Skip to content

Commit 53f9558

Browse files
committed
usbd: Add midi interface definition from @paulhamsh.
Taken from https://github.com/paulhamsh/Micropython-Midi-Device as of commit 2678d13.
1 parent afc79c2 commit 53f9558

File tree

1 file changed

+308
-0
lines changed

1 file changed

+308
-0
lines changed

micropython/usbd/midi.py

+308
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,308 @@
1+
# MicroPython USB MIDI module
2+
# MIT license; Copyright (c) 2023 Angus Gratton, Paul Hamshere
3+
from micropython import const
4+
import ustruct
5+
6+
from .device import USBInterface
7+
from .utils import endpoint_descriptor, EP_IN_FLAG
8+
9+
_INTERFACE_CLASS_AUDIO = const(0x01)
10+
_INTERFACE_SUBCLASS_AUDIO_CONTROL = const(0x01)
11+
_INTERFACE_SUBCLASS_AUDIO_MIDISTREAMING = const(0x03)
12+
_PROTOCOL_NONE = const(0x00)
13+
14+
_JACK_TYPE_EMBEDDED = const(0x01)
15+
_JACK_TYPE_EXTERNAL = const(0x02)
16+
17+
18+
class RingBuf:
19+
def __init__(self, size):
20+
self.data = bytearray(size)
21+
self.size = size
22+
self.index_put = 0
23+
self.index_get = 0
24+
25+
def put(self, value):
26+
next_index = (self.index_put + 1) % self.size
27+
# check for overflow
28+
if self.index_get != next_index:
29+
self.data[self.index_put] = value
30+
self.index_put = next_index
31+
return value
32+
else:
33+
return None
34+
35+
def get(self):
36+
if self.index_get == self.index_put:
37+
return None # buffer empty
38+
else:
39+
value = self.data[self.index_get]
40+
self.index_get = (self.index_get + 1) % self.size
41+
return value
42+
43+
def is_empty(self):
44+
return self.index_get == self.index_put
45+
46+
47+
class DummyAudioInterface(USBInterface):
48+
# An Audio Class interface is mandatory for MIDI Interfaces as well, this
49+
# class implements the minimum necessary for this.
50+
def __init__(self):
51+
super().__init__(_INTERFACE_CLASS_AUDIO, _INTERFACE_SUBCLASS_AUDIO_CONTROL, _PROTOCOL_NONE)
52+
53+
def get_itf_descriptor(self, num_eps, itf_idx, str_idx):
54+
# Return the MIDI USB interface descriptors.
55+
56+
# Get the parent interface class
57+
desc, strs = super().get_itf_descriptor(num_eps, itf_idx, str_idx)
58+
59+
# Append the class-specific AudioControl interface descriptor
60+
desc += ustruct.pack(
61+
"<BBBHHBB",
62+
9, # bLength
63+
0x24, # bDescriptorType CS_INTERFACE
64+
0x01, # bDescriptorSubtype MS_HEADER
65+
0x0100, # BcdADC
66+
0x0009, # wTotalLength
67+
0x01, # bInCollection,
68+
# baInterfaceNr value assumes the next interface will be MIDIInterface
69+
itf_idx + 1, # baInterfaceNr
70+
)
71+
72+
return (desc, strs)
73+
74+
75+
class MIDIInterface(USBInterface):
76+
# Base class to implement a USB MIDI device in Python.
77+
78+
# To be compliant two USB interfaces should be registered in series, first a
79+
# _DummyAudioInterface() and then this one immediately after.
80+
def __init__(self, num_rx=1, num_tx=1):
81+
# Arguments are number of MIDI IN and OUT connections (default 1 each way).
82+
83+
# 'rx' and 'tx' are from the point of view of this device, i.e. a 'tx'
84+
# connection is device to host. RX and TX are used here to avoid the
85+
# even more confusing "MIDI IN" and "MIDI OUT", which varies depending
86+
# on whether you look from the perspective of the device or the USB
87+
# interface.
88+
super().__init__(
89+
_INTERFACE_CLASS_AUDIO, _INTERFACE_SUBCLASS_AUDIO_MIDISTREAMING, _PROTOCOL_NONE
90+
)
91+
self._num_rx = num_rx
92+
self._num_tx = num_tx
93+
self.ep_out = None # Set during enumeration
94+
self.ep_in = None
95+
self._rx_buf = bytearray(64)
96+
97+
def send_data(self, tx_data):
98+
"""Helper function to send data."""
99+
self.submit_xfer(self.ep_out, tx_data)
100+
101+
def midi_received(self):
102+
return not self.rb.is_empty()
103+
104+
def get_rb(self):
105+
return self.rb.get()
106+
107+
def receive_data_callback(self, ep_addr, result, xferred_bytes):
108+
for i in range(0, xferred_bytes):
109+
self.rb.put(self.rx_data[i])
110+
self.submit_xfer(0x03, self.rx_data, self.receive_data_callback)
111+
112+
def start_receive_data(self):
113+
self.submit_xfer(
114+
self.ep_in, self.rx_data, self.receive_data_callback
115+
)
116+
117+
def get_itf_descriptor(self, num_eps, itf_idx, str_idx):
118+
# Return the MIDI USB interface descriptors.
119+
120+
# Get the parent interface class
121+
desc, strs = super().get_itf_descriptor(num_eps, itf_idx, str_idx)
122+
123+
# Append the class-specific interface descriptors
124+
125+
_JACK_IN_DESC_LEN = const(6)
126+
_JACK_OUT_DESC_LEN = const(9)
127+
128+
# Midi Streaming interface descriptor
129+
cs_ms_interface = ustruct.pack(
130+
"<BBBHH",
131+
7, # bLength
132+
0x24, # bDescriptorType CS_INTERFACE
133+
0x01, # bDescriptorSubtype MS_HEADER
134+
0x0100, # BcdADC
135+
# wTotalLength: this descriptor, plus length of all Jack descriptors
136+
(7 + (2 * (_JACK_IN_DESC_LEN + _JACK_OUT_DESC_LEN) * (self._num_rx + self._num_tx))),
137+
)
138+
139+
def jack_in_desc(bJackType, bJackID):
140+
return ustruct.pack(
141+
"<BBBBBB",
142+
_JACK_IN_DESC_LEN, # bLength
143+
0x24, # bDescriptorType CS_INTERFACE
144+
0x02, # bDescriptorSubtype MIDI_IN_JACK
145+
bJackType,
146+
bJackID,
147+
0x00, # iJack, no string descriptor support yet
148+
)
149+
150+
def jack_out_desc(bJackType, bJackID, bSourceId, bSourcePin):
151+
return ustruct.pack(
152+
"<BBBBBBBBB",
153+
_JACK_OUT_DESC_LEN, # bLength
154+
0x24, # bDescriptorType CS_INTERFACE
155+
0x03, # bDescriptorSubtype MIDI_OUT_JACK
156+
bJackType,
157+
bJackID,
158+
0x01, # bNrInputPins
159+
bSourceId, # baSourceID(1)
160+
bSourcePin, # baSourcePin(1)
161+
0x00, # iJack, no string descriptor support yet
162+
)
163+
164+
jacks = bytearray() # TODO: pre-allocate this whole descriptor and pack into it
165+
166+
# The USB MIDI standard 1.0 allows modelling a baffling range of MIDI
167+
# devices with different permutations of Jack descriptors, with a lot of
168+
# scope for indicating internal connections in the device (as
169+
# "virtualised" by the USB MIDI standard). Much of the options don't
170+
# really change the USB behaviour but provide metadata to the host.
171+
#
172+
# As observed elsewhere online, the standard ends up being pretty
173+
# complex and unclear in parts, but there is a clear simple example in
174+
# an Appendix. So nearly everyone implements the device from the
175+
# Appendix as-is, even when it's not a good fit for their application,
176+
# and ignores the rest of the standard.
177+
#
178+
# We'll try to implement a slightly more flexible subset that's still
179+
# very simple, without getting caught in the weeds:
180+
#
181+
# - For each rx (total _num_rx), we have data flowing from the USB host
182+
# to the USB MIDI device:
183+
# * Data comes from a MIDI OUT Endpoint (Host->Device)
184+
# * Data goes via an Embedded MIDI IN Jack ("into" the USB-MIDI device)
185+
# * Data goes out via a virtual External MIDI OUT Jack ("out" of the
186+
# USB-MIDI device and into the world). This "out" jack may be
187+
# theoretical, and only exists in the USB descriptor.
188+
#
189+
# - For each tx (total _num_tx), we have data flowing from the USB MIDI
190+
# device to the USB host:
191+
# * Data comes in via a virtual External MIDI IN Jack (from the
192+
# outside world, theoretically)
193+
# * Data goes via an Embedded MIDI OUT Jack ("out" of the USB-MIDI
194+
# device).
195+
# * Data goes into the host via MIDI IN Endpoint (Device->Host)
196+
197+
# rx side
198+
for idx in range(self._num_rx):
199+
emb_id = self._emb_id(False, idx)
200+
ext_id = emb_id + 1
201+
pin = idx + 1
202+
jacks += jack_in_desc(_JACK_TYPE_EMBEDDED, emb_id) # bJackID)
203+
jacks += jack_out_desc(
204+
_JACK_TYPE_EXTERNAL,
205+
ext_id, # bJackID
206+
emb_id, # baSourceID(1)
207+
pin, # baSourcePin(1)
208+
)
209+
210+
# tx side
211+
for idx in range(self._num_tx):
212+
emb_id = self._emb_id(True, idx)
213+
ext_id = emb_id + 1
214+
pin = idx + 1
215+
216+
jacks += jack_in_desc(
217+
_JACK_TYPE_EXTERNAL,
218+
ext_id, # bJackID
219+
)
220+
jacks += jack_out_desc(
221+
_JACK_TYPE_EMBEDDED,
222+
emb_id,
223+
ext_id, # baSourceID(1)
224+
pin, # baSourcePin(1)
225+
)
226+
227+
iface = desc + cs_ms_interface + jacks
228+
return (iface, strs)
229+
230+
def _emb_id(self, is_tx, idx):
231+
# Given a direction (False==rx, True==tx) and a 0-index
232+
# of the MIDI connection, return the embedded JackID value.
233+
#
234+
# Embedded JackIDs take odd numbers 1,3,5,etc with all
235+
# 'RX' jack numbers first and then all 'TX' jack numbers
236+
# (see long comment above for explanation of RX, TX in
237+
# this context.)
238+
#
239+
# This is used to keep jack IDs in sync between
240+
# get_itf_descriptor() and get_endpoint_descriptors()
241+
return 1 + 2 * (idx + (is_tx * self._num_rx))
242+
243+
def get_endpoint_descriptors(self, ep_addr, str_idx):
244+
# One MIDI endpoint in each direction, plus the
245+
# associated CS descriptors
246+
247+
# The following implementation is *very* memory inefficient
248+
# and needs optimising
249+
250+
self.ep_out = (ep_addr + 1)
251+
self.ep_in = ep_addr + 2 | EP_IN_FLAG
252+
253+
# rx side, USB "in" endpoint and embedded MIDI IN Jacks
254+
e_out = endpoint_descriptor(self.ep_in, "bulk", 64, 0)
255+
cs_out = ustruct.pack(
256+
"<BBBB" + "B" * self._num_rx,
257+
4 + self._num_rx, # bLength
258+
0x25, # bDescriptorType CS_ENDPOINT
259+
0x01, # bDescriptorSubtype MS_GENERAL
260+
self._num_rx, # bNumEmbMIDIJack
261+
*(self._emb_id(False, idx) for idx in range(self._num_rx)) # baSourcePin(1..._num_rx)
262+
)
263+
264+
# tx side, USB "out" endpoint and embedded MIDI OUT jacks
265+
e_in = endpoint_descriptor(self.ep_out, "bulk", 64, 0)
266+
cs_in = ustruct.pack(
267+
"<BBBB" + "B" * self._num_tx,
268+
4 + self._num_tx, # bLength
269+
0x25, # bDescriptorType CS_ENDPOINT
270+
0x01, # bDescriptorSubtype MS_GENERAL
271+
self._num_tx, # bNumEmbMIDIJack
272+
*(self._emb_id(True, idx) for idx in range(self._num_tx)) # baSourcePin(1..._num_rx)
273+
)
274+
275+
desc = e_out + cs_out + e_in + cs_in
276+
277+
return (desc, [], (self.ep_out, self.ep_in))
278+
279+
280+
class MidiUSB(MIDIInterface):
281+
# Very basic synchronous USB MIDI interface
282+
283+
def __init__(self):
284+
super().__init__()
285+
286+
def note_on(self, channel, pitch, vel):
287+
obuf = ustruct.pack("<BBBB", 0x09, 0x90 | channel, pitch, vel)
288+
super().send_data(obuf)
289+
290+
def note_off(self, channel, pitch, vel):
291+
obuf = ustruct.pack("<BBBB", 0x08, 0x80 | channel, pitch, vel)
292+
super().send_data(obuf)
293+
294+
def start(self):
295+
super().start_receive_data()
296+
297+
def midi_received(self):
298+
return super().midi_received()
299+
300+
def get_midi(self):
301+
if super().midi_received():
302+
cin = super().get_rb()
303+
cmd = super().get_rb()
304+
val1 = super().get_rb()
305+
val2 = super().get_rb()
306+
return (cin, cmd, val1, val2)
307+
else:
308+
return (None, None, None, None)

0 commit comments

Comments
 (0)