1399e6543SHans Petter Selasky /*-
2*4d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
3718cf2ccSPedro F. Giffuni *
48e06491aSEdward Tomasz Napierala * Copyright (c) 2010 Hans Petter Selasky
58e06491aSEdward Tomasz Napierala * Copyright (c) 2018 The FreeBSD Foundation
68e06491aSEdward Tomasz Napierala * All rights reserved.
78e06491aSEdward Tomasz Napierala *
88e06491aSEdward Tomasz Napierala * Portions of this software were developed by Edward Tomasz Napierala
98e06491aSEdward Tomasz Napierala * under sponsorship from the FreeBSD Foundation.
10399e6543SHans Petter Selasky *
11399e6543SHans Petter Selasky * Redistribution and use in source and binary forms, with or without
12399e6543SHans Petter Selasky * modification, are permitted provided that the following conditions
13399e6543SHans Petter Selasky * are met:
14399e6543SHans Petter Selasky * 1. Redistributions of source code must retain the above copyright
15399e6543SHans Petter Selasky * notice, this list of conditions and the following disclaimer.
16399e6543SHans Petter Selasky * 2. Redistributions in binary form must reproduce the above copyright
17399e6543SHans Petter Selasky * notice, this list of conditions and the following disclaimer in the
18399e6543SHans Petter Selasky * documentation and/or other materials provided with the distribution.
19399e6543SHans Petter Selasky *
20399e6543SHans Petter Selasky * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21399e6543SHans Petter Selasky * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22399e6543SHans Petter Selasky * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23399e6543SHans Petter Selasky * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
24399e6543SHans Petter Selasky * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25399e6543SHans Petter Selasky * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26399e6543SHans Petter Selasky * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27399e6543SHans Petter Selasky * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28399e6543SHans Petter Selasky * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29399e6543SHans Petter Selasky * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30399e6543SHans Petter Selasky * SUCH DAMAGE.
31399e6543SHans Petter Selasky */
32399e6543SHans Petter Selasky
33399e6543SHans Petter Selasky /*
34399e6543SHans Petter Selasky * This file contains the USB template for an USB Audio Device.
35399e6543SHans Petter Selasky */
36399e6543SHans Petter Selasky
37d2b99310SHans Petter Selasky #ifdef USB_GLOBAL_INCLUDE_FILE
38d2b99310SHans Petter Selasky #include USB_GLOBAL_INCLUDE_FILE
39d2b99310SHans Petter Selasky #else
40399e6543SHans Petter Selasky #include <sys/stdint.h>
41399e6543SHans Petter Selasky #include <sys/stddef.h>
42399e6543SHans Petter Selasky #include <sys/param.h>
43399e6543SHans Petter Selasky #include <sys/queue.h>
44399e6543SHans Petter Selasky #include <sys/types.h>
45399e6543SHans Petter Selasky #include <sys/systm.h>
46399e6543SHans Petter Selasky #include <sys/kernel.h>
47399e6543SHans Petter Selasky #include <sys/bus.h>
48399e6543SHans Petter Selasky #include <sys/module.h>
49399e6543SHans Petter Selasky #include <sys/lock.h>
50399e6543SHans Petter Selasky #include <sys/mutex.h>
51399e6543SHans Petter Selasky #include <sys/condvar.h>
52399e6543SHans Petter Selasky #include <sys/sysctl.h>
53399e6543SHans Petter Selasky #include <sys/sx.h>
54399e6543SHans Petter Selasky #include <sys/unistd.h>
55399e6543SHans Petter Selasky #include <sys/callout.h>
56399e6543SHans Petter Selasky #include <sys/malloc.h>
57399e6543SHans Petter Selasky #include <sys/priv.h>
58399e6543SHans Petter Selasky
59399e6543SHans Petter Selasky #include <dev/usb/usb.h>
60399e6543SHans Petter Selasky #include <dev/usb/usbdi.h>
6123ab0871SHans Petter Selasky #include <dev/usb/usb_core.h>
62399e6543SHans Petter Selasky #include <dev/usb/usb_cdc.h>
638e06491aSEdward Tomasz Napierala #include <dev/usb/usb_ioctl.h>
648e06491aSEdward Tomasz Napierala #include <dev/usb/usb_util.h>
65399e6543SHans Petter Selasky
66399e6543SHans Petter Selasky #include <dev/usb/template/usb_template.h>
67d2b99310SHans Petter Selasky #endif /* USB_GLOBAL_INCLUDE_FILE */
68399e6543SHans Petter Selasky
69399e6543SHans Petter Selasky enum {
708e06491aSEdward Tomasz Napierala AUDIO_LANG_INDEX,
718e06491aSEdward Tomasz Napierala AUDIO_MIXER_INDEX,
728e06491aSEdward Tomasz Napierala AUDIO_RECORD_INDEX,
738e06491aSEdward Tomasz Napierala AUDIO_PLAYBACK_INDEX,
741ee5bed7SEdward Tomasz Napierala AUDIO_MANUFACTURER_INDEX,
758e06491aSEdward Tomasz Napierala AUDIO_PRODUCT_INDEX,
761ee5bed7SEdward Tomasz Napierala AUDIO_SERIAL_NUMBER_INDEX,
778e06491aSEdward Tomasz Napierala AUDIO_MAX_INDEX,
78399e6543SHans Petter Selasky };
79399e6543SHans Petter Selasky
801558eec6SEdward Tomasz Napierala #define AUDIO_DEFAULT_VENDOR_ID USB_TEMPLATE_VENDOR
81defd918dSEdward Tomasz Napierala #define AUDIO_DEFAULT_PRODUCT_ID 0x27e0
828e06491aSEdward Tomasz Napierala #define AUDIO_DEFAULT_MIXER "Mixer interface"
838e06491aSEdward Tomasz Napierala #define AUDIO_DEFAULT_RECORD "Record interface"
848e06491aSEdward Tomasz Napierala #define AUDIO_DEFAULT_PLAYBACK "Playback interface"
85d01c1c8bSEdward Tomasz Napierala #define AUDIO_DEFAULT_MANUFACTURER USB_TEMPLATE_MANUFACTURER
861ee5bed7SEdward Tomasz Napierala #define AUDIO_DEFAULT_PRODUCT "Audio Test Device"
871ee5bed7SEdward Tomasz Napierala #define AUDIO_DEFAULT_SERIAL_NUMBER "March 2008"
88399e6543SHans Petter Selasky
898e06491aSEdward Tomasz Napierala static struct usb_string_descriptor audio_mixer;
908e06491aSEdward Tomasz Napierala static struct usb_string_descriptor audio_record;
918e06491aSEdward Tomasz Napierala static struct usb_string_descriptor audio_playback;
921ee5bed7SEdward Tomasz Napierala static struct usb_string_descriptor audio_manufacturer;
938e06491aSEdward Tomasz Napierala static struct usb_string_descriptor audio_product;
941ee5bed7SEdward Tomasz Napierala static struct usb_string_descriptor audio_serial_number;
95399e6543SHans Petter Selasky
968e06491aSEdward Tomasz Napierala static struct sysctl_ctx_list audio_ctx_list;
97399e6543SHans Petter Selasky
98399e6543SHans Petter Selasky /* prototypes */
99399e6543SHans Petter Selasky
100df142785SHans Petter Selasky /*
101df142785SHans Petter Selasky * Audio Mixer description structures
102df142785SHans Petter Selasky *
103df142785SHans Petter Selasky * Some of the audio descriptors were dumped
104df142785SHans Petter Selasky * from a Creative Labs USB audio device.
105df142785SHans Petter Selasky */
106399e6543SHans Petter Selasky
107399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_0[] = {
108399e6543SHans Petter Selasky 0x0a, 0x24, 0x01, 0x00, 0x01, 0xa9, 0x00, 0x02,
109399e6543SHans Petter Selasky 0x01, 0x02
110399e6543SHans Petter Selasky };
111399e6543SHans Petter Selasky
112399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_1[] = {
113399e6543SHans Petter Selasky 0x0c, 0x24, 0x02, 0x01, 0x01, 0x01, 0x00, 0x02,
114399e6543SHans Petter Selasky 0x03, 0x00, 0x00, 0x00
115399e6543SHans Petter Selasky };
116399e6543SHans Petter Selasky
117399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_2[] = {
118399e6543SHans Petter Selasky 0x0c, 0x24, 0x02, 0x02, 0x01, 0x02, 0x00, 0x02,
119399e6543SHans Petter Selasky 0x03, 0x00, 0x00, 0x00
120399e6543SHans Petter Selasky };
121399e6543SHans Petter Selasky
122399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_3[] = {
123399e6543SHans Petter Selasky 0x0c, 0x24, 0x02, 0x03, 0x03, 0x06, 0x00, 0x02,
124399e6543SHans Petter Selasky 0x03, 0x00, 0x00, 0x00
125399e6543SHans Petter Selasky };
126399e6543SHans Petter Selasky
127399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_4[] = {
128399e6543SHans Petter Selasky 0x0c, 0x24, 0x02, 0x04, 0x05, 0x06, 0x00, 0x02,
129399e6543SHans Petter Selasky 0x03, 0x00, 0x00, 0x00
130399e6543SHans Petter Selasky };
131399e6543SHans Petter Selasky
132399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_5[] = {
133399e6543SHans Petter Selasky 0x09, 0x24, 0x03, 0x05, 0x05, 0x06, 0x00, 0x01,
134399e6543SHans Petter Selasky 0x00
135399e6543SHans Petter Selasky };
136399e6543SHans Petter Selasky
137399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_6[] = {
138399e6543SHans Petter Selasky 0x09, 0x24, 0x03, 0x06, 0x01, 0x03, 0x00, 0x09,
139399e6543SHans Petter Selasky 0x00
140399e6543SHans Petter Selasky };
141399e6543SHans Petter Selasky
142399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_7[] = {
143399e6543SHans Petter Selasky 0x09, 0x24, 0x03, 0x07, 0x01, 0x01, 0x00, 0x08,
144399e6543SHans Petter Selasky 0x00
145399e6543SHans Petter Selasky };
146399e6543SHans Petter Selasky
147399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_8[] = {
148399e6543SHans Petter Selasky 0x09, 0x24, 0x05, 0x08, 0x03, 0x0a, 0x0b, 0x0c,
149399e6543SHans Petter Selasky 0x00
150399e6543SHans Petter Selasky };
151399e6543SHans Petter Selasky
152399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_9[] = {
153399e6543SHans Petter Selasky 0x0a, 0x24, 0x06, 0x09, 0x0f, 0x01, 0x01, 0x02,
154399e6543SHans Petter Selasky 0x02, 0x00
155399e6543SHans Petter Selasky };
156399e6543SHans Petter Selasky
157399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_10[] = {
158399e6543SHans Petter Selasky 0x0a, 0x24, 0x06, 0x0a, 0x02, 0x01, 0x43, 0x00,
159399e6543SHans Petter Selasky 0x00, 0x00
160399e6543SHans Petter Selasky };
161399e6543SHans Petter Selasky
162399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_11[] = {
163399e6543SHans Petter Selasky 0x0a, 0x24, 0x06, 0x0b, 0x03, 0x01, 0x01, 0x02,
164399e6543SHans Petter Selasky 0x02, 0x00
165399e6543SHans Petter Selasky };
166399e6543SHans Petter Selasky
167399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_12[] = {
168399e6543SHans Petter Selasky 0x0a, 0x24, 0x06, 0x0c, 0x04, 0x01, 0x01, 0x00,
169399e6543SHans Petter Selasky 0x00, 0x00
170399e6543SHans Petter Selasky };
171399e6543SHans Petter Selasky
172399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_13[] = {
173399e6543SHans Petter Selasky 0x0a, 0x24, 0x06, 0x0d, 0x02, 0x01, 0x03, 0x00,
174399e6543SHans Petter Selasky 0x00, 0x00
175399e6543SHans Petter Selasky };
176399e6543SHans Petter Selasky
177399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_14[] = {
178399e6543SHans Petter Selasky 0x0a, 0x24, 0x06, 0x0e, 0x03, 0x01, 0x01, 0x02,
179399e6543SHans Petter Selasky 0x02, 0x00
180399e6543SHans Petter Selasky };
181399e6543SHans Petter Selasky
182399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_15[] = {
183399e6543SHans Petter Selasky 0x0f, 0x24, 0x04, 0x0f, 0x03, 0x01, 0x0d, 0x0e,
184399e6543SHans Petter Selasky 0x02, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00
185399e6543SHans Petter Selasky };
186399e6543SHans Petter Selasky
187399e6543SHans Petter Selasky static const void *audio_raw_iface_0_desc[] = {
188399e6543SHans Petter Selasky audio_raw_desc_0,
189399e6543SHans Petter Selasky audio_raw_desc_1,
190399e6543SHans Petter Selasky audio_raw_desc_2,
191399e6543SHans Petter Selasky audio_raw_desc_3,
192399e6543SHans Petter Selasky audio_raw_desc_4,
193399e6543SHans Petter Selasky audio_raw_desc_5,
194399e6543SHans Petter Selasky audio_raw_desc_6,
195399e6543SHans Petter Selasky audio_raw_desc_7,
196399e6543SHans Petter Selasky audio_raw_desc_8,
197399e6543SHans Petter Selasky audio_raw_desc_9,
198399e6543SHans Petter Selasky audio_raw_desc_10,
199399e6543SHans Petter Selasky audio_raw_desc_11,
200399e6543SHans Petter Selasky audio_raw_desc_12,
201399e6543SHans Petter Selasky audio_raw_desc_13,
202399e6543SHans Petter Selasky audio_raw_desc_14,
203399e6543SHans Petter Selasky audio_raw_desc_15,
204399e6543SHans Petter Selasky NULL,
205399e6543SHans Petter Selasky };
206399e6543SHans Petter Selasky
207399e6543SHans Petter Selasky static const struct usb_temp_interface_desc audio_iface_0 = {
208399e6543SHans Petter Selasky .ppEndpoints = NULL, /* no endpoints */
209399e6543SHans Petter Selasky .ppRawDesc = audio_raw_iface_0_desc,
2104ffeccf1SEdward Tomasz Napierala .bInterfaceClass = UICLASS_AUDIO,
2114ffeccf1SEdward Tomasz Napierala .bInterfaceSubClass = UISUBCLASS_AUDIOCONTROL,
212399e6543SHans Petter Selasky .bInterfaceProtocol = 0,
2138e06491aSEdward Tomasz Napierala .iInterface = AUDIO_MIXER_INDEX,
214399e6543SHans Petter Selasky };
215399e6543SHans Petter Selasky
216399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_20[] = {
217399e6543SHans Petter Selasky 0x07, 0x24, 0x01, 0x01, 0x03, 0x01, 0x00
218399e6543SHans Petter Selasky
219399e6543SHans Petter Selasky };
220399e6543SHans Petter Selasky
221399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_21[] = {
222399e6543SHans Petter Selasky 0x0b, 0x24, 0x02, 0x01, 0x02, 0x02, 0x10, 0x01,
223399e6543SHans Petter Selasky /* 48kHz */
224399e6543SHans Petter Selasky 0x80, 0xbb, 0x00
225399e6543SHans Petter Selasky };
226399e6543SHans Petter Selasky
227399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_22[] = {
228399e6543SHans Petter Selasky 0x07, 0x25, 0x01, 0x00, 0x01, 0x04, 0x00
229399e6543SHans Petter Selasky };
230399e6543SHans Petter Selasky
231399e6543SHans Petter Selasky static const void *audio_raw_iface_1_desc[] = {
232399e6543SHans Petter Selasky audio_raw_desc_20,
233399e6543SHans Petter Selasky audio_raw_desc_21,
234399e6543SHans Petter Selasky NULL,
235399e6543SHans Petter Selasky };
236399e6543SHans Petter Selasky
237399e6543SHans Petter Selasky static const void *audio_raw_ep_1_desc[] = {
238399e6543SHans Petter Selasky audio_raw_desc_22,
239399e6543SHans Petter Selasky NULL,
240399e6543SHans Petter Selasky };
241399e6543SHans Petter Selasky
242399e6543SHans Petter Selasky static const struct usb_temp_packet_size audio_isoc_mps = {
243399e6543SHans Petter Selasky .mps[USB_SPEED_FULL] = 0xC8,
244399e6543SHans Petter Selasky .mps[USB_SPEED_HIGH] = 0xC8,
245399e6543SHans Petter Selasky };
246399e6543SHans Petter Selasky
247399e6543SHans Petter Selasky static const struct usb_temp_interval audio_isoc_interval = {
248399e6543SHans Petter Selasky .bInterval[USB_SPEED_FULL] = 1, /* 1:1 */
249399e6543SHans Petter Selasky .bInterval[USB_SPEED_HIGH] = 4, /* 1:8 */
250399e6543SHans Petter Selasky };
251399e6543SHans Petter Selasky
252399e6543SHans Petter Selasky static const struct usb_temp_endpoint_desc audio_isoc_out_ep = {
253399e6543SHans Petter Selasky .ppRawDesc = audio_raw_ep_1_desc,
254399e6543SHans Petter Selasky .pPacketSize = &audio_isoc_mps,
255399e6543SHans Petter Selasky .pIntervals = &audio_isoc_interval,
256399e6543SHans Petter Selasky .bEndpointAddress = UE_DIR_OUT,
257399e6543SHans Petter Selasky .bmAttributes = UE_ISOCHRONOUS | UE_ISO_ADAPT,
258399e6543SHans Petter Selasky };
259399e6543SHans Petter Selasky
260399e6543SHans Petter Selasky static const struct usb_temp_endpoint_desc *audio_iface_1_ep[] = {
261399e6543SHans Petter Selasky &audio_isoc_out_ep,
262399e6543SHans Petter Selasky NULL,
263399e6543SHans Petter Selasky };
264399e6543SHans Petter Selasky
265399e6543SHans Petter Selasky static const struct usb_temp_interface_desc audio_iface_1_alt_0 = {
266399e6543SHans Petter Selasky .ppEndpoints = NULL, /* no endpoints */
267399e6543SHans Petter Selasky .ppRawDesc = NULL, /* no raw descriptors */
2684ffeccf1SEdward Tomasz Napierala .bInterfaceClass = UICLASS_AUDIO,
2694ffeccf1SEdward Tomasz Napierala .bInterfaceSubClass = UISUBCLASS_AUDIOSTREAM,
270399e6543SHans Petter Selasky .bInterfaceProtocol = 0,
2718e06491aSEdward Tomasz Napierala .iInterface = AUDIO_PLAYBACK_INDEX,
272399e6543SHans Petter Selasky };
273399e6543SHans Petter Selasky
274399e6543SHans Petter Selasky static const struct usb_temp_interface_desc audio_iface_1_alt_1 = {
275399e6543SHans Petter Selasky .ppEndpoints = audio_iface_1_ep,
276399e6543SHans Petter Selasky .ppRawDesc = audio_raw_iface_1_desc,
2774ffeccf1SEdward Tomasz Napierala .bInterfaceClass = UICLASS_AUDIO,
2784ffeccf1SEdward Tomasz Napierala .bInterfaceSubClass = UISUBCLASS_AUDIOSTREAM,
279399e6543SHans Petter Selasky .bInterfaceProtocol = 0,
2808e06491aSEdward Tomasz Napierala .iInterface = AUDIO_PLAYBACK_INDEX,
281399e6543SHans Petter Selasky .isAltInterface = 1, /* this is an alternate setting */
282399e6543SHans Petter Selasky };
283399e6543SHans Petter Selasky
284399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_30[] = {
285399e6543SHans Petter Selasky 0x07, 0x24, 0x01, 0x07, 0x01, 0x01, 0x00
286399e6543SHans Petter Selasky
287399e6543SHans Petter Selasky };
288399e6543SHans Petter Selasky
289399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_31[] = {
290399e6543SHans Petter Selasky 0x0b, 0x24, 0x02, 0x01, 0x02, 0x02, 0x10, 0x01,
291399e6543SHans Petter Selasky /* 48kHz */
292399e6543SHans Petter Selasky 0x80, 0xbb, 0x00
293399e6543SHans Petter Selasky };
294399e6543SHans Petter Selasky
295399e6543SHans Petter Selasky static const uint8_t audio_raw_desc_32[] = {
296399e6543SHans Petter Selasky 0x07, 0x25, 0x01, 0x01, 0x00, 0x00, 0x00
297399e6543SHans Petter Selasky };
298399e6543SHans Petter Selasky
299399e6543SHans Petter Selasky static const void *audio_raw_iface_2_desc[] = {
300399e6543SHans Petter Selasky audio_raw_desc_30,
301399e6543SHans Petter Selasky audio_raw_desc_31,
302399e6543SHans Petter Selasky NULL,
303399e6543SHans Petter Selasky };
304399e6543SHans Petter Selasky
305399e6543SHans Petter Selasky static const void *audio_raw_ep_2_desc[] = {
306399e6543SHans Petter Selasky audio_raw_desc_32,
307399e6543SHans Petter Selasky NULL,
308399e6543SHans Petter Selasky };
309399e6543SHans Petter Selasky
310399e6543SHans Petter Selasky static const struct usb_temp_endpoint_desc audio_isoc_in_ep = {
311399e6543SHans Petter Selasky .ppRawDesc = audio_raw_ep_2_desc,
312399e6543SHans Petter Selasky .pPacketSize = &audio_isoc_mps,
313399e6543SHans Petter Selasky .pIntervals = &audio_isoc_interval,
314399e6543SHans Petter Selasky .bEndpointAddress = UE_DIR_IN,
315399e6543SHans Petter Selasky .bmAttributes = UE_ISOCHRONOUS | UE_ISO_ADAPT,
316399e6543SHans Petter Selasky };
317399e6543SHans Petter Selasky
318399e6543SHans Petter Selasky static const struct usb_temp_endpoint_desc *audio_iface_2_ep[] = {
319399e6543SHans Petter Selasky &audio_isoc_in_ep,
320399e6543SHans Petter Selasky NULL,
321399e6543SHans Petter Selasky };
322399e6543SHans Petter Selasky
323399e6543SHans Petter Selasky static const struct usb_temp_interface_desc audio_iface_2_alt_0 = {
324399e6543SHans Petter Selasky .ppEndpoints = NULL, /* no endpoints */
325399e6543SHans Petter Selasky .ppRawDesc = NULL, /* no raw descriptors */
3264ffeccf1SEdward Tomasz Napierala .bInterfaceClass = UICLASS_AUDIO,
3274ffeccf1SEdward Tomasz Napierala .bInterfaceSubClass = UISUBCLASS_AUDIOSTREAM,
328399e6543SHans Petter Selasky .bInterfaceProtocol = 0,
3298e06491aSEdward Tomasz Napierala .iInterface = AUDIO_RECORD_INDEX,
330399e6543SHans Petter Selasky };
331399e6543SHans Petter Selasky
332399e6543SHans Petter Selasky static const struct usb_temp_interface_desc audio_iface_2_alt_1 = {
333399e6543SHans Petter Selasky .ppEndpoints = audio_iface_2_ep,
334399e6543SHans Petter Selasky .ppRawDesc = audio_raw_iface_2_desc,
3354ffeccf1SEdward Tomasz Napierala .bInterfaceClass = UICLASS_AUDIO,
3364ffeccf1SEdward Tomasz Napierala .bInterfaceSubClass = UISUBCLASS_AUDIOSTREAM,
337399e6543SHans Petter Selasky .bInterfaceProtocol = 0,
3388e06491aSEdward Tomasz Napierala .iInterface = AUDIO_RECORD_INDEX,
339399e6543SHans Petter Selasky .isAltInterface = 1, /* this is an alternate setting */
340399e6543SHans Petter Selasky };
341399e6543SHans Petter Selasky
342399e6543SHans Petter Selasky static const struct usb_temp_interface_desc *audio_interfaces[] = {
343399e6543SHans Petter Selasky &audio_iface_0,
344399e6543SHans Petter Selasky &audio_iface_1_alt_0,
345399e6543SHans Petter Selasky &audio_iface_1_alt_1,
346399e6543SHans Petter Selasky &audio_iface_2_alt_0,
347399e6543SHans Petter Selasky &audio_iface_2_alt_1,
348399e6543SHans Petter Selasky NULL,
349399e6543SHans Petter Selasky };
350399e6543SHans Petter Selasky
351399e6543SHans Petter Selasky static const struct usb_temp_config_desc audio_config_desc = {
352399e6543SHans Petter Selasky .ppIfaceDesc = audio_interfaces,
353d008c0d7SEdward Tomasz Napierala .bmAttributes = 0,
354d008c0d7SEdward Tomasz Napierala .bMaxPower = 0,
3558e06491aSEdward Tomasz Napierala .iConfiguration = AUDIO_PRODUCT_INDEX,
356399e6543SHans Petter Selasky };
357399e6543SHans Petter Selasky
358399e6543SHans Petter Selasky static const struct usb_temp_config_desc *audio_configs[] = {
359399e6543SHans Petter Selasky &audio_config_desc,
360399e6543SHans Petter Selasky NULL,
361399e6543SHans Petter Selasky };
362399e6543SHans Petter Selasky
363399e6543SHans Petter Selasky static usb_temp_get_string_desc_t audio_get_string_desc;
364399e6543SHans Petter Selasky
3658e06491aSEdward Tomasz Napierala struct usb_temp_device_desc usb_template_audio = {
366399e6543SHans Petter Selasky .getStringDesc = &audio_get_string_desc,
367399e6543SHans Petter Selasky .ppConfigDesc = audio_configs,
3681558eec6SEdward Tomasz Napierala .idVendor = AUDIO_DEFAULT_VENDOR_ID,
3691558eec6SEdward Tomasz Napierala .idProduct = AUDIO_DEFAULT_PRODUCT_ID,
370399e6543SHans Petter Selasky .bcdDevice = 0x0100,
371399e6543SHans Petter Selasky .bDeviceClass = UDCLASS_COMM,
372399e6543SHans Petter Selasky .bDeviceSubClass = 0,
373399e6543SHans Petter Selasky .bDeviceProtocol = 0,
3741ee5bed7SEdward Tomasz Napierala .iManufacturer = AUDIO_MANUFACTURER_INDEX,
3758e06491aSEdward Tomasz Napierala .iProduct = AUDIO_PRODUCT_INDEX,
3761ee5bed7SEdward Tomasz Napierala .iSerialNumber = AUDIO_SERIAL_NUMBER_INDEX,
377399e6543SHans Petter Selasky };
378399e6543SHans Petter Selasky
379399e6543SHans Petter Selasky /*------------------------------------------------------------------------*
380399e6543SHans Petter Selasky * audio_get_string_desc
381399e6543SHans Petter Selasky *
382399e6543SHans Petter Selasky * Return values:
383399e6543SHans Petter Selasky * NULL: Failure. No such string.
384399e6543SHans Petter Selasky * Else: Success. Pointer to string descriptor is returned.
385399e6543SHans Petter Selasky *------------------------------------------------------------------------*/
386399e6543SHans Petter Selasky static const void *
audio_get_string_desc(uint16_t lang_id,uint8_t string_index)387399e6543SHans Petter Selasky audio_get_string_desc(uint16_t lang_id, uint8_t string_index)
388399e6543SHans Petter Selasky {
3898e06491aSEdward Tomasz Napierala static const void *ptr[AUDIO_MAX_INDEX] = {
3908e06491aSEdward Tomasz Napierala [AUDIO_LANG_INDEX] = &usb_string_lang_en,
3918e06491aSEdward Tomasz Napierala [AUDIO_MIXER_INDEX] = &audio_mixer,
3928e06491aSEdward Tomasz Napierala [AUDIO_RECORD_INDEX] = &audio_record,
3938e06491aSEdward Tomasz Napierala [AUDIO_PLAYBACK_INDEX] = &audio_playback,
3941ee5bed7SEdward Tomasz Napierala [AUDIO_MANUFACTURER_INDEX] = &audio_manufacturer,
3958e06491aSEdward Tomasz Napierala [AUDIO_PRODUCT_INDEX] = &audio_product,
3961ee5bed7SEdward Tomasz Napierala [AUDIO_SERIAL_NUMBER_INDEX] = &audio_serial_number,
397399e6543SHans Petter Selasky };
398399e6543SHans Petter Selasky
399399e6543SHans Petter Selasky if (string_index == 0) {
40023ab0871SHans Petter Selasky return (&usb_string_lang_en);
401399e6543SHans Petter Selasky }
402399e6543SHans Petter Selasky if (lang_id != 0x0409) {
403399e6543SHans Petter Selasky return (NULL);
404399e6543SHans Petter Selasky }
4058e06491aSEdward Tomasz Napierala if (string_index < AUDIO_MAX_INDEX) {
406399e6543SHans Petter Selasky return (ptr[string_index]);
407399e6543SHans Petter Selasky }
408399e6543SHans Petter Selasky return (NULL);
409399e6543SHans Petter Selasky }
4108e06491aSEdward Tomasz Napierala
4118e06491aSEdward Tomasz Napierala static void
audio_init(void * arg __unused)4128e06491aSEdward Tomasz Napierala audio_init(void *arg __unused)
4138e06491aSEdward Tomasz Napierala {
4148e06491aSEdward Tomasz Napierala struct sysctl_oid *parent;
4158e06491aSEdward Tomasz Napierala char parent_name[3];
4168e06491aSEdward Tomasz Napierala
4178e06491aSEdward Tomasz Napierala usb_make_str_desc(&audio_mixer, sizeof(audio_mixer),
4188e06491aSEdward Tomasz Napierala AUDIO_DEFAULT_MIXER);
4198e06491aSEdward Tomasz Napierala usb_make_str_desc(&audio_record, sizeof(audio_record),
4208e06491aSEdward Tomasz Napierala AUDIO_DEFAULT_RECORD);
4218e06491aSEdward Tomasz Napierala usb_make_str_desc(&audio_playback, sizeof(audio_playback),
4228e06491aSEdward Tomasz Napierala AUDIO_DEFAULT_PLAYBACK);
4231ee5bed7SEdward Tomasz Napierala usb_make_str_desc(&audio_manufacturer, sizeof(audio_manufacturer),
4241ee5bed7SEdward Tomasz Napierala AUDIO_DEFAULT_MANUFACTURER);
4258e06491aSEdward Tomasz Napierala usb_make_str_desc(&audio_product, sizeof(audio_product),
4268e06491aSEdward Tomasz Napierala AUDIO_DEFAULT_PRODUCT);
4271ee5bed7SEdward Tomasz Napierala usb_make_str_desc(&audio_serial_number, sizeof(audio_serial_number),
4281ee5bed7SEdward Tomasz Napierala AUDIO_DEFAULT_SERIAL_NUMBER);
4298e06491aSEdward Tomasz Napierala
4308e06491aSEdward Tomasz Napierala snprintf(parent_name, sizeof(parent_name), "%d", USB_TEMP_AUDIO);
4318e06491aSEdward Tomasz Napierala sysctl_ctx_init(&audio_ctx_list);
4328e06491aSEdward Tomasz Napierala
4338e06491aSEdward Tomasz Napierala parent = SYSCTL_ADD_NODE(&audio_ctx_list,
4348e06491aSEdward Tomasz Napierala SYSCTL_STATIC_CHILDREN(_hw_usb_templates), OID_AUTO,
435f8d2b1f3SPawel Biernacki parent_name, CTLFLAG_RW | CTLFLAG_MPSAFE,
4368e06491aSEdward Tomasz Napierala 0, "USB Audio Interface device side template");
4378e06491aSEdward Tomasz Napierala SYSCTL_ADD_U16(&audio_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
4388e06491aSEdward Tomasz Napierala "vendor_id", CTLFLAG_RWTUN, &usb_template_audio.idVendor,
4398e06491aSEdward Tomasz Napierala 1, "Vendor identifier");
4408e06491aSEdward Tomasz Napierala SYSCTL_ADD_U16(&audio_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
4418e06491aSEdward Tomasz Napierala "product_id", CTLFLAG_RWTUN, &usb_template_audio.idProduct,
4428e06491aSEdward Tomasz Napierala 1, "Product identifier");
4438e06491aSEdward Tomasz Napierala #if 0
4448e06491aSEdward Tomasz Napierala SYSCTL_ADD_PROC(&audio_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
4458e06491aSEdward Tomasz Napierala "mixer", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
4468e06491aSEdward Tomasz Napierala &audio_mixer, sizeof(audio_mixer), usb_temp_sysctl,
4478e06491aSEdward Tomasz Napierala "A", "Mixer interface string");
4488e06491aSEdward Tomasz Napierala SYSCTL_ADD_PROC(&audio_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
4498e06491aSEdward Tomasz Napierala "record", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
4508e06491aSEdward Tomasz Napierala &audio_record, sizeof(audio_record), usb_temp_sysctl,
4518e06491aSEdward Tomasz Napierala "A", "Record interface string");
4528e06491aSEdward Tomasz Napierala SYSCTL_ADD_PROC(&audio_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
4538e06491aSEdward Tomasz Napierala "playback", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
4548e06491aSEdward Tomasz Napierala &audio_playback, sizeof(audio_playback), usb_temp_sysctl,
4558e06491aSEdward Tomasz Napierala "A", "Playback interface string");
4568e06491aSEdward Tomasz Napierala #endif
4578e06491aSEdward Tomasz Napierala SYSCTL_ADD_PROC(&audio_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
4581ee5bed7SEdward Tomasz Napierala "manufacturer", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
4591ee5bed7SEdward Tomasz Napierala &audio_manufacturer, sizeof(audio_manufacturer), usb_temp_sysctl,
4601ee5bed7SEdward Tomasz Napierala "A", "Manufacturer string");
4611ee5bed7SEdward Tomasz Napierala SYSCTL_ADD_PROC(&audio_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
4628e06491aSEdward Tomasz Napierala "product", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
4638e06491aSEdward Tomasz Napierala &audio_product, sizeof(audio_product), usb_temp_sysctl,
4648e06491aSEdward Tomasz Napierala "A", "Product string");
4651ee5bed7SEdward Tomasz Napierala SYSCTL_ADD_PROC(&audio_ctx_list, SYSCTL_CHILDREN(parent), OID_AUTO,
4661ee5bed7SEdward Tomasz Napierala "serial_number", CTLTYPE_STRING | CTLFLAG_RWTUN | CTLFLAG_MPSAFE,
4671ee5bed7SEdward Tomasz Napierala &audio_serial_number, sizeof(audio_serial_number), usb_temp_sysctl,
4681ee5bed7SEdward Tomasz Napierala "A", "Serial number string");
4698e06491aSEdward Tomasz Napierala }
4708e06491aSEdward Tomasz Napierala
4718e06491aSEdward Tomasz Napierala static void
audio_uninit(void * arg __unused)4728e06491aSEdward Tomasz Napierala audio_uninit(void *arg __unused)
4738e06491aSEdward Tomasz Napierala {
4748e06491aSEdward Tomasz Napierala
4758e06491aSEdward Tomasz Napierala sysctl_ctx_free(&audio_ctx_list);
4768e06491aSEdward Tomasz Napierala }
4778e06491aSEdward Tomasz Napierala
4788e06491aSEdward Tomasz Napierala SYSINIT(audio_init, SI_SUB_LOCK, SI_ORDER_FIRST, audio_init, NULL);
479c70e38e4SHans Petter Selasky SYSUNINIT(audio_uninit, SI_SUB_LOCK, SI_ORDER_FIRST, audio_uninit, NULL);
480