xref: /freebsd/sys/dev/usb/template/usb_template_midi.c (revision 43e29d03f416d7dda52112a29600a7c82ee1a91e)
1 /* $FreeBSD$ */
2 /*-
3  * SPDX-License-Identifier: BSD-2-Clause
4  *
5  * Copyright (c) 2015 Hans Petter Selasky
6  * Copyright (c) 2018 The FreeBSD Foundation
7  * All rights reserved.
8  *
9  * Portions of this software were developed by Edward Tomasz Napierala
10  * under sponsorship from the FreeBSD Foundation.
11  *
12  * Redistribution and use in source and binary forms, with or without
13  * modification, are permitted provided that the following conditions
14  * are met:
15  * 1. Redistributions of source code must retain the above copyright
16  *    notice, this list of conditions and the following disclaimer.
17  * 2. Redistributions in binary form must reproduce the above copyright
18  *    notice, this list of conditions and the following disclaimer in the
19  *    documentation and/or other materials provided with the distribution.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
22  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
25  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31  * SUCH DAMAGE.
32  */
33 
34 /*
35  * This file contains the USB template for an USB MIDI Device.
36  */
37 
38 #ifdef USB_GLOBAL_INCLUDE_FILE
39 #include USB_GLOBAL_INCLUDE_FILE
40 #else
41 #include <sys/stdint.h>
42 #include <sys/stddef.h>
43 #include <sys/param.h>
44 #include <sys/queue.h>
45 #include <sys/types.h>
46 #include <sys/systm.h>
47 #include <sys/kernel.h>
48 #include <sys/bus.h>
49 #include <sys/module.h>
50 #include <sys/lock.h>
51 #include <sys/mutex.h>
52 #include <sys/condvar.h>
53 #include <sys/sysctl.h>
54 #include <sys/sx.h>
55 #include <sys/unistd.h>
56 #include <sys/callout.h>
57 #include <sys/malloc.h>
58 #include <sys/priv.h>
59 
60 #include <dev/usb/usb.h>
61 #include <dev/usb/usbdi.h>
62 #include <dev/usb/usb_core.h>
63 #include <dev/usb/usb_ioctl.h>
64 #include <dev/usb/usb_util.h>
65 
66 #include <dev/usb/template/usb_template.h>
67 #endif					/* USB_GLOBAL_INCLUDE_FILE */
68 
69 enum {
70 	MIDI_LANG_INDEX,
71 	MIDI_INTERFACE_INDEX,
72 	MIDI_MANUFACTURER_INDEX,
73 	MIDI_PRODUCT_INDEX,
74 	MIDI_SERIAL_NUMBER_INDEX,
75 	MIDI_MAX_INDEX,
76 };
77 
78 #define	MIDI_DEFAULT_VENDOR_ID		USB_TEMPLATE_VENDOR
79 #define	MIDI_DEFAULT_PRODUCT_ID		0x27de
80 #define	MIDI_DEFAULT_INTERFACE		"MIDI interface"
81 #define	MIDI_DEFAULT_MANUFACTURER	USB_TEMPLATE_MANUFACTURER
82 #define	MIDI_DEFAULT_PRODUCT		"MIDI Test Device"
83 #define	MIDI_DEFAULT_SERIAL_NUMBER	"March 2008"
84 
85 static struct usb_string_descriptor	midi_interface;
86 static struct usb_string_descriptor	midi_manufacturer;
87 static struct usb_string_descriptor	midi_product;
88 static struct usb_string_descriptor	midi_serial_number;
89 
90 static struct sysctl_ctx_list		midi_ctx_list;
91 
92 /* prototypes */
93 
94 static const uint8_t midi_desc_raw_0[9] = {
95 	0x09, 0x24, 0x01, 0x00, 0x01, 0x09, 0x00, 0x01, 0x01
96 };
97 
98 static const void *midi_descs_0[] = {
99 	&midi_desc_raw_0,
100 	NULL
101 };
102 
103 static const struct usb_temp_interface_desc midi_iface_0 = {
104 	.ppEndpoints = NULL,		/* no endpoints */
105 	.ppRawDesc = midi_descs_0,
106 	.bInterfaceClass = UICLASS_AUDIO,
107 	.bInterfaceSubClass = UISUBCLASS_AUDIOCONTROL,
108 	.bInterfaceProtocol = 0,
109 	.iInterface = MIDI_INTERFACE_INDEX,
110 };
111 
112 static const struct usb_temp_packet_size midi_mps = {
113 	.mps[USB_SPEED_LOW] = 8,
114 	.mps[USB_SPEED_FULL] = 64,
115 	.mps[USB_SPEED_HIGH] = 512,
116 };
117 
118 static const uint8_t midi_desc_raw_7[5] = {
119 	0x05, 0x25, 0x01, 0x01, 0x01
120 };
121 
122 static const void *midi_descs_2[] = {
123 	&midi_desc_raw_7,
124 	NULL
125 };
126 
127 static const struct usb_temp_endpoint_desc midi_bulk_out_ep = {
128 	.ppRawDesc = midi_descs_2,
129 	.pPacketSize = &midi_mps,
130 	.bEndpointAddress = UE_DIR_OUT,
131 	.bmAttributes = UE_BULK,
132 };
133 
134 static const uint8_t midi_desc_raw_6[5] = {
135 	0x05, 0x25, 0x01, 0x01, 0x03,
136 };
137 
138 static const void *midi_descs_3[] = {
139 	&midi_desc_raw_6,
140 	NULL
141 };
142 
143 static const struct usb_temp_endpoint_desc midi_bulk_in_ep = {
144 	.ppRawDesc = midi_descs_3,
145 	.pPacketSize = &midi_mps,
146 	.bEndpointAddress = UE_DIR_IN,
147 	.bmAttributes = UE_BULK,
148 };
149 
150 static const struct usb_temp_endpoint_desc *midi_iface_1_ep[] = {
151 	&midi_bulk_out_ep,
152 	&midi_bulk_in_ep,
153 	NULL,
154 };
155 
156 static const uint8_t midi_desc_raw_1[7] = {
157 	0x07, 0x24, 0x01, 0x00, 0x01, /* wTotalLength: */ 0x41, 0x00
158 };
159 
160 static const uint8_t midi_desc_raw_2[6] = {
161 	0x06, 0x24, 0x02, 0x01, 0x01, 0x00
162 };
163 
164 static const uint8_t midi_desc_raw_3[6] = {
165 	0x06, 0x24, 0x02, 0x02, 0x02, 0x00
166 };
167 
168 static const uint8_t midi_desc_raw_4[9] = {
169 	0x09, 0x24, 0x03, 0x01, 0x03, 0x01, 0x02, 0x01, 0x00
170 };
171 
172 static const uint8_t midi_desc_raw_5[9] = {
173 	0x09, 0x24, 0x03, 0x02, 0x04, 0x01, 0x01, 0x01, 0x00
174 };
175 
176 static const void *midi_descs_1[] = {
177 	&midi_desc_raw_1,
178 	&midi_desc_raw_2,
179 	&midi_desc_raw_3,
180 	&midi_desc_raw_4,
181 	&midi_desc_raw_5,
182 	NULL
183 };
184 
185 static const struct usb_temp_interface_desc midi_iface_1 = {
186 	.ppRawDesc = midi_descs_1,
187 	.ppEndpoints = midi_iface_1_ep,
188 	.bInterfaceClass = UICLASS_AUDIO,
189 	.bInterfaceSubClass = UISUBCLASS_MIDISTREAM,
190 	.bInterfaceProtocol = 0,
191 	.iInterface = MIDI_INTERFACE_INDEX,
192 };
193 
194 static const struct usb_temp_interface_desc *midi_interfaces[] = {
195 	&midi_iface_0,
196 	&midi_iface_1,
197 	NULL,
198 };
199 
200 static const struct usb_temp_config_desc midi_config_desc = {
201 	.ppIfaceDesc = midi_interfaces,
202 	.bmAttributes = 0,
203 	.bMaxPower = 0,
204 	.iConfiguration = MIDI_PRODUCT_INDEX,
205 };
206 
207 static const struct usb_temp_config_desc *midi_configs[] = {
208 	&midi_config_desc,
209 	NULL,
210 };
211 
212 static usb_temp_get_string_desc_t midi_get_string_desc;
213 
214 struct usb_temp_device_desc usb_template_midi = {
215 	.getStringDesc = &midi_get_string_desc,
216 	.ppConfigDesc = midi_configs,
217 	.idVendor = MIDI_DEFAULT_VENDOR_ID,
218 	.idProduct = MIDI_DEFAULT_PRODUCT_ID,
219 	.bcdDevice = 0x0100,
220 	.bDeviceClass = 0,
221 	.bDeviceSubClass = 0,
222 	.bDeviceProtocol = 0,
223 	.iManufacturer = MIDI_MANUFACTURER_INDEX,
224 	.iProduct = MIDI_PRODUCT_INDEX,
225 	.iSerialNumber = MIDI_SERIAL_NUMBER_INDEX,
226 };
227 
228 /*------------------------------------------------------------------------*
229  *	midi_get_string_desc
230  *
231  * Return values:
232  * NULL: Failure. No such string.
233  * Else: Success. Pointer to string descriptor is returned.
234  *------------------------------------------------------------------------*/
235 static const void *
236 midi_get_string_desc(uint16_t lang_id, uint8_t string_index)
237 {
238 	static const void *ptr[MIDI_MAX_INDEX] = {
239 		[MIDI_LANG_INDEX] = &usb_string_lang_en,
240 		[MIDI_INTERFACE_INDEX] = &midi_interface,
241 		[MIDI_MANUFACTURER_INDEX] = &midi_manufacturer,
242 		[MIDI_PRODUCT_INDEX] = &midi_product,
243 		[MIDI_SERIAL_NUMBER_INDEX] = &midi_serial_number,
244 	};
245 
246 	if (string_index == 0) {
247 		return (&usb_string_lang_en);
248 	}
249 	if (lang_id != 0x0409) {
250 		return (NULL);
251 	}
252 	if (string_index < MIDI_MAX_INDEX) {
253 		return (ptr[string_index]);
254 	}
255 	return (NULL);
256 }
257 
258 static void
259 midi_init(void *arg __unused)
260 {
261 	struct sysctl_oid *parent;
262 	char parent_name[3];
263 
264 	usb_make_str_desc(&midi_interface, sizeof(midi_interface),
265 	    MIDI_DEFAULT_INTERFACE);
266 	usb_make_str_desc(&midi_manufacturer, sizeof(midi_manufacturer),
267 	    MIDI_DEFAULT_MANUFACTURER);
268 	usb_make_str_desc(&midi_product, sizeof(midi_product),
269 	    MIDI_DEFAULT_PRODUCT);
270 	usb_make_str_desc(&midi_serial_number, sizeof(midi_serial_number),
271 	    MIDI_DEFAULT_SERIAL_NUMBER);
272 
273 	snprintf(parent_name, sizeof(parent_name), "%d", USB_TEMP_MIDI);
274 	sysctl_ctx_init(&midi_ctx_list);
275 
276 	parent = SYSCTL_ADD_NODE(&midi_ctx_list,
277 	    SYSCTL_STATIC_CHILDREN(_hw_usb_templates), OID_AUTO,
278 	    parent_name, CTLFLAG_RW | CTLFLAG_MPSAFE,
279 	    0, "USB MIDI device side template");
280 	SYSCTL_ADD_U16(&midi_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
281 	    "vendor_id", CTLFLAG_RWTUN,
282 	    &usb_template_midi.idVendor, 1, "Vendor identifier");
283 	SYSCTL_ADD_U16(&midi_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
284 	    "product_id", CTLFLAG_RWTUN,
285 	    &usb_template_midi.idProduct, 1, "Product identifier");
286 #if 0
287 	SYSCTL_ADD_PROC(&midi_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
288 	    "interface", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
289 	    &midi_interface, sizeof(midi_interface), usb_temp_sysctl,
290 	    "A", "Interface string");
291 #endif
292 	SYSCTL_ADD_PROC(&midi_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
293 	    "manufacturer", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
294 	    &midi_manufacturer, sizeof(midi_manufacturer), usb_temp_sysctl,
295 	    "A", "Manufacturer string");
296 	SYSCTL_ADD_PROC(&midi_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
297 	    "product", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
298 	    &midi_product, sizeof(midi_product), usb_temp_sysctl,
299 	    "A", "Product string");
300 	SYSCTL_ADD_PROC(&midi_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
301 	    "serial_number", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
302 	    &midi_serial_number, sizeof(midi_serial_number), usb_temp_sysctl,
303 	    "A", "Serial number string");
304 }
305 
306 static void
307 midi_uninit(void *arg __unused)
308 {
309 
310 	sysctl_ctx_free(&midi_ctx_list);
311 }
312 
313 SYSINIT(midi_init, SI_SUB_LOCK, SI_ORDER_FIRST, midi_init, NULL);
314 SYSUNINIT(midi_uninit, SI_SUB_LOCK, SI_ORDER_FIRST, midi_uninit, NULL);
315