1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * USB Audio Driver for ALSA
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Quirks and vendor-specific extensions for mixer interfaces
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Copyright (c) 2002 by Takashi Iwai <tiwai@suse.de>
8*4882a593Smuzhiyun *
9*4882a593Smuzhiyun * Many codes borrowed from audio.c by
10*4882a593Smuzhiyun * Alan Cox (alan@lxorguk.ukuu.org.uk)
11*4882a593Smuzhiyun * Thomas Sailer (sailer@ife.ee.ethz.ch)
12*4882a593Smuzhiyun *
13*4882a593Smuzhiyun * Audio Advantage Micro II support added by:
14*4882a593Smuzhiyun * Przemek Rudy (prudy1@o2.pl)
15*4882a593Smuzhiyun */
16*4882a593Smuzhiyun
17*4882a593Smuzhiyun #include <linux/hid.h>
18*4882a593Smuzhiyun #include <linux/init.h>
19*4882a593Smuzhiyun #include <linux/math64.h>
20*4882a593Smuzhiyun #include <linux/slab.h>
21*4882a593Smuzhiyun #include <linux/usb.h>
22*4882a593Smuzhiyun #include <linux/usb/audio.h>
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun #include <sound/asoundef.h>
25*4882a593Smuzhiyun #include <sound/core.h>
26*4882a593Smuzhiyun #include <sound/control.h>
27*4882a593Smuzhiyun #include <sound/hwdep.h>
28*4882a593Smuzhiyun #include <sound/info.h>
29*4882a593Smuzhiyun #include <sound/tlv.h>
30*4882a593Smuzhiyun
31*4882a593Smuzhiyun #include "usbaudio.h"
32*4882a593Smuzhiyun #include "mixer.h"
33*4882a593Smuzhiyun #include "mixer_quirks.h"
34*4882a593Smuzhiyun #include "mixer_scarlett.h"
35*4882a593Smuzhiyun #include "mixer_scarlett_gen2.h"
36*4882a593Smuzhiyun #include "mixer_us16x08.h"
37*4882a593Smuzhiyun #include "mixer_s1810c.h"
38*4882a593Smuzhiyun #include "helper.h"
39*4882a593Smuzhiyun
40*4882a593Smuzhiyun struct std_mono_table {
41*4882a593Smuzhiyun unsigned int unitid, control, cmask;
42*4882a593Smuzhiyun int val_type;
43*4882a593Smuzhiyun const char *name;
44*4882a593Smuzhiyun snd_kcontrol_tlv_rw_t *tlv_callback;
45*4882a593Smuzhiyun };
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun /* This function allows for the creation of standard UAC controls.
48*4882a593Smuzhiyun * See the quirks for M-Audio FTUs or Ebox-44.
49*4882a593Smuzhiyun * If you don't want to set a TLV callback pass NULL.
50*4882a593Smuzhiyun *
51*4882a593Smuzhiyun * Since there doesn't seem to be a devices that needs a multichannel
52*4882a593Smuzhiyun * version, we keep it mono for simplicity.
53*4882a593Smuzhiyun */
snd_create_std_mono_ctl_offset(struct usb_mixer_interface * mixer,unsigned int unitid,unsigned int control,unsigned int cmask,int val_type,unsigned int idx_off,const char * name,snd_kcontrol_tlv_rw_t * tlv_callback)54*4882a593Smuzhiyun static int snd_create_std_mono_ctl_offset(struct usb_mixer_interface *mixer,
55*4882a593Smuzhiyun unsigned int unitid,
56*4882a593Smuzhiyun unsigned int control,
57*4882a593Smuzhiyun unsigned int cmask,
58*4882a593Smuzhiyun int val_type,
59*4882a593Smuzhiyun unsigned int idx_off,
60*4882a593Smuzhiyun const char *name,
61*4882a593Smuzhiyun snd_kcontrol_tlv_rw_t *tlv_callback)
62*4882a593Smuzhiyun {
63*4882a593Smuzhiyun struct usb_mixer_elem_info *cval;
64*4882a593Smuzhiyun struct snd_kcontrol *kctl;
65*4882a593Smuzhiyun
66*4882a593Smuzhiyun cval = kzalloc(sizeof(*cval), GFP_KERNEL);
67*4882a593Smuzhiyun if (!cval)
68*4882a593Smuzhiyun return -ENOMEM;
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun snd_usb_mixer_elem_init_std(&cval->head, mixer, unitid);
71*4882a593Smuzhiyun cval->val_type = val_type;
72*4882a593Smuzhiyun cval->channels = 1;
73*4882a593Smuzhiyun cval->control = control;
74*4882a593Smuzhiyun cval->cmask = cmask;
75*4882a593Smuzhiyun cval->idx_off = idx_off;
76*4882a593Smuzhiyun
77*4882a593Smuzhiyun /* get_min_max() is called only for integer volumes later,
78*4882a593Smuzhiyun * so provide a short-cut for booleans */
79*4882a593Smuzhiyun cval->min = 0;
80*4882a593Smuzhiyun cval->max = 1;
81*4882a593Smuzhiyun cval->res = 0;
82*4882a593Smuzhiyun cval->dBmin = 0;
83*4882a593Smuzhiyun cval->dBmax = 0;
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun /* Create control */
86*4882a593Smuzhiyun kctl = snd_ctl_new1(snd_usb_feature_unit_ctl, cval);
87*4882a593Smuzhiyun if (!kctl) {
88*4882a593Smuzhiyun kfree(cval);
89*4882a593Smuzhiyun return -ENOMEM;
90*4882a593Smuzhiyun }
91*4882a593Smuzhiyun
92*4882a593Smuzhiyun /* Set name */
93*4882a593Smuzhiyun snprintf(kctl->id.name, sizeof(kctl->id.name), name);
94*4882a593Smuzhiyun kctl->private_free = snd_usb_mixer_elem_free;
95*4882a593Smuzhiyun
96*4882a593Smuzhiyun /* set TLV */
97*4882a593Smuzhiyun if (tlv_callback) {
98*4882a593Smuzhiyun kctl->tlv.c = tlv_callback;
99*4882a593Smuzhiyun kctl->vd[0].access |=
100*4882a593Smuzhiyun SNDRV_CTL_ELEM_ACCESS_TLV_READ |
101*4882a593Smuzhiyun SNDRV_CTL_ELEM_ACCESS_TLV_CALLBACK;
102*4882a593Smuzhiyun }
103*4882a593Smuzhiyun /* Add control to mixer */
104*4882a593Smuzhiyun return snd_usb_mixer_add_control(&cval->head, kctl);
105*4882a593Smuzhiyun }
106*4882a593Smuzhiyun
snd_create_std_mono_ctl(struct usb_mixer_interface * mixer,unsigned int unitid,unsigned int control,unsigned int cmask,int val_type,const char * name,snd_kcontrol_tlv_rw_t * tlv_callback)107*4882a593Smuzhiyun static int snd_create_std_mono_ctl(struct usb_mixer_interface *mixer,
108*4882a593Smuzhiyun unsigned int unitid,
109*4882a593Smuzhiyun unsigned int control,
110*4882a593Smuzhiyun unsigned int cmask,
111*4882a593Smuzhiyun int val_type,
112*4882a593Smuzhiyun const char *name,
113*4882a593Smuzhiyun snd_kcontrol_tlv_rw_t *tlv_callback)
114*4882a593Smuzhiyun {
115*4882a593Smuzhiyun return snd_create_std_mono_ctl_offset(mixer, unitid, control, cmask,
116*4882a593Smuzhiyun val_type, 0 /* Offset */, name, tlv_callback);
117*4882a593Smuzhiyun }
118*4882a593Smuzhiyun
119*4882a593Smuzhiyun /*
120*4882a593Smuzhiyun * Create a set of standard UAC controls from a table
121*4882a593Smuzhiyun */
snd_create_std_mono_table(struct usb_mixer_interface * mixer,const struct std_mono_table * t)122*4882a593Smuzhiyun static int snd_create_std_mono_table(struct usb_mixer_interface *mixer,
123*4882a593Smuzhiyun const struct std_mono_table *t)
124*4882a593Smuzhiyun {
125*4882a593Smuzhiyun int err;
126*4882a593Smuzhiyun
127*4882a593Smuzhiyun while (t->name != NULL) {
128*4882a593Smuzhiyun err = snd_create_std_mono_ctl(mixer, t->unitid, t->control,
129*4882a593Smuzhiyun t->cmask, t->val_type, t->name, t->tlv_callback);
130*4882a593Smuzhiyun if (err < 0)
131*4882a593Smuzhiyun return err;
132*4882a593Smuzhiyun t++;
133*4882a593Smuzhiyun }
134*4882a593Smuzhiyun
135*4882a593Smuzhiyun return 0;
136*4882a593Smuzhiyun }
137*4882a593Smuzhiyun
add_single_ctl_with_resume(struct usb_mixer_interface * mixer,int id,usb_mixer_elem_resume_func_t resume,const struct snd_kcontrol_new * knew,struct usb_mixer_elem_list ** listp)138*4882a593Smuzhiyun static int add_single_ctl_with_resume(struct usb_mixer_interface *mixer,
139*4882a593Smuzhiyun int id,
140*4882a593Smuzhiyun usb_mixer_elem_resume_func_t resume,
141*4882a593Smuzhiyun const struct snd_kcontrol_new *knew,
142*4882a593Smuzhiyun struct usb_mixer_elem_list **listp)
143*4882a593Smuzhiyun {
144*4882a593Smuzhiyun struct usb_mixer_elem_list *list;
145*4882a593Smuzhiyun struct snd_kcontrol *kctl;
146*4882a593Smuzhiyun
147*4882a593Smuzhiyun list = kzalloc(sizeof(*list), GFP_KERNEL);
148*4882a593Smuzhiyun if (!list)
149*4882a593Smuzhiyun return -ENOMEM;
150*4882a593Smuzhiyun if (listp)
151*4882a593Smuzhiyun *listp = list;
152*4882a593Smuzhiyun list->mixer = mixer;
153*4882a593Smuzhiyun list->id = id;
154*4882a593Smuzhiyun list->resume = resume;
155*4882a593Smuzhiyun kctl = snd_ctl_new1(knew, list);
156*4882a593Smuzhiyun if (!kctl) {
157*4882a593Smuzhiyun kfree(list);
158*4882a593Smuzhiyun return -ENOMEM;
159*4882a593Smuzhiyun }
160*4882a593Smuzhiyun kctl->private_free = snd_usb_mixer_elem_free;
161*4882a593Smuzhiyun /* don't use snd_usb_mixer_add_control() here, this is a special list element */
162*4882a593Smuzhiyun return snd_usb_mixer_add_list(list, kctl, false);
163*4882a593Smuzhiyun }
164*4882a593Smuzhiyun
165*4882a593Smuzhiyun /*
166*4882a593Smuzhiyun * Sound Blaster remote control configuration
167*4882a593Smuzhiyun *
168*4882a593Smuzhiyun * format of remote control data:
169*4882a593Smuzhiyun * Extigy: xx 00
170*4882a593Smuzhiyun * Audigy 2 NX: 06 80 xx 00 00 00
171*4882a593Smuzhiyun * Live! 24-bit: 06 80 xx yy 22 83
172*4882a593Smuzhiyun */
173*4882a593Smuzhiyun static const struct rc_config {
174*4882a593Smuzhiyun u32 usb_id;
175*4882a593Smuzhiyun u8 offset;
176*4882a593Smuzhiyun u8 length;
177*4882a593Smuzhiyun u8 packet_length;
178*4882a593Smuzhiyun u8 min_packet_length; /* minimum accepted length of the URB result */
179*4882a593Smuzhiyun u8 mute_mixer_id;
180*4882a593Smuzhiyun u32 mute_code;
181*4882a593Smuzhiyun } rc_configs[] = {
182*4882a593Smuzhiyun { USB_ID(0x041e, 0x3000), 0, 1, 2, 1, 18, 0x0013 }, /* Extigy */
183*4882a593Smuzhiyun { USB_ID(0x041e, 0x3020), 2, 1, 6, 6, 18, 0x0013 }, /* Audigy 2 NX */
184*4882a593Smuzhiyun { USB_ID(0x041e, 0x3040), 2, 2, 6, 6, 2, 0x6e91 }, /* Live! 24-bit */
185*4882a593Smuzhiyun { USB_ID(0x041e, 0x3042), 0, 1, 1, 1, 1, 0x000d }, /* Usb X-Fi S51 */
186*4882a593Smuzhiyun { USB_ID(0x041e, 0x30df), 0, 1, 1, 1, 1, 0x000d }, /* Usb X-Fi S51 Pro */
187*4882a593Smuzhiyun { USB_ID(0x041e, 0x3237), 0, 1, 1, 1, 1, 0x000d }, /* Usb X-Fi S51 Pro */
188*4882a593Smuzhiyun { USB_ID(0x041e, 0x3263), 0, 1, 1, 1, 1, 0x000d }, /* Usb X-Fi S51 Pro */
189*4882a593Smuzhiyun { USB_ID(0x041e, 0x3048), 2, 2, 6, 6, 2, 0x6e91 }, /* Toshiba SB0500 */
190*4882a593Smuzhiyun };
191*4882a593Smuzhiyun
snd_usb_soundblaster_remote_complete(struct urb * urb)192*4882a593Smuzhiyun static void snd_usb_soundblaster_remote_complete(struct urb *urb)
193*4882a593Smuzhiyun {
194*4882a593Smuzhiyun struct usb_mixer_interface *mixer = urb->context;
195*4882a593Smuzhiyun const struct rc_config *rc = mixer->rc_cfg;
196*4882a593Smuzhiyun u32 code;
197*4882a593Smuzhiyun
198*4882a593Smuzhiyun if (urb->status < 0 || urb->actual_length < rc->min_packet_length)
199*4882a593Smuzhiyun return;
200*4882a593Smuzhiyun
201*4882a593Smuzhiyun code = mixer->rc_buffer[rc->offset];
202*4882a593Smuzhiyun if (rc->length == 2)
203*4882a593Smuzhiyun code |= mixer->rc_buffer[rc->offset + 1] << 8;
204*4882a593Smuzhiyun
205*4882a593Smuzhiyun /* the Mute button actually changes the mixer control */
206*4882a593Smuzhiyun if (code == rc->mute_code)
207*4882a593Smuzhiyun snd_usb_mixer_notify_id(mixer, rc->mute_mixer_id);
208*4882a593Smuzhiyun mixer->rc_code = code;
209*4882a593Smuzhiyun wmb();
210*4882a593Smuzhiyun wake_up(&mixer->rc_waitq);
211*4882a593Smuzhiyun }
212*4882a593Smuzhiyun
snd_usb_sbrc_hwdep_read(struct snd_hwdep * hw,char __user * buf,long count,loff_t * offset)213*4882a593Smuzhiyun static long snd_usb_sbrc_hwdep_read(struct snd_hwdep *hw, char __user *buf,
214*4882a593Smuzhiyun long count, loff_t *offset)
215*4882a593Smuzhiyun {
216*4882a593Smuzhiyun struct usb_mixer_interface *mixer = hw->private_data;
217*4882a593Smuzhiyun int err;
218*4882a593Smuzhiyun u32 rc_code;
219*4882a593Smuzhiyun
220*4882a593Smuzhiyun if (count != 1 && count != 4)
221*4882a593Smuzhiyun return -EINVAL;
222*4882a593Smuzhiyun err = wait_event_interruptible(mixer->rc_waitq,
223*4882a593Smuzhiyun (rc_code = xchg(&mixer->rc_code, 0)) != 0);
224*4882a593Smuzhiyun if (err == 0) {
225*4882a593Smuzhiyun if (count == 1)
226*4882a593Smuzhiyun err = put_user(rc_code, buf);
227*4882a593Smuzhiyun else
228*4882a593Smuzhiyun err = put_user(rc_code, (u32 __user *)buf);
229*4882a593Smuzhiyun }
230*4882a593Smuzhiyun return err < 0 ? err : count;
231*4882a593Smuzhiyun }
232*4882a593Smuzhiyun
snd_usb_sbrc_hwdep_poll(struct snd_hwdep * hw,struct file * file,poll_table * wait)233*4882a593Smuzhiyun static __poll_t snd_usb_sbrc_hwdep_poll(struct snd_hwdep *hw, struct file *file,
234*4882a593Smuzhiyun poll_table *wait)
235*4882a593Smuzhiyun {
236*4882a593Smuzhiyun struct usb_mixer_interface *mixer = hw->private_data;
237*4882a593Smuzhiyun
238*4882a593Smuzhiyun poll_wait(file, &mixer->rc_waitq, wait);
239*4882a593Smuzhiyun return mixer->rc_code ? EPOLLIN | EPOLLRDNORM : 0;
240*4882a593Smuzhiyun }
241*4882a593Smuzhiyun
snd_usb_soundblaster_remote_init(struct usb_mixer_interface * mixer)242*4882a593Smuzhiyun static int snd_usb_soundblaster_remote_init(struct usb_mixer_interface *mixer)
243*4882a593Smuzhiyun {
244*4882a593Smuzhiyun struct snd_hwdep *hwdep;
245*4882a593Smuzhiyun int err, len, i;
246*4882a593Smuzhiyun
247*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(rc_configs); ++i)
248*4882a593Smuzhiyun if (rc_configs[i].usb_id == mixer->chip->usb_id)
249*4882a593Smuzhiyun break;
250*4882a593Smuzhiyun if (i >= ARRAY_SIZE(rc_configs))
251*4882a593Smuzhiyun return 0;
252*4882a593Smuzhiyun mixer->rc_cfg = &rc_configs[i];
253*4882a593Smuzhiyun
254*4882a593Smuzhiyun len = mixer->rc_cfg->packet_length;
255*4882a593Smuzhiyun
256*4882a593Smuzhiyun init_waitqueue_head(&mixer->rc_waitq);
257*4882a593Smuzhiyun err = snd_hwdep_new(mixer->chip->card, "SB remote control", 0, &hwdep);
258*4882a593Smuzhiyun if (err < 0)
259*4882a593Smuzhiyun return err;
260*4882a593Smuzhiyun snprintf(hwdep->name, sizeof(hwdep->name),
261*4882a593Smuzhiyun "%s remote control", mixer->chip->card->shortname);
262*4882a593Smuzhiyun hwdep->iface = SNDRV_HWDEP_IFACE_SB_RC;
263*4882a593Smuzhiyun hwdep->private_data = mixer;
264*4882a593Smuzhiyun hwdep->ops.read = snd_usb_sbrc_hwdep_read;
265*4882a593Smuzhiyun hwdep->ops.poll = snd_usb_sbrc_hwdep_poll;
266*4882a593Smuzhiyun hwdep->exclusive = 1;
267*4882a593Smuzhiyun
268*4882a593Smuzhiyun mixer->rc_urb = usb_alloc_urb(0, GFP_KERNEL);
269*4882a593Smuzhiyun if (!mixer->rc_urb)
270*4882a593Smuzhiyun return -ENOMEM;
271*4882a593Smuzhiyun mixer->rc_setup_packet = kmalloc(sizeof(*mixer->rc_setup_packet), GFP_KERNEL);
272*4882a593Smuzhiyun if (!mixer->rc_setup_packet) {
273*4882a593Smuzhiyun usb_free_urb(mixer->rc_urb);
274*4882a593Smuzhiyun mixer->rc_urb = NULL;
275*4882a593Smuzhiyun return -ENOMEM;
276*4882a593Smuzhiyun }
277*4882a593Smuzhiyun mixer->rc_setup_packet->bRequestType =
278*4882a593Smuzhiyun USB_DIR_IN | USB_TYPE_CLASS | USB_RECIP_INTERFACE;
279*4882a593Smuzhiyun mixer->rc_setup_packet->bRequest = UAC_GET_MEM;
280*4882a593Smuzhiyun mixer->rc_setup_packet->wValue = cpu_to_le16(0);
281*4882a593Smuzhiyun mixer->rc_setup_packet->wIndex = cpu_to_le16(0);
282*4882a593Smuzhiyun mixer->rc_setup_packet->wLength = cpu_to_le16(len);
283*4882a593Smuzhiyun usb_fill_control_urb(mixer->rc_urb, mixer->chip->dev,
284*4882a593Smuzhiyun usb_rcvctrlpipe(mixer->chip->dev, 0),
285*4882a593Smuzhiyun (u8*)mixer->rc_setup_packet, mixer->rc_buffer, len,
286*4882a593Smuzhiyun snd_usb_soundblaster_remote_complete, mixer);
287*4882a593Smuzhiyun return 0;
288*4882a593Smuzhiyun }
289*4882a593Smuzhiyun
290*4882a593Smuzhiyun #define snd_audigy2nx_led_info snd_ctl_boolean_mono_info
291*4882a593Smuzhiyun
snd_audigy2nx_led_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)292*4882a593Smuzhiyun static int snd_audigy2nx_led_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
293*4882a593Smuzhiyun {
294*4882a593Smuzhiyun ucontrol->value.integer.value[0] = kcontrol->private_value >> 8;
295*4882a593Smuzhiyun return 0;
296*4882a593Smuzhiyun }
297*4882a593Smuzhiyun
snd_audigy2nx_led_update(struct usb_mixer_interface * mixer,int value,int index)298*4882a593Smuzhiyun static int snd_audigy2nx_led_update(struct usb_mixer_interface *mixer,
299*4882a593Smuzhiyun int value, int index)
300*4882a593Smuzhiyun {
301*4882a593Smuzhiyun struct snd_usb_audio *chip = mixer->chip;
302*4882a593Smuzhiyun int err;
303*4882a593Smuzhiyun
304*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
305*4882a593Smuzhiyun if (err < 0)
306*4882a593Smuzhiyun return err;
307*4882a593Smuzhiyun
308*4882a593Smuzhiyun if (chip->usb_id == USB_ID(0x041e, 0x3042))
309*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
310*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0), 0x24,
311*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_OTHER,
312*4882a593Smuzhiyun !value, 0, NULL, 0);
313*4882a593Smuzhiyun /* USB X-Fi S51 Pro */
314*4882a593Smuzhiyun if (chip->usb_id == USB_ID(0x041e, 0x30df))
315*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
316*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0), 0x24,
317*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_OTHER,
318*4882a593Smuzhiyun !value, 0, NULL, 0);
319*4882a593Smuzhiyun else
320*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
321*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0), 0x24,
322*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_OTHER,
323*4882a593Smuzhiyun value, index + 2, NULL, 0);
324*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
325*4882a593Smuzhiyun return err;
326*4882a593Smuzhiyun }
327*4882a593Smuzhiyun
snd_audigy2nx_led_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)328*4882a593Smuzhiyun static int snd_audigy2nx_led_put(struct snd_kcontrol *kcontrol,
329*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
330*4882a593Smuzhiyun {
331*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
332*4882a593Smuzhiyun struct usb_mixer_interface *mixer = list->mixer;
333*4882a593Smuzhiyun int index = kcontrol->private_value & 0xff;
334*4882a593Smuzhiyun unsigned int value = ucontrol->value.integer.value[0];
335*4882a593Smuzhiyun int old_value = kcontrol->private_value >> 8;
336*4882a593Smuzhiyun int err;
337*4882a593Smuzhiyun
338*4882a593Smuzhiyun if (value > 1)
339*4882a593Smuzhiyun return -EINVAL;
340*4882a593Smuzhiyun if (value == old_value)
341*4882a593Smuzhiyun return 0;
342*4882a593Smuzhiyun kcontrol->private_value = (value << 8) | index;
343*4882a593Smuzhiyun err = snd_audigy2nx_led_update(mixer, value, index);
344*4882a593Smuzhiyun return err < 0 ? err : 1;
345*4882a593Smuzhiyun }
346*4882a593Smuzhiyun
snd_audigy2nx_led_resume(struct usb_mixer_elem_list * list)347*4882a593Smuzhiyun static int snd_audigy2nx_led_resume(struct usb_mixer_elem_list *list)
348*4882a593Smuzhiyun {
349*4882a593Smuzhiyun int priv_value = list->kctl->private_value;
350*4882a593Smuzhiyun
351*4882a593Smuzhiyun return snd_audigy2nx_led_update(list->mixer, priv_value >> 8,
352*4882a593Smuzhiyun priv_value & 0xff);
353*4882a593Smuzhiyun }
354*4882a593Smuzhiyun
355*4882a593Smuzhiyun /* name and private_value are set dynamically */
356*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_audigy2nx_control = {
357*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
358*4882a593Smuzhiyun .info = snd_audigy2nx_led_info,
359*4882a593Smuzhiyun .get = snd_audigy2nx_led_get,
360*4882a593Smuzhiyun .put = snd_audigy2nx_led_put,
361*4882a593Smuzhiyun };
362*4882a593Smuzhiyun
363*4882a593Smuzhiyun static const char * const snd_audigy2nx_led_names[] = {
364*4882a593Smuzhiyun "CMSS LED Switch",
365*4882a593Smuzhiyun "Power LED Switch",
366*4882a593Smuzhiyun "Dolby Digital LED Switch",
367*4882a593Smuzhiyun };
368*4882a593Smuzhiyun
snd_audigy2nx_controls_create(struct usb_mixer_interface * mixer)369*4882a593Smuzhiyun static int snd_audigy2nx_controls_create(struct usb_mixer_interface *mixer)
370*4882a593Smuzhiyun {
371*4882a593Smuzhiyun int i, err;
372*4882a593Smuzhiyun
373*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(snd_audigy2nx_led_names); ++i) {
374*4882a593Smuzhiyun struct snd_kcontrol_new knew;
375*4882a593Smuzhiyun
376*4882a593Smuzhiyun /* USB X-Fi S51 doesn't have a CMSS LED */
377*4882a593Smuzhiyun if ((mixer->chip->usb_id == USB_ID(0x041e, 0x3042)) && i == 0)
378*4882a593Smuzhiyun continue;
379*4882a593Smuzhiyun /* USB X-Fi S51 Pro doesn't have one either */
380*4882a593Smuzhiyun if ((mixer->chip->usb_id == USB_ID(0x041e, 0x30df)) && i == 0)
381*4882a593Smuzhiyun continue;
382*4882a593Smuzhiyun if (i > 1 && /* Live24ext has 2 LEDs only */
383*4882a593Smuzhiyun (mixer->chip->usb_id == USB_ID(0x041e, 0x3040) ||
384*4882a593Smuzhiyun mixer->chip->usb_id == USB_ID(0x041e, 0x3042) ||
385*4882a593Smuzhiyun mixer->chip->usb_id == USB_ID(0x041e, 0x30df) ||
386*4882a593Smuzhiyun mixer->chip->usb_id == USB_ID(0x041e, 0x3048)))
387*4882a593Smuzhiyun break;
388*4882a593Smuzhiyun
389*4882a593Smuzhiyun knew = snd_audigy2nx_control;
390*4882a593Smuzhiyun knew.name = snd_audigy2nx_led_names[i];
391*4882a593Smuzhiyun knew.private_value = (1 << 8) | i; /* LED on as default */
392*4882a593Smuzhiyun err = add_single_ctl_with_resume(mixer, 0,
393*4882a593Smuzhiyun snd_audigy2nx_led_resume,
394*4882a593Smuzhiyun &knew, NULL);
395*4882a593Smuzhiyun if (err < 0)
396*4882a593Smuzhiyun return err;
397*4882a593Smuzhiyun }
398*4882a593Smuzhiyun return 0;
399*4882a593Smuzhiyun }
400*4882a593Smuzhiyun
snd_audigy2nx_proc_read(struct snd_info_entry * entry,struct snd_info_buffer * buffer)401*4882a593Smuzhiyun static void snd_audigy2nx_proc_read(struct snd_info_entry *entry,
402*4882a593Smuzhiyun struct snd_info_buffer *buffer)
403*4882a593Smuzhiyun {
404*4882a593Smuzhiyun static const struct sb_jack {
405*4882a593Smuzhiyun int unitid;
406*4882a593Smuzhiyun const char *name;
407*4882a593Smuzhiyun } jacks_audigy2nx[] = {
408*4882a593Smuzhiyun {4, "dig in "},
409*4882a593Smuzhiyun {7, "line in"},
410*4882a593Smuzhiyun {19, "spk out"},
411*4882a593Smuzhiyun {20, "hph out"},
412*4882a593Smuzhiyun {-1, NULL}
413*4882a593Smuzhiyun }, jacks_live24ext[] = {
414*4882a593Smuzhiyun {4, "line in"}, /* &1=Line, &2=Mic*/
415*4882a593Smuzhiyun {3, "hph out"}, /* headphones */
416*4882a593Smuzhiyun {0, "RC "}, /* last command, 6 bytes see rc_config above */
417*4882a593Smuzhiyun {-1, NULL}
418*4882a593Smuzhiyun };
419*4882a593Smuzhiyun const struct sb_jack *jacks;
420*4882a593Smuzhiyun struct usb_mixer_interface *mixer = entry->private_data;
421*4882a593Smuzhiyun int i, err;
422*4882a593Smuzhiyun u8 buf[3];
423*4882a593Smuzhiyun
424*4882a593Smuzhiyun snd_iprintf(buffer, "%s jacks\n\n", mixer->chip->card->shortname);
425*4882a593Smuzhiyun if (mixer->chip->usb_id == USB_ID(0x041e, 0x3020))
426*4882a593Smuzhiyun jacks = jacks_audigy2nx;
427*4882a593Smuzhiyun else if (mixer->chip->usb_id == USB_ID(0x041e, 0x3040) ||
428*4882a593Smuzhiyun mixer->chip->usb_id == USB_ID(0x041e, 0x3048))
429*4882a593Smuzhiyun jacks = jacks_live24ext;
430*4882a593Smuzhiyun else
431*4882a593Smuzhiyun return;
432*4882a593Smuzhiyun
433*4882a593Smuzhiyun for (i = 0; jacks[i].name; ++i) {
434*4882a593Smuzhiyun snd_iprintf(buffer, "%s: ", jacks[i].name);
435*4882a593Smuzhiyun err = snd_usb_lock_shutdown(mixer->chip);
436*4882a593Smuzhiyun if (err < 0)
437*4882a593Smuzhiyun return;
438*4882a593Smuzhiyun err = snd_usb_ctl_msg(mixer->chip->dev,
439*4882a593Smuzhiyun usb_rcvctrlpipe(mixer->chip->dev, 0),
440*4882a593Smuzhiyun UAC_GET_MEM, USB_DIR_IN | USB_TYPE_CLASS |
441*4882a593Smuzhiyun USB_RECIP_INTERFACE, 0,
442*4882a593Smuzhiyun jacks[i].unitid << 8, buf, 3);
443*4882a593Smuzhiyun snd_usb_unlock_shutdown(mixer->chip);
444*4882a593Smuzhiyun if (err == 3 && (buf[0] == 3 || buf[0] == 6))
445*4882a593Smuzhiyun snd_iprintf(buffer, "%02x %02x\n", buf[1], buf[2]);
446*4882a593Smuzhiyun else
447*4882a593Smuzhiyun snd_iprintf(buffer, "?\n");
448*4882a593Smuzhiyun }
449*4882a593Smuzhiyun }
450*4882a593Smuzhiyun
451*4882a593Smuzhiyun /* EMU0204 */
snd_emu0204_ch_switch_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)452*4882a593Smuzhiyun static int snd_emu0204_ch_switch_info(struct snd_kcontrol *kcontrol,
453*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
454*4882a593Smuzhiyun {
455*4882a593Smuzhiyun static const char * const texts[2] = {"1/2", "3/4"};
456*4882a593Smuzhiyun
457*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1, ARRAY_SIZE(texts), texts);
458*4882a593Smuzhiyun }
459*4882a593Smuzhiyun
snd_emu0204_ch_switch_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)460*4882a593Smuzhiyun static int snd_emu0204_ch_switch_get(struct snd_kcontrol *kcontrol,
461*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
462*4882a593Smuzhiyun {
463*4882a593Smuzhiyun ucontrol->value.enumerated.item[0] = kcontrol->private_value;
464*4882a593Smuzhiyun return 0;
465*4882a593Smuzhiyun }
466*4882a593Smuzhiyun
snd_emu0204_ch_switch_update(struct usb_mixer_interface * mixer,int value)467*4882a593Smuzhiyun static int snd_emu0204_ch_switch_update(struct usb_mixer_interface *mixer,
468*4882a593Smuzhiyun int value)
469*4882a593Smuzhiyun {
470*4882a593Smuzhiyun struct snd_usb_audio *chip = mixer->chip;
471*4882a593Smuzhiyun int err;
472*4882a593Smuzhiyun unsigned char buf[2];
473*4882a593Smuzhiyun
474*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
475*4882a593Smuzhiyun if (err < 0)
476*4882a593Smuzhiyun return err;
477*4882a593Smuzhiyun
478*4882a593Smuzhiyun buf[0] = 0x01;
479*4882a593Smuzhiyun buf[1] = value ? 0x02 : 0x01;
480*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
481*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0), UAC_SET_CUR,
482*4882a593Smuzhiyun USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_OUT,
483*4882a593Smuzhiyun 0x0400, 0x0e00, buf, 2);
484*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
485*4882a593Smuzhiyun return err;
486*4882a593Smuzhiyun }
487*4882a593Smuzhiyun
snd_emu0204_ch_switch_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)488*4882a593Smuzhiyun static int snd_emu0204_ch_switch_put(struct snd_kcontrol *kcontrol,
489*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
490*4882a593Smuzhiyun {
491*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
492*4882a593Smuzhiyun struct usb_mixer_interface *mixer = list->mixer;
493*4882a593Smuzhiyun unsigned int value = ucontrol->value.enumerated.item[0];
494*4882a593Smuzhiyun int err;
495*4882a593Smuzhiyun
496*4882a593Smuzhiyun if (value > 1)
497*4882a593Smuzhiyun return -EINVAL;
498*4882a593Smuzhiyun
499*4882a593Smuzhiyun if (value == kcontrol->private_value)
500*4882a593Smuzhiyun return 0;
501*4882a593Smuzhiyun
502*4882a593Smuzhiyun kcontrol->private_value = value;
503*4882a593Smuzhiyun err = snd_emu0204_ch_switch_update(mixer, value);
504*4882a593Smuzhiyun return err < 0 ? err : 1;
505*4882a593Smuzhiyun }
506*4882a593Smuzhiyun
snd_emu0204_ch_switch_resume(struct usb_mixer_elem_list * list)507*4882a593Smuzhiyun static int snd_emu0204_ch_switch_resume(struct usb_mixer_elem_list *list)
508*4882a593Smuzhiyun {
509*4882a593Smuzhiyun return snd_emu0204_ch_switch_update(list->mixer,
510*4882a593Smuzhiyun list->kctl->private_value);
511*4882a593Smuzhiyun }
512*4882a593Smuzhiyun
513*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_emu0204_control = {
514*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
515*4882a593Smuzhiyun .name = "Front Jack Channels",
516*4882a593Smuzhiyun .info = snd_emu0204_ch_switch_info,
517*4882a593Smuzhiyun .get = snd_emu0204_ch_switch_get,
518*4882a593Smuzhiyun .put = snd_emu0204_ch_switch_put,
519*4882a593Smuzhiyun .private_value = 0,
520*4882a593Smuzhiyun };
521*4882a593Smuzhiyun
snd_emu0204_controls_create(struct usb_mixer_interface * mixer)522*4882a593Smuzhiyun static int snd_emu0204_controls_create(struct usb_mixer_interface *mixer)
523*4882a593Smuzhiyun {
524*4882a593Smuzhiyun return add_single_ctl_with_resume(mixer, 0,
525*4882a593Smuzhiyun snd_emu0204_ch_switch_resume,
526*4882a593Smuzhiyun &snd_emu0204_control, NULL);
527*4882a593Smuzhiyun }
528*4882a593Smuzhiyun
529*4882a593Smuzhiyun /* ASUS Xonar U1 / U3 controls */
530*4882a593Smuzhiyun
snd_xonar_u1_switch_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)531*4882a593Smuzhiyun static int snd_xonar_u1_switch_get(struct snd_kcontrol *kcontrol,
532*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
533*4882a593Smuzhiyun {
534*4882a593Smuzhiyun ucontrol->value.integer.value[0] = !!(kcontrol->private_value & 0x02);
535*4882a593Smuzhiyun return 0;
536*4882a593Smuzhiyun }
537*4882a593Smuzhiyun
snd_xonar_u1_switch_update(struct usb_mixer_interface * mixer,unsigned char status)538*4882a593Smuzhiyun static int snd_xonar_u1_switch_update(struct usb_mixer_interface *mixer,
539*4882a593Smuzhiyun unsigned char status)
540*4882a593Smuzhiyun {
541*4882a593Smuzhiyun struct snd_usb_audio *chip = mixer->chip;
542*4882a593Smuzhiyun int err;
543*4882a593Smuzhiyun
544*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
545*4882a593Smuzhiyun if (err < 0)
546*4882a593Smuzhiyun return err;
547*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
548*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0), 0x08,
549*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_OTHER,
550*4882a593Smuzhiyun 50, 0, &status, 1);
551*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
552*4882a593Smuzhiyun return err;
553*4882a593Smuzhiyun }
554*4882a593Smuzhiyun
snd_xonar_u1_switch_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)555*4882a593Smuzhiyun static int snd_xonar_u1_switch_put(struct snd_kcontrol *kcontrol,
556*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
557*4882a593Smuzhiyun {
558*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
559*4882a593Smuzhiyun u8 old_status, new_status;
560*4882a593Smuzhiyun int err;
561*4882a593Smuzhiyun
562*4882a593Smuzhiyun old_status = kcontrol->private_value;
563*4882a593Smuzhiyun if (ucontrol->value.integer.value[0])
564*4882a593Smuzhiyun new_status = old_status | 0x02;
565*4882a593Smuzhiyun else
566*4882a593Smuzhiyun new_status = old_status & ~0x02;
567*4882a593Smuzhiyun if (new_status == old_status)
568*4882a593Smuzhiyun return 0;
569*4882a593Smuzhiyun
570*4882a593Smuzhiyun kcontrol->private_value = new_status;
571*4882a593Smuzhiyun err = snd_xonar_u1_switch_update(list->mixer, new_status);
572*4882a593Smuzhiyun return err < 0 ? err : 1;
573*4882a593Smuzhiyun }
574*4882a593Smuzhiyun
snd_xonar_u1_switch_resume(struct usb_mixer_elem_list * list)575*4882a593Smuzhiyun static int snd_xonar_u1_switch_resume(struct usb_mixer_elem_list *list)
576*4882a593Smuzhiyun {
577*4882a593Smuzhiyun return snd_xonar_u1_switch_update(list->mixer,
578*4882a593Smuzhiyun list->kctl->private_value);
579*4882a593Smuzhiyun }
580*4882a593Smuzhiyun
581*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_xonar_u1_output_switch = {
582*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
583*4882a593Smuzhiyun .name = "Digital Playback Switch",
584*4882a593Smuzhiyun .info = snd_ctl_boolean_mono_info,
585*4882a593Smuzhiyun .get = snd_xonar_u1_switch_get,
586*4882a593Smuzhiyun .put = snd_xonar_u1_switch_put,
587*4882a593Smuzhiyun .private_value = 0x05,
588*4882a593Smuzhiyun };
589*4882a593Smuzhiyun
snd_xonar_u1_controls_create(struct usb_mixer_interface * mixer)590*4882a593Smuzhiyun static int snd_xonar_u1_controls_create(struct usb_mixer_interface *mixer)
591*4882a593Smuzhiyun {
592*4882a593Smuzhiyun return add_single_ctl_with_resume(mixer, 0,
593*4882a593Smuzhiyun snd_xonar_u1_switch_resume,
594*4882a593Smuzhiyun &snd_xonar_u1_output_switch, NULL);
595*4882a593Smuzhiyun }
596*4882a593Smuzhiyun
597*4882a593Smuzhiyun /* Digidesign Mbox 1 clock source switch (internal/spdif) */
598*4882a593Smuzhiyun
snd_mbox1_switch_get(struct snd_kcontrol * kctl,struct snd_ctl_elem_value * ucontrol)599*4882a593Smuzhiyun static int snd_mbox1_switch_get(struct snd_kcontrol *kctl,
600*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
601*4882a593Smuzhiyun {
602*4882a593Smuzhiyun ucontrol->value.enumerated.item[0] = kctl->private_value;
603*4882a593Smuzhiyun return 0;
604*4882a593Smuzhiyun }
605*4882a593Smuzhiyun
snd_mbox1_switch_update(struct usb_mixer_interface * mixer,int val)606*4882a593Smuzhiyun static int snd_mbox1_switch_update(struct usb_mixer_interface *mixer, int val)
607*4882a593Smuzhiyun {
608*4882a593Smuzhiyun struct snd_usb_audio *chip = mixer->chip;
609*4882a593Smuzhiyun int err;
610*4882a593Smuzhiyun unsigned char buff[3];
611*4882a593Smuzhiyun
612*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
613*4882a593Smuzhiyun if (err < 0)
614*4882a593Smuzhiyun return err;
615*4882a593Smuzhiyun
616*4882a593Smuzhiyun /* Prepare for magic command to toggle clock source */
617*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
618*4882a593Smuzhiyun usb_rcvctrlpipe(chip->dev, 0), 0x81,
619*4882a593Smuzhiyun USB_DIR_IN |
620*4882a593Smuzhiyun USB_TYPE_CLASS |
621*4882a593Smuzhiyun USB_RECIP_INTERFACE, 0x00, 0x500, buff, 1);
622*4882a593Smuzhiyun if (err < 0)
623*4882a593Smuzhiyun goto err;
624*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
625*4882a593Smuzhiyun usb_rcvctrlpipe(chip->dev, 0), 0x81,
626*4882a593Smuzhiyun USB_DIR_IN |
627*4882a593Smuzhiyun USB_TYPE_CLASS |
628*4882a593Smuzhiyun USB_RECIP_ENDPOINT, 0x100, 0x81, buff, 3);
629*4882a593Smuzhiyun if (err < 0)
630*4882a593Smuzhiyun goto err;
631*4882a593Smuzhiyun
632*4882a593Smuzhiyun /* 2 possibilities: Internal -> send sample rate
633*4882a593Smuzhiyun * S/PDIF sync -> send zeroes
634*4882a593Smuzhiyun * NB: Sample rate locked to 48kHz on purpose to
635*4882a593Smuzhiyun * prevent user from resetting the sample rate
636*4882a593Smuzhiyun * while S/PDIF sync is enabled and confusing
637*4882a593Smuzhiyun * this configuration.
638*4882a593Smuzhiyun */
639*4882a593Smuzhiyun if (val == 0) {
640*4882a593Smuzhiyun buff[0] = 0x80;
641*4882a593Smuzhiyun buff[1] = 0xbb;
642*4882a593Smuzhiyun buff[2] = 0x00;
643*4882a593Smuzhiyun } else {
644*4882a593Smuzhiyun buff[0] = buff[1] = buff[2] = 0x00;
645*4882a593Smuzhiyun }
646*4882a593Smuzhiyun
647*4882a593Smuzhiyun /* Send the magic command to toggle the clock source */
648*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
649*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0), 0x1,
650*4882a593Smuzhiyun USB_TYPE_CLASS |
651*4882a593Smuzhiyun USB_RECIP_ENDPOINT, 0x100, 0x81, buff, 3);
652*4882a593Smuzhiyun if (err < 0)
653*4882a593Smuzhiyun goto err;
654*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
655*4882a593Smuzhiyun usb_rcvctrlpipe(chip->dev, 0), 0x81,
656*4882a593Smuzhiyun USB_DIR_IN |
657*4882a593Smuzhiyun USB_TYPE_CLASS |
658*4882a593Smuzhiyun USB_RECIP_ENDPOINT, 0x100, 0x81, buff, 3);
659*4882a593Smuzhiyun if (err < 0)
660*4882a593Smuzhiyun goto err;
661*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
662*4882a593Smuzhiyun usb_rcvctrlpipe(chip->dev, 0), 0x81,
663*4882a593Smuzhiyun USB_DIR_IN |
664*4882a593Smuzhiyun USB_TYPE_CLASS |
665*4882a593Smuzhiyun USB_RECIP_ENDPOINT, 0x100, 0x2, buff, 3);
666*4882a593Smuzhiyun if (err < 0)
667*4882a593Smuzhiyun goto err;
668*4882a593Smuzhiyun
669*4882a593Smuzhiyun err:
670*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
671*4882a593Smuzhiyun return err;
672*4882a593Smuzhiyun }
673*4882a593Smuzhiyun
snd_mbox1_switch_put(struct snd_kcontrol * kctl,struct snd_ctl_elem_value * ucontrol)674*4882a593Smuzhiyun static int snd_mbox1_switch_put(struct snd_kcontrol *kctl,
675*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
676*4882a593Smuzhiyun {
677*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kctl);
678*4882a593Smuzhiyun struct usb_mixer_interface *mixer = list->mixer;
679*4882a593Smuzhiyun int err;
680*4882a593Smuzhiyun bool cur_val, new_val;
681*4882a593Smuzhiyun
682*4882a593Smuzhiyun cur_val = kctl->private_value;
683*4882a593Smuzhiyun new_val = ucontrol->value.enumerated.item[0];
684*4882a593Smuzhiyun if (cur_val == new_val)
685*4882a593Smuzhiyun return 0;
686*4882a593Smuzhiyun
687*4882a593Smuzhiyun kctl->private_value = new_val;
688*4882a593Smuzhiyun err = snd_mbox1_switch_update(mixer, new_val);
689*4882a593Smuzhiyun return err < 0 ? err : 1;
690*4882a593Smuzhiyun }
691*4882a593Smuzhiyun
snd_mbox1_switch_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)692*4882a593Smuzhiyun static int snd_mbox1_switch_info(struct snd_kcontrol *kcontrol,
693*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
694*4882a593Smuzhiyun {
695*4882a593Smuzhiyun static const char *const texts[2] = {
696*4882a593Smuzhiyun "Internal",
697*4882a593Smuzhiyun "S/PDIF"
698*4882a593Smuzhiyun };
699*4882a593Smuzhiyun
700*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1, ARRAY_SIZE(texts), texts);
701*4882a593Smuzhiyun }
702*4882a593Smuzhiyun
snd_mbox1_switch_resume(struct usb_mixer_elem_list * list)703*4882a593Smuzhiyun static int snd_mbox1_switch_resume(struct usb_mixer_elem_list *list)
704*4882a593Smuzhiyun {
705*4882a593Smuzhiyun return snd_mbox1_switch_update(list->mixer, list->kctl->private_value);
706*4882a593Smuzhiyun }
707*4882a593Smuzhiyun
708*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_mbox1_switch = {
709*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
710*4882a593Smuzhiyun .name = "Clock Source",
711*4882a593Smuzhiyun .index = 0,
712*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
713*4882a593Smuzhiyun .info = snd_mbox1_switch_info,
714*4882a593Smuzhiyun .get = snd_mbox1_switch_get,
715*4882a593Smuzhiyun .put = snd_mbox1_switch_put,
716*4882a593Smuzhiyun .private_value = 0
717*4882a593Smuzhiyun };
718*4882a593Smuzhiyun
snd_mbox1_create_sync_switch(struct usb_mixer_interface * mixer)719*4882a593Smuzhiyun static int snd_mbox1_create_sync_switch(struct usb_mixer_interface *mixer)
720*4882a593Smuzhiyun {
721*4882a593Smuzhiyun return add_single_ctl_with_resume(mixer, 0,
722*4882a593Smuzhiyun snd_mbox1_switch_resume,
723*4882a593Smuzhiyun &snd_mbox1_switch, NULL);
724*4882a593Smuzhiyun }
725*4882a593Smuzhiyun
726*4882a593Smuzhiyun /* Native Instruments device quirks */
727*4882a593Smuzhiyun
728*4882a593Smuzhiyun #define _MAKE_NI_CONTROL(bRequest,wIndex) ((bRequest) << 16 | (wIndex))
729*4882a593Smuzhiyun
snd_ni_control_init_val(struct usb_mixer_interface * mixer,struct snd_kcontrol * kctl)730*4882a593Smuzhiyun static int snd_ni_control_init_val(struct usb_mixer_interface *mixer,
731*4882a593Smuzhiyun struct snd_kcontrol *kctl)
732*4882a593Smuzhiyun {
733*4882a593Smuzhiyun struct usb_device *dev = mixer->chip->dev;
734*4882a593Smuzhiyun unsigned int pval = kctl->private_value;
735*4882a593Smuzhiyun u8 value;
736*4882a593Smuzhiyun int err;
737*4882a593Smuzhiyun
738*4882a593Smuzhiyun err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0),
739*4882a593Smuzhiyun (pval >> 16) & 0xff,
740*4882a593Smuzhiyun USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
741*4882a593Smuzhiyun 0, pval & 0xffff, &value, 1);
742*4882a593Smuzhiyun if (err < 0) {
743*4882a593Smuzhiyun dev_err(&dev->dev,
744*4882a593Smuzhiyun "unable to issue vendor read request (ret = %d)", err);
745*4882a593Smuzhiyun return err;
746*4882a593Smuzhiyun }
747*4882a593Smuzhiyun
748*4882a593Smuzhiyun kctl->private_value |= ((unsigned int)value << 24);
749*4882a593Smuzhiyun return 0;
750*4882a593Smuzhiyun }
751*4882a593Smuzhiyun
snd_nativeinstruments_control_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)752*4882a593Smuzhiyun static int snd_nativeinstruments_control_get(struct snd_kcontrol *kcontrol,
753*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
754*4882a593Smuzhiyun {
755*4882a593Smuzhiyun ucontrol->value.integer.value[0] = kcontrol->private_value >> 24;
756*4882a593Smuzhiyun return 0;
757*4882a593Smuzhiyun }
758*4882a593Smuzhiyun
snd_ni_update_cur_val(struct usb_mixer_elem_list * list)759*4882a593Smuzhiyun static int snd_ni_update_cur_val(struct usb_mixer_elem_list *list)
760*4882a593Smuzhiyun {
761*4882a593Smuzhiyun struct snd_usb_audio *chip = list->mixer->chip;
762*4882a593Smuzhiyun unsigned int pval = list->kctl->private_value;
763*4882a593Smuzhiyun int err;
764*4882a593Smuzhiyun
765*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
766*4882a593Smuzhiyun if (err < 0)
767*4882a593Smuzhiyun return err;
768*4882a593Smuzhiyun err = usb_control_msg(chip->dev, usb_sndctrlpipe(chip->dev, 0),
769*4882a593Smuzhiyun (pval >> 16) & 0xff,
770*4882a593Smuzhiyun USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_OUT,
771*4882a593Smuzhiyun pval >> 24, pval & 0xffff, NULL, 0, 1000);
772*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
773*4882a593Smuzhiyun return err;
774*4882a593Smuzhiyun }
775*4882a593Smuzhiyun
snd_nativeinstruments_control_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)776*4882a593Smuzhiyun static int snd_nativeinstruments_control_put(struct snd_kcontrol *kcontrol,
777*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
778*4882a593Smuzhiyun {
779*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
780*4882a593Smuzhiyun u8 oldval = (kcontrol->private_value >> 24) & 0xff;
781*4882a593Smuzhiyun u8 newval = ucontrol->value.integer.value[0];
782*4882a593Smuzhiyun int err;
783*4882a593Smuzhiyun
784*4882a593Smuzhiyun if (oldval == newval)
785*4882a593Smuzhiyun return 0;
786*4882a593Smuzhiyun
787*4882a593Smuzhiyun kcontrol->private_value &= ~(0xff << 24);
788*4882a593Smuzhiyun kcontrol->private_value |= (unsigned int)newval << 24;
789*4882a593Smuzhiyun err = snd_ni_update_cur_val(list);
790*4882a593Smuzhiyun return err < 0 ? err : 1;
791*4882a593Smuzhiyun }
792*4882a593Smuzhiyun
793*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_nativeinstruments_ta6_mixers[] = {
794*4882a593Smuzhiyun {
795*4882a593Smuzhiyun .name = "Direct Thru Channel A",
796*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x01, 0x03),
797*4882a593Smuzhiyun },
798*4882a593Smuzhiyun {
799*4882a593Smuzhiyun .name = "Direct Thru Channel B",
800*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x01, 0x05),
801*4882a593Smuzhiyun },
802*4882a593Smuzhiyun {
803*4882a593Smuzhiyun .name = "Phono Input Channel A",
804*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x02, 0x03),
805*4882a593Smuzhiyun },
806*4882a593Smuzhiyun {
807*4882a593Smuzhiyun .name = "Phono Input Channel B",
808*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x02, 0x05),
809*4882a593Smuzhiyun },
810*4882a593Smuzhiyun };
811*4882a593Smuzhiyun
812*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_nativeinstruments_ta10_mixers[] = {
813*4882a593Smuzhiyun {
814*4882a593Smuzhiyun .name = "Direct Thru Channel A",
815*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x01, 0x03),
816*4882a593Smuzhiyun },
817*4882a593Smuzhiyun {
818*4882a593Smuzhiyun .name = "Direct Thru Channel B",
819*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x01, 0x05),
820*4882a593Smuzhiyun },
821*4882a593Smuzhiyun {
822*4882a593Smuzhiyun .name = "Direct Thru Channel C",
823*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x01, 0x07),
824*4882a593Smuzhiyun },
825*4882a593Smuzhiyun {
826*4882a593Smuzhiyun .name = "Direct Thru Channel D",
827*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x01, 0x09),
828*4882a593Smuzhiyun },
829*4882a593Smuzhiyun {
830*4882a593Smuzhiyun .name = "Phono Input Channel A",
831*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x02, 0x03),
832*4882a593Smuzhiyun },
833*4882a593Smuzhiyun {
834*4882a593Smuzhiyun .name = "Phono Input Channel B",
835*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x02, 0x05),
836*4882a593Smuzhiyun },
837*4882a593Smuzhiyun {
838*4882a593Smuzhiyun .name = "Phono Input Channel C",
839*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x02, 0x07),
840*4882a593Smuzhiyun },
841*4882a593Smuzhiyun {
842*4882a593Smuzhiyun .name = "Phono Input Channel D",
843*4882a593Smuzhiyun .private_value = _MAKE_NI_CONTROL(0x02, 0x09),
844*4882a593Smuzhiyun },
845*4882a593Smuzhiyun };
846*4882a593Smuzhiyun
snd_nativeinstruments_create_mixer(struct usb_mixer_interface * mixer,const struct snd_kcontrol_new * kc,unsigned int count)847*4882a593Smuzhiyun static int snd_nativeinstruments_create_mixer(struct usb_mixer_interface *mixer,
848*4882a593Smuzhiyun const struct snd_kcontrol_new *kc,
849*4882a593Smuzhiyun unsigned int count)
850*4882a593Smuzhiyun {
851*4882a593Smuzhiyun int i, err = 0;
852*4882a593Smuzhiyun struct snd_kcontrol_new template = {
853*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
854*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
855*4882a593Smuzhiyun .get = snd_nativeinstruments_control_get,
856*4882a593Smuzhiyun .put = snd_nativeinstruments_control_put,
857*4882a593Smuzhiyun .info = snd_ctl_boolean_mono_info,
858*4882a593Smuzhiyun };
859*4882a593Smuzhiyun
860*4882a593Smuzhiyun for (i = 0; i < count; i++) {
861*4882a593Smuzhiyun struct usb_mixer_elem_list *list;
862*4882a593Smuzhiyun
863*4882a593Smuzhiyun template.name = kc[i].name;
864*4882a593Smuzhiyun template.private_value = kc[i].private_value;
865*4882a593Smuzhiyun
866*4882a593Smuzhiyun err = add_single_ctl_with_resume(mixer, 0,
867*4882a593Smuzhiyun snd_ni_update_cur_val,
868*4882a593Smuzhiyun &template, &list);
869*4882a593Smuzhiyun if (err < 0)
870*4882a593Smuzhiyun break;
871*4882a593Smuzhiyun snd_ni_control_init_val(mixer, list->kctl);
872*4882a593Smuzhiyun }
873*4882a593Smuzhiyun
874*4882a593Smuzhiyun return err;
875*4882a593Smuzhiyun }
876*4882a593Smuzhiyun
877*4882a593Smuzhiyun /* M-Audio FastTrack Ultra quirks */
878*4882a593Smuzhiyun /* FTU Effect switch (also used by C400/C600) */
snd_ftu_eff_switch_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)879*4882a593Smuzhiyun static int snd_ftu_eff_switch_info(struct snd_kcontrol *kcontrol,
880*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
881*4882a593Smuzhiyun {
882*4882a593Smuzhiyun static const char *const texts[8] = {
883*4882a593Smuzhiyun "Room 1", "Room 2", "Room 3", "Hall 1",
884*4882a593Smuzhiyun "Hall 2", "Plate", "Delay", "Echo"
885*4882a593Smuzhiyun };
886*4882a593Smuzhiyun
887*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1, ARRAY_SIZE(texts), texts);
888*4882a593Smuzhiyun }
889*4882a593Smuzhiyun
snd_ftu_eff_switch_init(struct usb_mixer_interface * mixer,struct snd_kcontrol * kctl)890*4882a593Smuzhiyun static int snd_ftu_eff_switch_init(struct usb_mixer_interface *mixer,
891*4882a593Smuzhiyun struct snd_kcontrol *kctl)
892*4882a593Smuzhiyun {
893*4882a593Smuzhiyun struct usb_device *dev = mixer->chip->dev;
894*4882a593Smuzhiyun unsigned int pval = kctl->private_value;
895*4882a593Smuzhiyun int err;
896*4882a593Smuzhiyun unsigned char value[2];
897*4882a593Smuzhiyun
898*4882a593Smuzhiyun value[0] = 0x00;
899*4882a593Smuzhiyun value[1] = 0x00;
900*4882a593Smuzhiyun
901*4882a593Smuzhiyun err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC_GET_CUR,
902*4882a593Smuzhiyun USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
903*4882a593Smuzhiyun pval & 0xff00,
904*4882a593Smuzhiyun snd_usb_ctrl_intf(mixer->chip) | ((pval & 0xff) << 8),
905*4882a593Smuzhiyun value, 2);
906*4882a593Smuzhiyun if (err < 0)
907*4882a593Smuzhiyun return err;
908*4882a593Smuzhiyun
909*4882a593Smuzhiyun kctl->private_value |= (unsigned int)value[0] << 24;
910*4882a593Smuzhiyun return 0;
911*4882a593Smuzhiyun }
912*4882a593Smuzhiyun
snd_ftu_eff_switch_get(struct snd_kcontrol * kctl,struct snd_ctl_elem_value * ucontrol)913*4882a593Smuzhiyun static int snd_ftu_eff_switch_get(struct snd_kcontrol *kctl,
914*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
915*4882a593Smuzhiyun {
916*4882a593Smuzhiyun ucontrol->value.enumerated.item[0] = kctl->private_value >> 24;
917*4882a593Smuzhiyun return 0;
918*4882a593Smuzhiyun }
919*4882a593Smuzhiyun
snd_ftu_eff_switch_update(struct usb_mixer_elem_list * list)920*4882a593Smuzhiyun static int snd_ftu_eff_switch_update(struct usb_mixer_elem_list *list)
921*4882a593Smuzhiyun {
922*4882a593Smuzhiyun struct snd_usb_audio *chip = list->mixer->chip;
923*4882a593Smuzhiyun unsigned int pval = list->kctl->private_value;
924*4882a593Smuzhiyun unsigned char value[2];
925*4882a593Smuzhiyun int err;
926*4882a593Smuzhiyun
927*4882a593Smuzhiyun value[0] = pval >> 24;
928*4882a593Smuzhiyun value[1] = 0;
929*4882a593Smuzhiyun
930*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
931*4882a593Smuzhiyun if (err < 0)
932*4882a593Smuzhiyun return err;
933*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
934*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0),
935*4882a593Smuzhiyun UAC_SET_CUR,
936*4882a593Smuzhiyun USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_OUT,
937*4882a593Smuzhiyun pval & 0xff00,
938*4882a593Smuzhiyun snd_usb_ctrl_intf(chip) | ((pval & 0xff) << 8),
939*4882a593Smuzhiyun value, 2);
940*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
941*4882a593Smuzhiyun return err;
942*4882a593Smuzhiyun }
943*4882a593Smuzhiyun
snd_ftu_eff_switch_put(struct snd_kcontrol * kctl,struct snd_ctl_elem_value * ucontrol)944*4882a593Smuzhiyun static int snd_ftu_eff_switch_put(struct snd_kcontrol *kctl,
945*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
946*4882a593Smuzhiyun {
947*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kctl);
948*4882a593Smuzhiyun unsigned int pval = list->kctl->private_value;
949*4882a593Smuzhiyun int cur_val, err, new_val;
950*4882a593Smuzhiyun
951*4882a593Smuzhiyun cur_val = pval >> 24;
952*4882a593Smuzhiyun new_val = ucontrol->value.enumerated.item[0];
953*4882a593Smuzhiyun if (cur_val == new_val)
954*4882a593Smuzhiyun return 0;
955*4882a593Smuzhiyun
956*4882a593Smuzhiyun kctl->private_value &= ~(0xff << 24);
957*4882a593Smuzhiyun kctl->private_value |= new_val << 24;
958*4882a593Smuzhiyun err = snd_ftu_eff_switch_update(list);
959*4882a593Smuzhiyun return err < 0 ? err : 1;
960*4882a593Smuzhiyun }
961*4882a593Smuzhiyun
snd_ftu_create_effect_switch(struct usb_mixer_interface * mixer,int validx,int bUnitID)962*4882a593Smuzhiyun static int snd_ftu_create_effect_switch(struct usb_mixer_interface *mixer,
963*4882a593Smuzhiyun int validx, int bUnitID)
964*4882a593Smuzhiyun {
965*4882a593Smuzhiyun static struct snd_kcontrol_new template = {
966*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
967*4882a593Smuzhiyun .name = "Effect Program Switch",
968*4882a593Smuzhiyun .index = 0,
969*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
970*4882a593Smuzhiyun .info = snd_ftu_eff_switch_info,
971*4882a593Smuzhiyun .get = snd_ftu_eff_switch_get,
972*4882a593Smuzhiyun .put = snd_ftu_eff_switch_put
973*4882a593Smuzhiyun };
974*4882a593Smuzhiyun struct usb_mixer_elem_list *list;
975*4882a593Smuzhiyun int err;
976*4882a593Smuzhiyun
977*4882a593Smuzhiyun err = add_single_ctl_with_resume(mixer, bUnitID,
978*4882a593Smuzhiyun snd_ftu_eff_switch_update,
979*4882a593Smuzhiyun &template, &list);
980*4882a593Smuzhiyun if (err < 0)
981*4882a593Smuzhiyun return err;
982*4882a593Smuzhiyun list->kctl->private_value = (validx << 8) | bUnitID;
983*4882a593Smuzhiyun snd_ftu_eff_switch_init(mixer, list->kctl);
984*4882a593Smuzhiyun return 0;
985*4882a593Smuzhiyun }
986*4882a593Smuzhiyun
987*4882a593Smuzhiyun /* Create volume controls for FTU devices*/
snd_ftu_create_volume_ctls(struct usb_mixer_interface * mixer)988*4882a593Smuzhiyun static int snd_ftu_create_volume_ctls(struct usb_mixer_interface *mixer)
989*4882a593Smuzhiyun {
990*4882a593Smuzhiyun char name[64];
991*4882a593Smuzhiyun unsigned int control, cmask;
992*4882a593Smuzhiyun int in, out, err;
993*4882a593Smuzhiyun
994*4882a593Smuzhiyun const unsigned int id = 5;
995*4882a593Smuzhiyun const int val_type = USB_MIXER_S16;
996*4882a593Smuzhiyun
997*4882a593Smuzhiyun for (out = 0; out < 8; out++) {
998*4882a593Smuzhiyun control = out + 1;
999*4882a593Smuzhiyun for (in = 0; in < 8; in++) {
1000*4882a593Smuzhiyun cmask = 1 << in;
1001*4882a593Smuzhiyun snprintf(name, sizeof(name),
1002*4882a593Smuzhiyun "AIn%d - Out%d Capture Volume",
1003*4882a593Smuzhiyun in + 1, out + 1);
1004*4882a593Smuzhiyun err = snd_create_std_mono_ctl(mixer, id, control,
1005*4882a593Smuzhiyun cmask, val_type, name,
1006*4882a593Smuzhiyun &snd_usb_mixer_vol_tlv);
1007*4882a593Smuzhiyun if (err < 0)
1008*4882a593Smuzhiyun return err;
1009*4882a593Smuzhiyun }
1010*4882a593Smuzhiyun for (in = 8; in < 16; in++) {
1011*4882a593Smuzhiyun cmask = 1 << in;
1012*4882a593Smuzhiyun snprintf(name, sizeof(name),
1013*4882a593Smuzhiyun "DIn%d - Out%d Playback Volume",
1014*4882a593Smuzhiyun in - 7, out + 1);
1015*4882a593Smuzhiyun err = snd_create_std_mono_ctl(mixer, id, control,
1016*4882a593Smuzhiyun cmask, val_type, name,
1017*4882a593Smuzhiyun &snd_usb_mixer_vol_tlv);
1018*4882a593Smuzhiyun if (err < 0)
1019*4882a593Smuzhiyun return err;
1020*4882a593Smuzhiyun }
1021*4882a593Smuzhiyun }
1022*4882a593Smuzhiyun
1023*4882a593Smuzhiyun return 0;
1024*4882a593Smuzhiyun }
1025*4882a593Smuzhiyun
1026*4882a593Smuzhiyun /* This control needs a volume quirk, see mixer.c */
snd_ftu_create_effect_volume_ctl(struct usb_mixer_interface * mixer)1027*4882a593Smuzhiyun static int snd_ftu_create_effect_volume_ctl(struct usb_mixer_interface *mixer)
1028*4882a593Smuzhiyun {
1029*4882a593Smuzhiyun static const char name[] = "Effect Volume";
1030*4882a593Smuzhiyun const unsigned int id = 6;
1031*4882a593Smuzhiyun const int val_type = USB_MIXER_U8;
1032*4882a593Smuzhiyun const unsigned int control = 2;
1033*4882a593Smuzhiyun const unsigned int cmask = 0;
1034*4882a593Smuzhiyun
1035*4882a593Smuzhiyun return snd_create_std_mono_ctl(mixer, id, control, cmask, val_type,
1036*4882a593Smuzhiyun name, snd_usb_mixer_vol_tlv);
1037*4882a593Smuzhiyun }
1038*4882a593Smuzhiyun
1039*4882a593Smuzhiyun /* This control needs a volume quirk, see mixer.c */
snd_ftu_create_effect_duration_ctl(struct usb_mixer_interface * mixer)1040*4882a593Smuzhiyun static int snd_ftu_create_effect_duration_ctl(struct usb_mixer_interface *mixer)
1041*4882a593Smuzhiyun {
1042*4882a593Smuzhiyun static const char name[] = "Effect Duration";
1043*4882a593Smuzhiyun const unsigned int id = 6;
1044*4882a593Smuzhiyun const int val_type = USB_MIXER_S16;
1045*4882a593Smuzhiyun const unsigned int control = 3;
1046*4882a593Smuzhiyun const unsigned int cmask = 0;
1047*4882a593Smuzhiyun
1048*4882a593Smuzhiyun return snd_create_std_mono_ctl(mixer, id, control, cmask, val_type,
1049*4882a593Smuzhiyun name, snd_usb_mixer_vol_tlv);
1050*4882a593Smuzhiyun }
1051*4882a593Smuzhiyun
1052*4882a593Smuzhiyun /* This control needs a volume quirk, see mixer.c */
snd_ftu_create_effect_feedback_ctl(struct usb_mixer_interface * mixer)1053*4882a593Smuzhiyun static int snd_ftu_create_effect_feedback_ctl(struct usb_mixer_interface *mixer)
1054*4882a593Smuzhiyun {
1055*4882a593Smuzhiyun static const char name[] = "Effect Feedback Volume";
1056*4882a593Smuzhiyun const unsigned int id = 6;
1057*4882a593Smuzhiyun const int val_type = USB_MIXER_U8;
1058*4882a593Smuzhiyun const unsigned int control = 4;
1059*4882a593Smuzhiyun const unsigned int cmask = 0;
1060*4882a593Smuzhiyun
1061*4882a593Smuzhiyun return snd_create_std_mono_ctl(mixer, id, control, cmask, val_type,
1062*4882a593Smuzhiyun name, NULL);
1063*4882a593Smuzhiyun }
1064*4882a593Smuzhiyun
snd_ftu_create_effect_return_ctls(struct usb_mixer_interface * mixer)1065*4882a593Smuzhiyun static int snd_ftu_create_effect_return_ctls(struct usb_mixer_interface *mixer)
1066*4882a593Smuzhiyun {
1067*4882a593Smuzhiyun unsigned int cmask;
1068*4882a593Smuzhiyun int err, ch;
1069*4882a593Smuzhiyun char name[48];
1070*4882a593Smuzhiyun
1071*4882a593Smuzhiyun const unsigned int id = 7;
1072*4882a593Smuzhiyun const int val_type = USB_MIXER_S16;
1073*4882a593Smuzhiyun const unsigned int control = 7;
1074*4882a593Smuzhiyun
1075*4882a593Smuzhiyun for (ch = 0; ch < 4; ++ch) {
1076*4882a593Smuzhiyun cmask = 1 << ch;
1077*4882a593Smuzhiyun snprintf(name, sizeof(name),
1078*4882a593Smuzhiyun "Effect Return %d Volume", ch + 1);
1079*4882a593Smuzhiyun err = snd_create_std_mono_ctl(mixer, id, control,
1080*4882a593Smuzhiyun cmask, val_type, name,
1081*4882a593Smuzhiyun snd_usb_mixer_vol_tlv);
1082*4882a593Smuzhiyun if (err < 0)
1083*4882a593Smuzhiyun return err;
1084*4882a593Smuzhiyun }
1085*4882a593Smuzhiyun
1086*4882a593Smuzhiyun return 0;
1087*4882a593Smuzhiyun }
1088*4882a593Smuzhiyun
snd_ftu_create_effect_send_ctls(struct usb_mixer_interface * mixer)1089*4882a593Smuzhiyun static int snd_ftu_create_effect_send_ctls(struct usb_mixer_interface *mixer)
1090*4882a593Smuzhiyun {
1091*4882a593Smuzhiyun unsigned int cmask;
1092*4882a593Smuzhiyun int err, ch;
1093*4882a593Smuzhiyun char name[48];
1094*4882a593Smuzhiyun
1095*4882a593Smuzhiyun const unsigned int id = 5;
1096*4882a593Smuzhiyun const int val_type = USB_MIXER_S16;
1097*4882a593Smuzhiyun const unsigned int control = 9;
1098*4882a593Smuzhiyun
1099*4882a593Smuzhiyun for (ch = 0; ch < 8; ++ch) {
1100*4882a593Smuzhiyun cmask = 1 << ch;
1101*4882a593Smuzhiyun snprintf(name, sizeof(name),
1102*4882a593Smuzhiyun "Effect Send AIn%d Volume", ch + 1);
1103*4882a593Smuzhiyun err = snd_create_std_mono_ctl(mixer, id, control, cmask,
1104*4882a593Smuzhiyun val_type, name,
1105*4882a593Smuzhiyun snd_usb_mixer_vol_tlv);
1106*4882a593Smuzhiyun if (err < 0)
1107*4882a593Smuzhiyun return err;
1108*4882a593Smuzhiyun }
1109*4882a593Smuzhiyun for (ch = 8; ch < 16; ++ch) {
1110*4882a593Smuzhiyun cmask = 1 << ch;
1111*4882a593Smuzhiyun snprintf(name, sizeof(name),
1112*4882a593Smuzhiyun "Effect Send DIn%d Volume", ch - 7);
1113*4882a593Smuzhiyun err = snd_create_std_mono_ctl(mixer, id, control, cmask,
1114*4882a593Smuzhiyun val_type, name,
1115*4882a593Smuzhiyun snd_usb_mixer_vol_tlv);
1116*4882a593Smuzhiyun if (err < 0)
1117*4882a593Smuzhiyun return err;
1118*4882a593Smuzhiyun }
1119*4882a593Smuzhiyun return 0;
1120*4882a593Smuzhiyun }
1121*4882a593Smuzhiyun
snd_ftu_create_mixer(struct usb_mixer_interface * mixer)1122*4882a593Smuzhiyun static int snd_ftu_create_mixer(struct usb_mixer_interface *mixer)
1123*4882a593Smuzhiyun {
1124*4882a593Smuzhiyun int err;
1125*4882a593Smuzhiyun
1126*4882a593Smuzhiyun err = snd_ftu_create_volume_ctls(mixer);
1127*4882a593Smuzhiyun if (err < 0)
1128*4882a593Smuzhiyun return err;
1129*4882a593Smuzhiyun
1130*4882a593Smuzhiyun err = snd_ftu_create_effect_switch(mixer, 1, 6);
1131*4882a593Smuzhiyun if (err < 0)
1132*4882a593Smuzhiyun return err;
1133*4882a593Smuzhiyun
1134*4882a593Smuzhiyun err = snd_ftu_create_effect_volume_ctl(mixer);
1135*4882a593Smuzhiyun if (err < 0)
1136*4882a593Smuzhiyun return err;
1137*4882a593Smuzhiyun
1138*4882a593Smuzhiyun err = snd_ftu_create_effect_duration_ctl(mixer);
1139*4882a593Smuzhiyun if (err < 0)
1140*4882a593Smuzhiyun return err;
1141*4882a593Smuzhiyun
1142*4882a593Smuzhiyun err = snd_ftu_create_effect_feedback_ctl(mixer);
1143*4882a593Smuzhiyun if (err < 0)
1144*4882a593Smuzhiyun return err;
1145*4882a593Smuzhiyun
1146*4882a593Smuzhiyun err = snd_ftu_create_effect_return_ctls(mixer);
1147*4882a593Smuzhiyun if (err < 0)
1148*4882a593Smuzhiyun return err;
1149*4882a593Smuzhiyun
1150*4882a593Smuzhiyun err = snd_ftu_create_effect_send_ctls(mixer);
1151*4882a593Smuzhiyun if (err < 0)
1152*4882a593Smuzhiyun return err;
1153*4882a593Smuzhiyun
1154*4882a593Smuzhiyun return 0;
1155*4882a593Smuzhiyun }
1156*4882a593Smuzhiyun
snd_emuusb_set_samplerate(struct snd_usb_audio * chip,unsigned char samplerate_id)1157*4882a593Smuzhiyun void snd_emuusb_set_samplerate(struct snd_usb_audio *chip,
1158*4882a593Smuzhiyun unsigned char samplerate_id)
1159*4882a593Smuzhiyun {
1160*4882a593Smuzhiyun struct usb_mixer_interface *mixer;
1161*4882a593Smuzhiyun struct usb_mixer_elem_info *cval;
1162*4882a593Smuzhiyun int unitid = 12; /* SampleRate ExtensionUnit ID */
1163*4882a593Smuzhiyun
1164*4882a593Smuzhiyun list_for_each_entry(mixer, &chip->mixer_list, list) {
1165*4882a593Smuzhiyun if (mixer->id_elems[unitid]) {
1166*4882a593Smuzhiyun cval = mixer_elem_list_to_info(mixer->id_elems[unitid]);
1167*4882a593Smuzhiyun snd_usb_mixer_set_ctl_value(cval, UAC_SET_CUR,
1168*4882a593Smuzhiyun cval->control << 8,
1169*4882a593Smuzhiyun samplerate_id);
1170*4882a593Smuzhiyun snd_usb_mixer_notify_id(mixer, unitid);
1171*4882a593Smuzhiyun break;
1172*4882a593Smuzhiyun }
1173*4882a593Smuzhiyun }
1174*4882a593Smuzhiyun }
1175*4882a593Smuzhiyun
1176*4882a593Smuzhiyun /* M-Audio Fast Track C400/C600 */
1177*4882a593Smuzhiyun /* C400/C600 volume controls, this control needs a volume quirk, see mixer.c */
snd_c400_create_vol_ctls(struct usb_mixer_interface * mixer)1178*4882a593Smuzhiyun static int snd_c400_create_vol_ctls(struct usb_mixer_interface *mixer)
1179*4882a593Smuzhiyun {
1180*4882a593Smuzhiyun char name[64];
1181*4882a593Smuzhiyun unsigned int cmask, offset;
1182*4882a593Smuzhiyun int out, chan, err;
1183*4882a593Smuzhiyun int num_outs = 0;
1184*4882a593Smuzhiyun int num_ins = 0;
1185*4882a593Smuzhiyun
1186*4882a593Smuzhiyun const unsigned int id = 0x40;
1187*4882a593Smuzhiyun const int val_type = USB_MIXER_S16;
1188*4882a593Smuzhiyun const int control = 1;
1189*4882a593Smuzhiyun
1190*4882a593Smuzhiyun switch (mixer->chip->usb_id) {
1191*4882a593Smuzhiyun case USB_ID(0x0763, 0x2030):
1192*4882a593Smuzhiyun num_outs = 6;
1193*4882a593Smuzhiyun num_ins = 4;
1194*4882a593Smuzhiyun break;
1195*4882a593Smuzhiyun case USB_ID(0x0763, 0x2031):
1196*4882a593Smuzhiyun num_outs = 8;
1197*4882a593Smuzhiyun num_ins = 6;
1198*4882a593Smuzhiyun break;
1199*4882a593Smuzhiyun }
1200*4882a593Smuzhiyun
1201*4882a593Smuzhiyun for (chan = 0; chan < num_outs + num_ins; chan++) {
1202*4882a593Smuzhiyun for (out = 0; out < num_outs; out++) {
1203*4882a593Smuzhiyun if (chan < num_outs) {
1204*4882a593Smuzhiyun snprintf(name, sizeof(name),
1205*4882a593Smuzhiyun "PCM%d-Out%d Playback Volume",
1206*4882a593Smuzhiyun chan + 1, out + 1);
1207*4882a593Smuzhiyun } else {
1208*4882a593Smuzhiyun snprintf(name, sizeof(name),
1209*4882a593Smuzhiyun "In%d-Out%d Playback Volume",
1210*4882a593Smuzhiyun chan - num_outs + 1, out + 1);
1211*4882a593Smuzhiyun }
1212*4882a593Smuzhiyun
1213*4882a593Smuzhiyun cmask = (out == 0) ? 0 : 1 << (out - 1);
1214*4882a593Smuzhiyun offset = chan * num_outs;
1215*4882a593Smuzhiyun err = snd_create_std_mono_ctl_offset(mixer, id, control,
1216*4882a593Smuzhiyun cmask, val_type, offset, name,
1217*4882a593Smuzhiyun &snd_usb_mixer_vol_tlv);
1218*4882a593Smuzhiyun if (err < 0)
1219*4882a593Smuzhiyun return err;
1220*4882a593Smuzhiyun }
1221*4882a593Smuzhiyun }
1222*4882a593Smuzhiyun
1223*4882a593Smuzhiyun return 0;
1224*4882a593Smuzhiyun }
1225*4882a593Smuzhiyun
1226*4882a593Smuzhiyun /* This control needs a volume quirk, see mixer.c */
snd_c400_create_effect_volume_ctl(struct usb_mixer_interface * mixer)1227*4882a593Smuzhiyun static int snd_c400_create_effect_volume_ctl(struct usb_mixer_interface *mixer)
1228*4882a593Smuzhiyun {
1229*4882a593Smuzhiyun static const char name[] = "Effect Volume";
1230*4882a593Smuzhiyun const unsigned int id = 0x43;
1231*4882a593Smuzhiyun const int val_type = USB_MIXER_U8;
1232*4882a593Smuzhiyun const unsigned int control = 3;
1233*4882a593Smuzhiyun const unsigned int cmask = 0;
1234*4882a593Smuzhiyun
1235*4882a593Smuzhiyun return snd_create_std_mono_ctl(mixer, id, control, cmask, val_type,
1236*4882a593Smuzhiyun name, snd_usb_mixer_vol_tlv);
1237*4882a593Smuzhiyun }
1238*4882a593Smuzhiyun
1239*4882a593Smuzhiyun /* This control needs a volume quirk, see mixer.c */
snd_c400_create_effect_duration_ctl(struct usb_mixer_interface * mixer)1240*4882a593Smuzhiyun static int snd_c400_create_effect_duration_ctl(struct usb_mixer_interface *mixer)
1241*4882a593Smuzhiyun {
1242*4882a593Smuzhiyun static const char name[] = "Effect Duration";
1243*4882a593Smuzhiyun const unsigned int id = 0x43;
1244*4882a593Smuzhiyun const int val_type = USB_MIXER_S16;
1245*4882a593Smuzhiyun const unsigned int control = 4;
1246*4882a593Smuzhiyun const unsigned int cmask = 0;
1247*4882a593Smuzhiyun
1248*4882a593Smuzhiyun return snd_create_std_mono_ctl(mixer, id, control, cmask, val_type,
1249*4882a593Smuzhiyun name, snd_usb_mixer_vol_tlv);
1250*4882a593Smuzhiyun }
1251*4882a593Smuzhiyun
1252*4882a593Smuzhiyun /* This control needs a volume quirk, see mixer.c */
snd_c400_create_effect_feedback_ctl(struct usb_mixer_interface * mixer)1253*4882a593Smuzhiyun static int snd_c400_create_effect_feedback_ctl(struct usb_mixer_interface *mixer)
1254*4882a593Smuzhiyun {
1255*4882a593Smuzhiyun static const char name[] = "Effect Feedback Volume";
1256*4882a593Smuzhiyun const unsigned int id = 0x43;
1257*4882a593Smuzhiyun const int val_type = USB_MIXER_U8;
1258*4882a593Smuzhiyun const unsigned int control = 5;
1259*4882a593Smuzhiyun const unsigned int cmask = 0;
1260*4882a593Smuzhiyun
1261*4882a593Smuzhiyun return snd_create_std_mono_ctl(mixer, id, control, cmask, val_type,
1262*4882a593Smuzhiyun name, NULL);
1263*4882a593Smuzhiyun }
1264*4882a593Smuzhiyun
snd_c400_create_effect_vol_ctls(struct usb_mixer_interface * mixer)1265*4882a593Smuzhiyun static int snd_c400_create_effect_vol_ctls(struct usb_mixer_interface *mixer)
1266*4882a593Smuzhiyun {
1267*4882a593Smuzhiyun char name[64];
1268*4882a593Smuzhiyun unsigned int cmask;
1269*4882a593Smuzhiyun int chan, err;
1270*4882a593Smuzhiyun int num_outs = 0;
1271*4882a593Smuzhiyun int num_ins = 0;
1272*4882a593Smuzhiyun
1273*4882a593Smuzhiyun const unsigned int id = 0x42;
1274*4882a593Smuzhiyun const int val_type = USB_MIXER_S16;
1275*4882a593Smuzhiyun const int control = 1;
1276*4882a593Smuzhiyun
1277*4882a593Smuzhiyun switch (mixer->chip->usb_id) {
1278*4882a593Smuzhiyun case USB_ID(0x0763, 0x2030):
1279*4882a593Smuzhiyun num_outs = 6;
1280*4882a593Smuzhiyun num_ins = 4;
1281*4882a593Smuzhiyun break;
1282*4882a593Smuzhiyun case USB_ID(0x0763, 0x2031):
1283*4882a593Smuzhiyun num_outs = 8;
1284*4882a593Smuzhiyun num_ins = 6;
1285*4882a593Smuzhiyun break;
1286*4882a593Smuzhiyun }
1287*4882a593Smuzhiyun
1288*4882a593Smuzhiyun for (chan = 0; chan < num_outs + num_ins; chan++) {
1289*4882a593Smuzhiyun if (chan < num_outs) {
1290*4882a593Smuzhiyun snprintf(name, sizeof(name),
1291*4882a593Smuzhiyun "Effect Send DOut%d",
1292*4882a593Smuzhiyun chan + 1);
1293*4882a593Smuzhiyun } else {
1294*4882a593Smuzhiyun snprintf(name, sizeof(name),
1295*4882a593Smuzhiyun "Effect Send AIn%d",
1296*4882a593Smuzhiyun chan - num_outs + 1);
1297*4882a593Smuzhiyun }
1298*4882a593Smuzhiyun
1299*4882a593Smuzhiyun cmask = (chan == 0) ? 0 : 1 << (chan - 1);
1300*4882a593Smuzhiyun err = snd_create_std_mono_ctl(mixer, id, control,
1301*4882a593Smuzhiyun cmask, val_type, name,
1302*4882a593Smuzhiyun &snd_usb_mixer_vol_tlv);
1303*4882a593Smuzhiyun if (err < 0)
1304*4882a593Smuzhiyun return err;
1305*4882a593Smuzhiyun }
1306*4882a593Smuzhiyun
1307*4882a593Smuzhiyun return 0;
1308*4882a593Smuzhiyun }
1309*4882a593Smuzhiyun
snd_c400_create_effect_ret_vol_ctls(struct usb_mixer_interface * mixer)1310*4882a593Smuzhiyun static int snd_c400_create_effect_ret_vol_ctls(struct usb_mixer_interface *mixer)
1311*4882a593Smuzhiyun {
1312*4882a593Smuzhiyun char name[64];
1313*4882a593Smuzhiyun unsigned int cmask;
1314*4882a593Smuzhiyun int chan, err;
1315*4882a593Smuzhiyun int num_outs = 0;
1316*4882a593Smuzhiyun int offset = 0;
1317*4882a593Smuzhiyun
1318*4882a593Smuzhiyun const unsigned int id = 0x40;
1319*4882a593Smuzhiyun const int val_type = USB_MIXER_S16;
1320*4882a593Smuzhiyun const int control = 1;
1321*4882a593Smuzhiyun
1322*4882a593Smuzhiyun switch (mixer->chip->usb_id) {
1323*4882a593Smuzhiyun case USB_ID(0x0763, 0x2030):
1324*4882a593Smuzhiyun num_outs = 6;
1325*4882a593Smuzhiyun offset = 0x3c;
1326*4882a593Smuzhiyun /* { 0x3c, 0x43, 0x3e, 0x45, 0x40, 0x47 } */
1327*4882a593Smuzhiyun break;
1328*4882a593Smuzhiyun case USB_ID(0x0763, 0x2031):
1329*4882a593Smuzhiyun num_outs = 8;
1330*4882a593Smuzhiyun offset = 0x70;
1331*4882a593Smuzhiyun /* { 0x70, 0x79, 0x72, 0x7b, 0x74, 0x7d, 0x76, 0x7f } */
1332*4882a593Smuzhiyun break;
1333*4882a593Smuzhiyun }
1334*4882a593Smuzhiyun
1335*4882a593Smuzhiyun for (chan = 0; chan < num_outs; chan++) {
1336*4882a593Smuzhiyun snprintf(name, sizeof(name),
1337*4882a593Smuzhiyun "Effect Return %d",
1338*4882a593Smuzhiyun chan + 1);
1339*4882a593Smuzhiyun
1340*4882a593Smuzhiyun cmask = (chan == 0) ? 0 :
1341*4882a593Smuzhiyun 1 << (chan + (chan % 2) * num_outs - 1);
1342*4882a593Smuzhiyun err = snd_create_std_mono_ctl_offset(mixer, id, control,
1343*4882a593Smuzhiyun cmask, val_type, offset, name,
1344*4882a593Smuzhiyun &snd_usb_mixer_vol_tlv);
1345*4882a593Smuzhiyun if (err < 0)
1346*4882a593Smuzhiyun return err;
1347*4882a593Smuzhiyun }
1348*4882a593Smuzhiyun
1349*4882a593Smuzhiyun return 0;
1350*4882a593Smuzhiyun }
1351*4882a593Smuzhiyun
snd_c400_create_mixer(struct usb_mixer_interface * mixer)1352*4882a593Smuzhiyun static int snd_c400_create_mixer(struct usb_mixer_interface *mixer)
1353*4882a593Smuzhiyun {
1354*4882a593Smuzhiyun int err;
1355*4882a593Smuzhiyun
1356*4882a593Smuzhiyun err = snd_c400_create_vol_ctls(mixer);
1357*4882a593Smuzhiyun if (err < 0)
1358*4882a593Smuzhiyun return err;
1359*4882a593Smuzhiyun
1360*4882a593Smuzhiyun err = snd_c400_create_effect_vol_ctls(mixer);
1361*4882a593Smuzhiyun if (err < 0)
1362*4882a593Smuzhiyun return err;
1363*4882a593Smuzhiyun
1364*4882a593Smuzhiyun err = snd_c400_create_effect_ret_vol_ctls(mixer);
1365*4882a593Smuzhiyun if (err < 0)
1366*4882a593Smuzhiyun return err;
1367*4882a593Smuzhiyun
1368*4882a593Smuzhiyun err = snd_ftu_create_effect_switch(mixer, 2, 0x43);
1369*4882a593Smuzhiyun if (err < 0)
1370*4882a593Smuzhiyun return err;
1371*4882a593Smuzhiyun
1372*4882a593Smuzhiyun err = snd_c400_create_effect_volume_ctl(mixer);
1373*4882a593Smuzhiyun if (err < 0)
1374*4882a593Smuzhiyun return err;
1375*4882a593Smuzhiyun
1376*4882a593Smuzhiyun err = snd_c400_create_effect_duration_ctl(mixer);
1377*4882a593Smuzhiyun if (err < 0)
1378*4882a593Smuzhiyun return err;
1379*4882a593Smuzhiyun
1380*4882a593Smuzhiyun err = snd_c400_create_effect_feedback_ctl(mixer);
1381*4882a593Smuzhiyun if (err < 0)
1382*4882a593Smuzhiyun return err;
1383*4882a593Smuzhiyun
1384*4882a593Smuzhiyun return 0;
1385*4882a593Smuzhiyun }
1386*4882a593Smuzhiyun
1387*4882a593Smuzhiyun /*
1388*4882a593Smuzhiyun * The mixer units for Ebox-44 are corrupt, and even where they
1389*4882a593Smuzhiyun * are valid they presents mono controls as L and R channels of
1390*4882a593Smuzhiyun * stereo. So we provide a good mixer here.
1391*4882a593Smuzhiyun */
1392*4882a593Smuzhiyun static const struct std_mono_table ebox44_table[] = {
1393*4882a593Smuzhiyun {
1394*4882a593Smuzhiyun .unitid = 4,
1395*4882a593Smuzhiyun .control = 1,
1396*4882a593Smuzhiyun .cmask = 0x0,
1397*4882a593Smuzhiyun .val_type = USB_MIXER_INV_BOOLEAN,
1398*4882a593Smuzhiyun .name = "Headphone Playback Switch"
1399*4882a593Smuzhiyun },
1400*4882a593Smuzhiyun {
1401*4882a593Smuzhiyun .unitid = 4,
1402*4882a593Smuzhiyun .control = 2,
1403*4882a593Smuzhiyun .cmask = 0x1,
1404*4882a593Smuzhiyun .val_type = USB_MIXER_S16,
1405*4882a593Smuzhiyun .name = "Headphone A Mix Playback Volume"
1406*4882a593Smuzhiyun },
1407*4882a593Smuzhiyun {
1408*4882a593Smuzhiyun .unitid = 4,
1409*4882a593Smuzhiyun .control = 2,
1410*4882a593Smuzhiyun .cmask = 0x2,
1411*4882a593Smuzhiyun .val_type = USB_MIXER_S16,
1412*4882a593Smuzhiyun .name = "Headphone B Mix Playback Volume"
1413*4882a593Smuzhiyun },
1414*4882a593Smuzhiyun
1415*4882a593Smuzhiyun {
1416*4882a593Smuzhiyun .unitid = 7,
1417*4882a593Smuzhiyun .control = 1,
1418*4882a593Smuzhiyun .cmask = 0x0,
1419*4882a593Smuzhiyun .val_type = USB_MIXER_INV_BOOLEAN,
1420*4882a593Smuzhiyun .name = "Output Playback Switch"
1421*4882a593Smuzhiyun },
1422*4882a593Smuzhiyun {
1423*4882a593Smuzhiyun .unitid = 7,
1424*4882a593Smuzhiyun .control = 2,
1425*4882a593Smuzhiyun .cmask = 0x1,
1426*4882a593Smuzhiyun .val_type = USB_MIXER_S16,
1427*4882a593Smuzhiyun .name = "Output A Playback Volume"
1428*4882a593Smuzhiyun },
1429*4882a593Smuzhiyun {
1430*4882a593Smuzhiyun .unitid = 7,
1431*4882a593Smuzhiyun .control = 2,
1432*4882a593Smuzhiyun .cmask = 0x2,
1433*4882a593Smuzhiyun .val_type = USB_MIXER_S16,
1434*4882a593Smuzhiyun .name = "Output B Playback Volume"
1435*4882a593Smuzhiyun },
1436*4882a593Smuzhiyun
1437*4882a593Smuzhiyun {
1438*4882a593Smuzhiyun .unitid = 10,
1439*4882a593Smuzhiyun .control = 1,
1440*4882a593Smuzhiyun .cmask = 0x0,
1441*4882a593Smuzhiyun .val_type = USB_MIXER_INV_BOOLEAN,
1442*4882a593Smuzhiyun .name = "Input Capture Switch"
1443*4882a593Smuzhiyun },
1444*4882a593Smuzhiyun {
1445*4882a593Smuzhiyun .unitid = 10,
1446*4882a593Smuzhiyun .control = 2,
1447*4882a593Smuzhiyun .cmask = 0x1,
1448*4882a593Smuzhiyun .val_type = USB_MIXER_S16,
1449*4882a593Smuzhiyun .name = "Input A Capture Volume"
1450*4882a593Smuzhiyun },
1451*4882a593Smuzhiyun {
1452*4882a593Smuzhiyun .unitid = 10,
1453*4882a593Smuzhiyun .control = 2,
1454*4882a593Smuzhiyun .cmask = 0x2,
1455*4882a593Smuzhiyun .val_type = USB_MIXER_S16,
1456*4882a593Smuzhiyun .name = "Input B Capture Volume"
1457*4882a593Smuzhiyun },
1458*4882a593Smuzhiyun
1459*4882a593Smuzhiyun {}
1460*4882a593Smuzhiyun };
1461*4882a593Smuzhiyun
1462*4882a593Smuzhiyun /* Audio Advantage Micro II findings:
1463*4882a593Smuzhiyun *
1464*4882a593Smuzhiyun * Mapping spdif AES bits to vendor register.bit:
1465*4882a593Smuzhiyun * AES0: [0 0 0 0 2.3 2.2 2.1 2.0] - default 0x00
1466*4882a593Smuzhiyun * AES1: [3.3 3.2.3.1.3.0 2.7 2.6 2.5 2.4] - default: 0x01
1467*4882a593Smuzhiyun * AES2: [0 0 0 0 0 0 0 0]
1468*4882a593Smuzhiyun * AES3: [0 0 0 0 0 0 x 0] - 'x' bit is set basing on standard usb request
1469*4882a593Smuzhiyun * (UAC_EP_CS_ATTR_SAMPLE_RATE) for Audio Devices
1470*4882a593Smuzhiyun *
1471*4882a593Smuzhiyun * power on values:
1472*4882a593Smuzhiyun * r2: 0x10
1473*4882a593Smuzhiyun * r3: 0x20 (b7 is zeroed just before playback (except IEC61937) and set
1474*4882a593Smuzhiyun * just after it to 0xa0, presumably it disables/mutes some analog
1475*4882a593Smuzhiyun * parts when there is no audio.)
1476*4882a593Smuzhiyun * r9: 0x28
1477*4882a593Smuzhiyun *
1478*4882a593Smuzhiyun * Optical transmitter on/off:
1479*4882a593Smuzhiyun * vendor register.bit: 9.1
1480*4882a593Smuzhiyun * 0 - on (0x28 register value)
1481*4882a593Smuzhiyun * 1 - off (0x2a register value)
1482*4882a593Smuzhiyun *
1483*4882a593Smuzhiyun */
snd_microii_spdif_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)1484*4882a593Smuzhiyun static int snd_microii_spdif_info(struct snd_kcontrol *kcontrol,
1485*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
1486*4882a593Smuzhiyun {
1487*4882a593Smuzhiyun uinfo->type = SNDRV_CTL_ELEM_TYPE_IEC958;
1488*4882a593Smuzhiyun uinfo->count = 1;
1489*4882a593Smuzhiyun return 0;
1490*4882a593Smuzhiyun }
1491*4882a593Smuzhiyun
snd_microii_spdif_default_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1492*4882a593Smuzhiyun static int snd_microii_spdif_default_get(struct snd_kcontrol *kcontrol,
1493*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1494*4882a593Smuzhiyun {
1495*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
1496*4882a593Smuzhiyun struct snd_usb_audio *chip = list->mixer->chip;
1497*4882a593Smuzhiyun int err;
1498*4882a593Smuzhiyun struct usb_interface *iface;
1499*4882a593Smuzhiyun struct usb_host_interface *alts;
1500*4882a593Smuzhiyun unsigned int ep;
1501*4882a593Smuzhiyun unsigned char data[3];
1502*4882a593Smuzhiyun int rate;
1503*4882a593Smuzhiyun
1504*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
1505*4882a593Smuzhiyun if (err < 0)
1506*4882a593Smuzhiyun return err;
1507*4882a593Smuzhiyun
1508*4882a593Smuzhiyun ucontrol->value.iec958.status[0] = kcontrol->private_value & 0xff;
1509*4882a593Smuzhiyun ucontrol->value.iec958.status[1] = (kcontrol->private_value >> 8) & 0xff;
1510*4882a593Smuzhiyun ucontrol->value.iec958.status[2] = 0x00;
1511*4882a593Smuzhiyun
1512*4882a593Smuzhiyun /* use known values for that card: interface#1 altsetting#1 */
1513*4882a593Smuzhiyun iface = usb_ifnum_to_if(chip->dev, 1);
1514*4882a593Smuzhiyun if (!iface || iface->num_altsetting < 2) {
1515*4882a593Smuzhiyun err = -EINVAL;
1516*4882a593Smuzhiyun goto end;
1517*4882a593Smuzhiyun }
1518*4882a593Smuzhiyun alts = &iface->altsetting[1];
1519*4882a593Smuzhiyun if (get_iface_desc(alts)->bNumEndpoints < 1) {
1520*4882a593Smuzhiyun err = -EINVAL;
1521*4882a593Smuzhiyun goto end;
1522*4882a593Smuzhiyun }
1523*4882a593Smuzhiyun ep = get_endpoint(alts, 0)->bEndpointAddress;
1524*4882a593Smuzhiyun
1525*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
1526*4882a593Smuzhiyun usb_rcvctrlpipe(chip->dev, 0),
1527*4882a593Smuzhiyun UAC_GET_CUR,
1528*4882a593Smuzhiyun USB_TYPE_CLASS | USB_RECIP_ENDPOINT | USB_DIR_IN,
1529*4882a593Smuzhiyun UAC_EP_CS_ATTR_SAMPLE_RATE << 8,
1530*4882a593Smuzhiyun ep,
1531*4882a593Smuzhiyun data,
1532*4882a593Smuzhiyun sizeof(data));
1533*4882a593Smuzhiyun if (err < 0)
1534*4882a593Smuzhiyun goto end;
1535*4882a593Smuzhiyun
1536*4882a593Smuzhiyun rate = data[0] | (data[1] << 8) | (data[2] << 16);
1537*4882a593Smuzhiyun ucontrol->value.iec958.status[3] = (rate == 48000) ?
1538*4882a593Smuzhiyun IEC958_AES3_CON_FS_48000 : IEC958_AES3_CON_FS_44100;
1539*4882a593Smuzhiyun
1540*4882a593Smuzhiyun err = 0;
1541*4882a593Smuzhiyun end:
1542*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
1543*4882a593Smuzhiyun return err;
1544*4882a593Smuzhiyun }
1545*4882a593Smuzhiyun
snd_microii_spdif_default_update(struct usb_mixer_elem_list * list)1546*4882a593Smuzhiyun static int snd_microii_spdif_default_update(struct usb_mixer_elem_list *list)
1547*4882a593Smuzhiyun {
1548*4882a593Smuzhiyun struct snd_usb_audio *chip = list->mixer->chip;
1549*4882a593Smuzhiyun unsigned int pval = list->kctl->private_value;
1550*4882a593Smuzhiyun u8 reg;
1551*4882a593Smuzhiyun int err;
1552*4882a593Smuzhiyun
1553*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
1554*4882a593Smuzhiyun if (err < 0)
1555*4882a593Smuzhiyun return err;
1556*4882a593Smuzhiyun
1557*4882a593Smuzhiyun reg = ((pval >> 4) & 0xf0) | (pval & 0x0f);
1558*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
1559*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0),
1560*4882a593Smuzhiyun UAC_SET_CUR,
1561*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_OTHER,
1562*4882a593Smuzhiyun reg,
1563*4882a593Smuzhiyun 2,
1564*4882a593Smuzhiyun NULL,
1565*4882a593Smuzhiyun 0);
1566*4882a593Smuzhiyun if (err < 0)
1567*4882a593Smuzhiyun goto end;
1568*4882a593Smuzhiyun
1569*4882a593Smuzhiyun reg = (pval & IEC958_AES0_NONAUDIO) ? 0xa0 : 0x20;
1570*4882a593Smuzhiyun reg |= (pval >> 12) & 0x0f;
1571*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
1572*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0),
1573*4882a593Smuzhiyun UAC_SET_CUR,
1574*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_OTHER,
1575*4882a593Smuzhiyun reg,
1576*4882a593Smuzhiyun 3,
1577*4882a593Smuzhiyun NULL,
1578*4882a593Smuzhiyun 0);
1579*4882a593Smuzhiyun if (err < 0)
1580*4882a593Smuzhiyun goto end;
1581*4882a593Smuzhiyun
1582*4882a593Smuzhiyun end:
1583*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
1584*4882a593Smuzhiyun return err;
1585*4882a593Smuzhiyun }
1586*4882a593Smuzhiyun
snd_microii_spdif_default_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1587*4882a593Smuzhiyun static int snd_microii_spdif_default_put(struct snd_kcontrol *kcontrol,
1588*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1589*4882a593Smuzhiyun {
1590*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
1591*4882a593Smuzhiyun unsigned int pval, pval_old;
1592*4882a593Smuzhiyun int err;
1593*4882a593Smuzhiyun
1594*4882a593Smuzhiyun pval = pval_old = kcontrol->private_value;
1595*4882a593Smuzhiyun pval &= 0xfffff0f0;
1596*4882a593Smuzhiyun pval |= (ucontrol->value.iec958.status[1] & 0x0f) << 8;
1597*4882a593Smuzhiyun pval |= (ucontrol->value.iec958.status[0] & 0x0f);
1598*4882a593Smuzhiyun
1599*4882a593Smuzhiyun pval &= 0xffff0fff;
1600*4882a593Smuzhiyun pval |= (ucontrol->value.iec958.status[1] & 0xf0) << 8;
1601*4882a593Smuzhiyun
1602*4882a593Smuzhiyun /* The frequency bits in AES3 cannot be set via register access. */
1603*4882a593Smuzhiyun
1604*4882a593Smuzhiyun /* Silently ignore any bits from the request that cannot be set. */
1605*4882a593Smuzhiyun
1606*4882a593Smuzhiyun if (pval == pval_old)
1607*4882a593Smuzhiyun return 0;
1608*4882a593Smuzhiyun
1609*4882a593Smuzhiyun kcontrol->private_value = pval;
1610*4882a593Smuzhiyun err = snd_microii_spdif_default_update(list);
1611*4882a593Smuzhiyun return err < 0 ? err : 1;
1612*4882a593Smuzhiyun }
1613*4882a593Smuzhiyun
snd_microii_spdif_mask_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1614*4882a593Smuzhiyun static int snd_microii_spdif_mask_get(struct snd_kcontrol *kcontrol,
1615*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1616*4882a593Smuzhiyun {
1617*4882a593Smuzhiyun ucontrol->value.iec958.status[0] = 0x0f;
1618*4882a593Smuzhiyun ucontrol->value.iec958.status[1] = 0xff;
1619*4882a593Smuzhiyun ucontrol->value.iec958.status[2] = 0x00;
1620*4882a593Smuzhiyun ucontrol->value.iec958.status[3] = 0x00;
1621*4882a593Smuzhiyun
1622*4882a593Smuzhiyun return 0;
1623*4882a593Smuzhiyun }
1624*4882a593Smuzhiyun
snd_microii_spdif_switch_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1625*4882a593Smuzhiyun static int snd_microii_spdif_switch_get(struct snd_kcontrol *kcontrol,
1626*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1627*4882a593Smuzhiyun {
1628*4882a593Smuzhiyun ucontrol->value.integer.value[0] = !(kcontrol->private_value & 0x02);
1629*4882a593Smuzhiyun
1630*4882a593Smuzhiyun return 0;
1631*4882a593Smuzhiyun }
1632*4882a593Smuzhiyun
snd_microii_spdif_switch_update(struct usb_mixer_elem_list * list)1633*4882a593Smuzhiyun static int snd_microii_spdif_switch_update(struct usb_mixer_elem_list *list)
1634*4882a593Smuzhiyun {
1635*4882a593Smuzhiyun struct snd_usb_audio *chip = list->mixer->chip;
1636*4882a593Smuzhiyun u8 reg = list->kctl->private_value;
1637*4882a593Smuzhiyun int err;
1638*4882a593Smuzhiyun
1639*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
1640*4882a593Smuzhiyun if (err < 0)
1641*4882a593Smuzhiyun return err;
1642*4882a593Smuzhiyun
1643*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
1644*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0),
1645*4882a593Smuzhiyun UAC_SET_CUR,
1646*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_OTHER,
1647*4882a593Smuzhiyun reg,
1648*4882a593Smuzhiyun 9,
1649*4882a593Smuzhiyun NULL,
1650*4882a593Smuzhiyun 0);
1651*4882a593Smuzhiyun
1652*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
1653*4882a593Smuzhiyun return err;
1654*4882a593Smuzhiyun }
1655*4882a593Smuzhiyun
snd_microii_spdif_switch_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1656*4882a593Smuzhiyun static int snd_microii_spdif_switch_put(struct snd_kcontrol *kcontrol,
1657*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1658*4882a593Smuzhiyun {
1659*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
1660*4882a593Smuzhiyun u8 reg;
1661*4882a593Smuzhiyun int err;
1662*4882a593Smuzhiyun
1663*4882a593Smuzhiyun reg = ucontrol->value.integer.value[0] ? 0x28 : 0x2a;
1664*4882a593Smuzhiyun if (reg != list->kctl->private_value)
1665*4882a593Smuzhiyun return 0;
1666*4882a593Smuzhiyun
1667*4882a593Smuzhiyun kcontrol->private_value = reg;
1668*4882a593Smuzhiyun err = snd_microii_spdif_switch_update(list);
1669*4882a593Smuzhiyun return err < 0 ? err : 1;
1670*4882a593Smuzhiyun }
1671*4882a593Smuzhiyun
1672*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_microii_mixer_spdif[] = {
1673*4882a593Smuzhiyun {
1674*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_PCM,
1675*4882a593Smuzhiyun .name = SNDRV_CTL_NAME_IEC958("", PLAYBACK, DEFAULT),
1676*4882a593Smuzhiyun .info = snd_microii_spdif_info,
1677*4882a593Smuzhiyun .get = snd_microii_spdif_default_get,
1678*4882a593Smuzhiyun .put = snd_microii_spdif_default_put,
1679*4882a593Smuzhiyun .private_value = 0x00000100UL,/* reset value */
1680*4882a593Smuzhiyun },
1681*4882a593Smuzhiyun {
1682*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ,
1683*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_PCM,
1684*4882a593Smuzhiyun .name = SNDRV_CTL_NAME_IEC958("", PLAYBACK, MASK),
1685*4882a593Smuzhiyun .info = snd_microii_spdif_info,
1686*4882a593Smuzhiyun .get = snd_microii_spdif_mask_get,
1687*4882a593Smuzhiyun },
1688*4882a593Smuzhiyun {
1689*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1690*4882a593Smuzhiyun .name = SNDRV_CTL_NAME_IEC958("", PLAYBACK, SWITCH),
1691*4882a593Smuzhiyun .info = snd_ctl_boolean_mono_info,
1692*4882a593Smuzhiyun .get = snd_microii_spdif_switch_get,
1693*4882a593Smuzhiyun .put = snd_microii_spdif_switch_put,
1694*4882a593Smuzhiyun .private_value = 0x00000028UL,/* reset value */
1695*4882a593Smuzhiyun }
1696*4882a593Smuzhiyun };
1697*4882a593Smuzhiyun
snd_microii_controls_create(struct usb_mixer_interface * mixer)1698*4882a593Smuzhiyun static int snd_microii_controls_create(struct usb_mixer_interface *mixer)
1699*4882a593Smuzhiyun {
1700*4882a593Smuzhiyun int err, i;
1701*4882a593Smuzhiyun static const usb_mixer_elem_resume_func_t resume_funcs[] = {
1702*4882a593Smuzhiyun snd_microii_spdif_default_update,
1703*4882a593Smuzhiyun NULL,
1704*4882a593Smuzhiyun snd_microii_spdif_switch_update
1705*4882a593Smuzhiyun };
1706*4882a593Smuzhiyun
1707*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(snd_microii_mixer_spdif); ++i) {
1708*4882a593Smuzhiyun err = add_single_ctl_with_resume(mixer, 0,
1709*4882a593Smuzhiyun resume_funcs[i],
1710*4882a593Smuzhiyun &snd_microii_mixer_spdif[i],
1711*4882a593Smuzhiyun NULL);
1712*4882a593Smuzhiyun if (err < 0)
1713*4882a593Smuzhiyun return err;
1714*4882a593Smuzhiyun }
1715*4882a593Smuzhiyun
1716*4882a593Smuzhiyun return 0;
1717*4882a593Smuzhiyun }
1718*4882a593Smuzhiyun
1719*4882a593Smuzhiyun /* Creative Sound Blaster E1 */
1720*4882a593Smuzhiyun
snd_soundblaster_e1_switch_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1721*4882a593Smuzhiyun static int snd_soundblaster_e1_switch_get(struct snd_kcontrol *kcontrol,
1722*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1723*4882a593Smuzhiyun {
1724*4882a593Smuzhiyun ucontrol->value.integer.value[0] = kcontrol->private_value;
1725*4882a593Smuzhiyun return 0;
1726*4882a593Smuzhiyun }
1727*4882a593Smuzhiyun
snd_soundblaster_e1_switch_update(struct usb_mixer_interface * mixer,unsigned char state)1728*4882a593Smuzhiyun static int snd_soundblaster_e1_switch_update(struct usb_mixer_interface *mixer,
1729*4882a593Smuzhiyun unsigned char state)
1730*4882a593Smuzhiyun {
1731*4882a593Smuzhiyun struct snd_usb_audio *chip = mixer->chip;
1732*4882a593Smuzhiyun int err;
1733*4882a593Smuzhiyun unsigned char buff[2];
1734*4882a593Smuzhiyun
1735*4882a593Smuzhiyun buff[0] = 0x02;
1736*4882a593Smuzhiyun buff[1] = state ? 0x02 : 0x00;
1737*4882a593Smuzhiyun
1738*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
1739*4882a593Smuzhiyun if (err < 0)
1740*4882a593Smuzhiyun return err;
1741*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
1742*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0), HID_REQ_SET_REPORT,
1743*4882a593Smuzhiyun USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT,
1744*4882a593Smuzhiyun 0x0202, 3, buff, 2);
1745*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
1746*4882a593Smuzhiyun return err;
1747*4882a593Smuzhiyun }
1748*4882a593Smuzhiyun
snd_soundblaster_e1_switch_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1749*4882a593Smuzhiyun static int snd_soundblaster_e1_switch_put(struct snd_kcontrol *kcontrol,
1750*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1751*4882a593Smuzhiyun {
1752*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
1753*4882a593Smuzhiyun unsigned char value = !!ucontrol->value.integer.value[0];
1754*4882a593Smuzhiyun int err;
1755*4882a593Smuzhiyun
1756*4882a593Smuzhiyun if (kcontrol->private_value == value)
1757*4882a593Smuzhiyun return 0;
1758*4882a593Smuzhiyun kcontrol->private_value = value;
1759*4882a593Smuzhiyun err = snd_soundblaster_e1_switch_update(list->mixer, value);
1760*4882a593Smuzhiyun return err < 0 ? err : 1;
1761*4882a593Smuzhiyun }
1762*4882a593Smuzhiyun
snd_soundblaster_e1_switch_resume(struct usb_mixer_elem_list * list)1763*4882a593Smuzhiyun static int snd_soundblaster_e1_switch_resume(struct usb_mixer_elem_list *list)
1764*4882a593Smuzhiyun {
1765*4882a593Smuzhiyun return snd_soundblaster_e1_switch_update(list->mixer,
1766*4882a593Smuzhiyun list->kctl->private_value);
1767*4882a593Smuzhiyun }
1768*4882a593Smuzhiyun
snd_soundblaster_e1_switch_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)1769*4882a593Smuzhiyun static int snd_soundblaster_e1_switch_info(struct snd_kcontrol *kcontrol,
1770*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
1771*4882a593Smuzhiyun {
1772*4882a593Smuzhiyun static const char *const texts[2] = {
1773*4882a593Smuzhiyun "Mic", "Aux"
1774*4882a593Smuzhiyun };
1775*4882a593Smuzhiyun
1776*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1, ARRAY_SIZE(texts), texts);
1777*4882a593Smuzhiyun }
1778*4882a593Smuzhiyun
1779*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_soundblaster_e1_input_switch = {
1780*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1781*4882a593Smuzhiyun .name = "Input Source",
1782*4882a593Smuzhiyun .info = snd_soundblaster_e1_switch_info,
1783*4882a593Smuzhiyun .get = snd_soundblaster_e1_switch_get,
1784*4882a593Smuzhiyun .put = snd_soundblaster_e1_switch_put,
1785*4882a593Smuzhiyun .private_value = 0,
1786*4882a593Smuzhiyun };
1787*4882a593Smuzhiyun
snd_soundblaster_e1_switch_create(struct usb_mixer_interface * mixer)1788*4882a593Smuzhiyun static int snd_soundblaster_e1_switch_create(struct usb_mixer_interface *mixer)
1789*4882a593Smuzhiyun {
1790*4882a593Smuzhiyun return add_single_ctl_with_resume(mixer, 0,
1791*4882a593Smuzhiyun snd_soundblaster_e1_switch_resume,
1792*4882a593Smuzhiyun &snd_soundblaster_e1_input_switch,
1793*4882a593Smuzhiyun NULL);
1794*4882a593Smuzhiyun }
1795*4882a593Smuzhiyun
dell_dock_init_vol(struct snd_usb_audio * chip,int ch,int id)1796*4882a593Smuzhiyun static void dell_dock_init_vol(struct snd_usb_audio *chip, int ch, int id)
1797*4882a593Smuzhiyun {
1798*4882a593Smuzhiyun u16 buf = 0;
1799*4882a593Smuzhiyun
1800*4882a593Smuzhiyun snd_usb_ctl_msg(chip->dev, usb_sndctrlpipe(chip->dev, 0), UAC_SET_CUR,
1801*4882a593Smuzhiyun USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_OUT,
1802*4882a593Smuzhiyun ch, snd_usb_ctrl_intf(chip) | (id << 8),
1803*4882a593Smuzhiyun &buf, 2);
1804*4882a593Smuzhiyun }
1805*4882a593Smuzhiyun
dell_dock_mixer_init(struct usb_mixer_interface * mixer)1806*4882a593Smuzhiyun static int dell_dock_mixer_init(struct usb_mixer_interface *mixer)
1807*4882a593Smuzhiyun {
1808*4882a593Smuzhiyun /* fix to 0dB playback volumes */
1809*4882a593Smuzhiyun dell_dock_init_vol(mixer->chip, 1, 16);
1810*4882a593Smuzhiyun dell_dock_init_vol(mixer->chip, 2, 16);
1811*4882a593Smuzhiyun dell_dock_init_vol(mixer->chip, 1, 19);
1812*4882a593Smuzhiyun dell_dock_init_vol(mixer->chip, 2, 19);
1813*4882a593Smuzhiyun return 0;
1814*4882a593Smuzhiyun }
1815*4882a593Smuzhiyun
1816*4882a593Smuzhiyun /* RME Class Compliant device quirks */
1817*4882a593Smuzhiyun
1818*4882a593Smuzhiyun #define SND_RME_GET_STATUS1 23
1819*4882a593Smuzhiyun #define SND_RME_GET_CURRENT_FREQ 17
1820*4882a593Smuzhiyun #define SND_RME_CLK_SYSTEM_SHIFT 16
1821*4882a593Smuzhiyun #define SND_RME_CLK_SYSTEM_MASK 0x1f
1822*4882a593Smuzhiyun #define SND_RME_CLK_AES_SHIFT 8
1823*4882a593Smuzhiyun #define SND_RME_CLK_SPDIF_SHIFT 12
1824*4882a593Smuzhiyun #define SND_RME_CLK_AES_SPDIF_MASK 0xf
1825*4882a593Smuzhiyun #define SND_RME_CLK_SYNC_SHIFT 6
1826*4882a593Smuzhiyun #define SND_RME_CLK_SYNC_MASK 0x3
1827*4882a593Smuzhiyun #define SND_RME_CLK_FREQMUL_SHIFT 18
1828*4882a593Smuzhiyun #define SND_RME_CLK_FREQMUL_MASK 0x7
1829*4882a593Smuzhiyun #define SND_RME_CLK_SYSTEM(x) \
1830*4882a593Smuzhiyun ((x >> SND_RME_CLK_SYSTEM_SHIFT) & SND_RME_CLK_SYSTEM_MASK)
1831*4882a593Smuzhiyun #define SND_RME_CLK_AES(x) \
1832*4882a593Smuzhiyun ((x >> SND_RME_CLK_AES_SHIFT) & SND_RME_CLK_AES_SPDIF_MASK)
1833*4882a593Smuzhiyun #define SND_RME_CLK_SPDIF(x) \
1834*4882a593Smuzhiyun ((x >> SND_RME_CLK_SPDIF_SHIFT) & SND_RME_CLK_AES_SPDIF_MASK)
1835*4882a593Smuzhiyun #define SND_RME_CLK_SYNC(x) \
1836*4882a593Smuzhiyun ((x >> SND_RME_CLK_SYNC_SHIFT) & SND_RME_CLK_SYNC_MASK)
1837*4882a593Smuzhiyun #define SND_RME_CLK_FREQMUL(x) \
1838*4882a593Smuzhiyun ((x >> SND_RME_CLK_FREQMUL_SHIFT) & SND_RME_CLK_FREQMUL_MASK)
1839*4882a593Smuzhiyun #define SND_RME_CLK_AES_LOCK 0x1
1840*4882a593Smuzhiyun #define SND_RME_CLK_AES_SYNC 0x4
1841*4882a593Smuzhiyun #define SND_RME_CLK_SPDIF_LOCK 0x2
1842*4882a593Smuzhiyun #define SND_RME_CLK_SPDIF_SYNC 0x8
1843*4882a593Smuzhiyun #define SND_RME_SPDIF_IF_SHIFT 4
1844*4882a593Smuzhiyun #define SND_RME_SPDIF_FORMAT_SHIFT 5
1845*4882a593Smuzhiyun #define SND_RME_BINARY_MASK 0x1
1846*4882a593Smuzhiyun #define SND_RME_SPDIF_IF(x) \
1847*4882a593Smuzhiyun ((x >> SND_RME_SPDIF_IF_SHIFT) & SND_RME_BINARY_MASK)
1848*4882a593Smuzhiyun #define SND_RME_SPDIF_FORMAT(x) \
1849*4882a593Smuzhiyun ((x >> SND_RME_SPDIF_FORMAT_SHIFT) & SND_RME_BINARY_MASK)
1850*4882a593Smuzhiyun
1851*4882a593Smuzhiyun static const u32 snd_rme_rate_table[] = {
1852*4882a593Smuzhiyun 32000, 44100, 48000, 50000,
1853*4882a593Smuzhiyun 64000, 88200, 96000, 100000,
1854*4882a593Smuzhiyun 128000, 176400, 192000, 200000,
1855*4882a593Smuzhiyun 256000, 352800, 384000, 400000,
1856*4882a593Smuzhiyun 512000, 705600, 768000, 800000
1857*4882a593Smuzhiyun };
1858*4882a593Smuzhiyun /* maximum number of items for AES and S/PDIF rates for above table */
1859*4882a593Smuzhiyun #define SND_RME_RATE_IDX_AES_SPDIF_NUM 12
1860*4882a593Smuzhiyun
1861*4882a593Smuzhiyun enum snd_rme_domain {
1862*4882a593Smuzhiyun SND_RME_DOMAIN_SYSTEM,
1863*4882a593Smuzhiyun SND_RME_DOMAIN_AES,
1864*4882a593Smuzhiyun SND_RME_DOMAIN_SPDIF
1865*4882a593Smuzhiyun };
1866*4882a593Smuzhiyun
1867*4882a593Smuzhiyun enum snd_rme_clock_status {
1868*4882a593Smuzhiyun SND_RME_CLOCK_NOLOCK,
1869*4882a593Smuzhiyun SND_RME_CLOCK_LOCK,
1870*4882a593Smuzhiyun SND_RME_CLOCK_SYNC
1871*4882a593Smuzhiyun };
1872*4882a593Smuzhiyun
snd_rme_read_value(struct snd_usb_audio * chip,unsigned int item,u32 * value)1873*4882a593Smuzhiyun static int snd_rme_read_value(struct snd_usb_audio *chip,
1874*4882a593Smuzhiyun unsigned int item,
1875*4882a593Smuzhiyun u32 *value)
1876*4882a593Smuzhiyun {
1877*4882a593Smuzhiyun struct usb_device *dev = chip->dev;
1878*4882a593Smuzhiyun int err;
1879*4882a593Smuzhiyun
1880*4882a593Smuzhiyun err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0),
1881*4882a593Smuzhiyun item,
1882*4882a593Smuzhiyun USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
1883*4882a593Smuzhiyun 0, 0,
1884*4882a593Smuzhiyun value, sizeof(*value));
1885*4882a593Smuzhiyun if (err < 0)
1886*4882a593Smuzhiyun dev_err(&dev->dev,
1887*4882a593Smuzhiyun "unable to issue vendor read request %d (ret = %d)",
1888*4882a593Smuzhiyun item, err);
1889*4882a593Smuzhiyun return err;
1890*4882a593Smuzhiyun }
1891*4882a593Smuzhiyun
snd_rme_get_status1(struct snd_kcontrol * kcontrol,u32 * status1)1892*4882a593Smuzhiyun static int snd_rme_get_status1(struct snd_kcontrol *kcontrol,
1893*4882a593Smuzhiyun u32 *status1)
1894*4882a593Smuzhiyun {
1895*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
1896*4882a593Smuzhiyun struct snd_usb_audio *chip = list->mixer->chip;
1897*4882a593Smuzhiyun int err;
1898*4882a593Smuzhiyun
1899*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
1900*4882a593Smuzhiyun if (err < 0)
1901*4882a593Smuzhiyun return err;
1902*4882a593Smuzhiyun err = snd_rme_read_value(chip, SND_RME_GET_STATUS1, status1);
1903*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
1904*4882a593Smuzhiyun return err;
1905*4882a593Smuzhiyun }
1906*4882a593Smuzhiyun
snd_rme_rate_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1907*4882a593Smuzhiyun static int snd_rme_rate_get(struct snd_kcontrol *kcontrol,
1908*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1909*4882a593Smuzhiyun {
1910*4882a593Smuzhiyun u32 status1;
1911*4882a593Smuzhiyun u32 rate = 0;
1912*4882a593Smuzhiyun int idx;
1913*4882a593Smuzhiyun int err;
1914*4882a593Smuzhiyun
1915*4882a593Smuzhiyun err = snd_rme_get_status1(kcontrol, &status1);
1916*4882a593Smuzhiyun if (err < 0)
1917*4882a593Smuzhiyun return err;
1918*4882a593Smuzhiyun switch (kcontrol->private_value) {
1919*4882a593Smuzhiyun case SND_RME_DOMAIN_SYSTEM:
1920*4882a593Smuzhiyun idx = SND_RME_CLK_SYSTEM(status1);
1921*4882a593Smuzhiyun if (idx < ARRAY_SIZE(snd_rme_rate_table))
1922*4882a593Smuzhiyun rate = snd_rme_rate_table[idx];
1923*4882a593Smuzhiyun break;
1924*4882a593Smuzhiyun case SND_RME_DOMAIN_AES:
1925*4882a593Smuzhiyun idx = SND_RME_CLK_AES(status1);
1926*4882a593Smuzhiyun if (idx < SND_RME_RATE_IDX_AES_SPDIF_NUM)
1927*4882a593Smuzhiyun rate = snd_rme_rate_table[idx];
1928*4882a593Smuzhiyun break;
1929*4882a593Smuzhiyun case SND_RME_DOMAIN_SPDIF:
1930*4882a593Smuzhiyun idx = SND_RME_CLK_SPDIF(status1);
1931*4882a593Smuzhiyun if (idx < SND_RME_RATE_IDX_AES_SPDIF_NUM)
1932*4882a593Smuzhiyun rate = snd_rme_rate_table[idx];
1933*4882a593Smuzhiyun break;
1934*4882a593Smuzhiyun default:
1935*4882a593Smuzhiyun return -EINVAL;
1936*4882a593Smuzhiyun }
1937*4882a593Smuzhiyun ucontrol->value.integer.value[0] = rate;
1938*4882a593Smuzhiyun return 0;
1939*4882a593Smuzhiyun }
1940*4882a593Smuzhiyun
snd_rme_sync_state_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1941*4882a593Smuzhiyun static int snd_rme_sync_state_get(struct snd_kcontrol *kcontrol,
1942*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1943*4882a593Smuzhiyun {
1944*4882a593Smuzhiyun u32 status1;
1945*4882a593Smuzhiyun int idx = SND_RME_CLOCK_NOLOCK;
1946*4882a593Smuzhiyun int err;
1947*4882a593Smuzhiyun
1948*4882a593Smuzhiyun err = snd_rme_get_status1(kcontrol, &status1);
1949*4882a593Smuzhiyun if (err < 0)
1950*4882a593Smuzhiyun return err;
1951*4882a593Smuzhiyun switch (kcontrol->private_value) {
1952*4882a593Smuzhiyun case SND_RME_DOMAIN_AES: /* AES */
1953*4882a593Smuzhiyun if (status1 & SND_RME_CLK_AES_SYNC)
1954*4882a593Smuzhiyun idx = SND_RME_CLOCK_SYNC;
1955*4882a593Smuzhiyun else if (status1 & SND_RME_CLK_AES_LOCK)
1956*4882a593Smuzhiyun idx = SND_RME_CLOCK_LOCK;
1957*4882a593Smuzhiyun break;
1958*4882a593Smuzhiyun case SND_RME_DOMAIN_SPDIF: /* SPDIF */
1959*4882a593Smuzhiyun if (status1 & SND_RME_CLK_SPDIF_SYNC)
1960*4882a593Smuzhiyun idx = SND_RME_CLOCK_SYNC;
1961*4882a593Smuzhiyun else if (status1 & SND_RME_CLK_SPDIF_LOCK)
1962*4882a593Smuzhiyun idx = SND_RME_CLOCK_LOCK;
1963*4882a593Smuzhiyun break;
1964*4882a593Smuzhiyun default:
1965*4882a593Smuzhiyun return -EINVAL;
1966*4882a593Smuzhiyun }
1967*4882a593Smuzhiyun ucontrol->value.enumerated.item[0] = idx;
1968*4882a593Smuzhiyun return 0;
1969*4882a593Smuzhiyun }
1970*4882a593Smuzhiyun
snd_rme_spdif_if_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1971*4882a593Smuzhiyun static int snd_rme_spdif_if_get(struct snd_kcontrol *kcontrol,
1972*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1973*4882a593Smuzhiyun {
1974*4882a593Smuzhiyun u32 status1;
1975*4882a593Smuzhiyun int err;
1976*4882a593Smuzhiyun
1977*4882a593Smuzhiyun err = snd_rme_get_status1(kcontrol, &status1);
1978*4882a593Smuzhiyun if (err < 0)
1979*4882a593Smuzhiyun return err;
1980*4882a593Smuzhiyun ucontrol->value.enumerated.item[0] = SND_RME_SPDIF_IF(status1);
1981*4882a593Smuzhiyun return 0;
1982*4882a593Smuzhiyun }
1983*4882a593Smuzhiyun
snd_rme_spdif_format_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1984*4882a593Smuzhiyun static int snd_rme_spdif_format_get(struct snd_kcontrol *kcontrol,
1985*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1986*4882a593Smuzhiyun {
1987*4882a593Smuzhiyun u32 status1;
1988*4882a593Smuzhiyun int err;
1989*4882a593Smuzhiyun
1990*4882a593Smuzhiyun err = snd_rme_get_status1(kcontrol, &status1);
1991*4882a593Smuzhiyun if (err < 0)
1992*4882a593Smuzhiyun return err;
1993*4882a593Smuzhiyun ucontrol->value.enumerated.item[0] = SND_RME_SPDIF_FORMAT(status1);
1994*4882a593Smuzhiyun return 0;
1995*4882a593Smuzhiyun }
1996*4882a593Smuzhiyun
snd_rme_sync_source_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)1997*4882a593Smuzhiyun static int snd_rme_sync_source_get(struct snd_kcontrol *kcontrol,
1998*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
1999*4882a593Smuzhiyun {
2000*4882a593Smuzhiyun u32 status1;
2001*4882a593Smuzhiyun int err;
2002*4882a593Smuzhiyun
2003*4882a593Smuzhiyun err = snd_rme_get_status1(kcontrol, &status1);
2004*4882a593Smuzhiyun if (err < 0)
2005*4882a593Smuzhiyun return err;
2006*4882a593Smuzhiyun ucontrol->value.enumerated.item[0] = SND_RME_CLK_SYNC(status1);
2007*4882a593Smuzhiyun return 0;
2008*4882a593Smuzhiyun }
2009*4882a593Smuzhiyun
snd_rme_current_freq_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)2010*4882a593Smuzhiyun static int snd_rme_current_freq_get(struct snd_kcontrol *kcontrol,
2011*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
2012*4882a593Smuzhiyun {
2013*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
2014*4882a593Smuzhiyun struct snd_usb_audio *chip = list->mixer->chip;
2015*4882a593Smuzhiyun u32 status1;
2016*4882a593Smuzhiyun const u64 num = 104857600000000ULL;
2017*4882a593Smuzhiyun u32 den;
2018*4882a593Smuzhiyun unsigned int freq;
2019*4882a593Smuzhiyun int err;
2020*4882a593Smuzhiyun
2021*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
2022*4882a593Smuzhiyun if (err < 0)
2023*4882a593Smuzhiyun return err;
2024*4882a593Smuzhiyun err = snd_rme_read_value(chip, SND_RME_GET_STATUS1, &status1);
2025*4882a593Smuzhiyun if (err < 0)
2026*4882a593Smuzhiyun goto end;
2027*4882a593Smuzhiyun err = snd_rme_read_value(chip, SND_RME_GET_CURRENT_FREQ, &den);
2028*4882a593Smuzhiyun if (err < 0)
2029*4882a593Smuzhiyun goto end;
2030*4882a593Smuzhiyun freq = (den == 0) ? 0 : div64_u64(num, den);
2031*4882a593Smuzhiyun freq <<= SND_RME_CLK_FREQMUL(status1);
2032*4882a593Smuzhiyun ucontrol->value.integer.value[0] = freq;
2033*4882a593Smuzhiyun
2034*4882a593Smuzhiyun end:
2035*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
2036*4882a593Smuzhiyun return err;
2037*4882a593Smuzhiyun }
2038*4882a593Smuzhiyun
snd_rme_rate_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)2039*4882a593Smuzhiyun static int snd_rme_rate_info(struct snd_kcontrol *kcontrol,
2040*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
2041*4882a593Smuzhiyun {
2042*4882a593Smuzhiyun uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
2043*4882a593Smuzhiyun uinfo->count = 1;
2044*4882a593Smuzhiyun switch (kcontrol->private_value) {
2045*4882a593Smuzhiyun case SND_RME_DOMAIN_SYSTEM:
2046*4882a593Smuzhiyun uinfo->value.integer.min = 32000;
2047*4882a593Smuzhiyun uinfo->value.integer.max = 800000;
2048*4882a593Smuzhiyun break;
2049*4882a593Smuzhiyun case SND_RME_DOMAIN_AES:
2050*4882a593Smuzhiyun case SND_RME_DOMAIN_SPDIF:
2051*4882a593Smuzhiyun default:
2052*4882a593Smuzhiyun uinfo->value.integer.min = 0;
2053*4882a593Smuzhiyun uinfo->value.integer.max = 200000;
2054*4882a593Smuzhiyun }
2055*4882a593Smuzhiyun uinfo->value.integer.step = 0;
2056*4882a593Smuzhiyun return 0;
2057*4882a593Smuzhiyun }
2058*4882a593Smuzhiyun
snd_rme_sync_state_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)2059*4882a593Smuzhiyun static int snd_rme_sync_state_info(struct snd_kcontrol *kcontrol,
2060*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
2061*4882a593Smuzhiyun {
2062*4882a593Smuzhiyun static const char *const sync_states[] = {
2063*4882a593Smuzhiyun "No Lock", "Lock", "Sync"
2064*4882a593Smuzhiyun };
2065*4882a593Smuzhiyun
2066*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1,
2067*4882a593Smuzhiyun ARRAY_SIZE(sync_states), sync_states);
2068*4882a593Smuzhiyun }
2069*4882a593Smuzhiyun
snd_rme_spdif_if_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)2070*4882a593Smuzhiyun static int snd_rme_spdif_if_info(struct snd_kcontrol *kcontrol,
2071*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
2072*4882a593Smuzhiyun {
2073*4882a593Smuzhiyun static const char *const spdif_if[] = {
2074*4882a593Smuzhiyun "Coaxial", "Optical"
2075*4882a593Smuzhiyun };
2076*4882a593Smuzhiyun
2077*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1,
2078*4882a593Smuzhiyun ARRAY_SIZE(spdif_if), spdif_if);
2079*4882a593Smuzhiyun }
2080*4882a593Smuzhiyun
snd_rme_spdif_format_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)2081*4882a593Smuzhiyun static int snd_rme_spdif_format_info(struct snd_kcontrol *kcontrol,
2082*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
2083*4882a593Smuzhiyun {
2084*4882a593Smuzhiyun static const char *const optical_type[] = {
2085*4882a593Smuzhiyun "Consumer", "Professional"
2086*4882a593Smuzhiyun };
2087*4882a593Smuzhiyun
2088*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1,
2089*4882a593Smuzhiyun ARRAY_SIZE(optical_type), optical_type);
2090*4882a593Smuzhiyun }
2091*4882a593Smuzhiyun
snd_rme_sync_source_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)2092*4882a593Smuzhiyun static int snd_rme_sync_source_info(struct snd_kcontrol *kcontrol,
2093*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
2094*4882a593Smuzhiyun {
2095*4882a593Smuzhiyun static const char *const sync_sources[] = {
2096*4882a593Smuzhiyun "Internal", "AES", "SPDIF", "Internal"
2097*4882a593Smuzhiyun };
2098*4882a593Smuzhiyun
2099*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1,
2100*4882a593Smuzhiyun ARRAY_SIZE(sync_sources), sync_sources);
2101*4882a593Smuzhiyun }
2102*4882a593Smuzhiyun
2103*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_rme_controls[] = {
2104*4882a593Smuzhiyun {
2105*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2106*4882a593Smuzhiyun .name = "AES Rate",
2107*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
2108*4882a593Smuzhiyun .info = snd_rme_rate_info,
2109*4882a593Smuzhiyun .get = snd_rme_rate_get,
2110*4882a593Smuzhiyun .private_value = SND_RME_DOMAIN_AES
2111*4882a593Smuzhiyun },
2112*4882a593Smuzhiyun {
2113*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2114*4882a593Smuzhiyun .name = "AES Sync",
2115*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
2116*4882a593Smuzhiyun .info = snd_rme_sync_state_info,
2117*4882a593Smuzhiyun .get = snd_rme_sync_state_get,
2118*4882a593Smuzhiyun .private_value = SND_RME_DOMAIN_AES
2119*4882a593Smuzhiyun },
2120*4882a593Smuzhiyun {
2121*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2122*4882a593Smuzhiyun .name = "SPDIF Rate",
2123*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
2124*4882a593Smuzhiyun .info = snd_rme_rate_info,
2125*4882a593Smuzhiyun .get = snd_rme_rate_get,
2126*4882a593Smuzhiyun .private_value = SND_RME_DOMAIN_SPDIF
2127*4882a593Smuzhiyun },
2128*4882a593Smuzhiyun {
2129*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2130*4882a593Smuzhiyun .name = "SPDIF Sync",
2131*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
2132*4882a593Smuzhiyun .info = snd_rme_sync_state_info,
2133*4882a593Smuzhiyun .get = snd_rme_sync_state_get,
2134*4882a593Smuzhiyun .private_value = SND_RME_DOMAIN_SPDIF
2135*4882a593Smuzhiyun },
2136*4882a593Smuzhiyun {
2137*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2138*4882a593Smuzhiyun .name = "SPDIF Interface",
2139*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
2140*4882a593Smuzhiyun .info = snd_rme_spdif_if_info,
2141*4882a593Smuzhiyun .get = snd_rme_spdif_if_get,
2142*4882a593Smuzhiyun },
2143*4882a593Smuzhiyun {
2144*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2145*4882a593Smuzhiyun .name = "SPDIF Format",
2146*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
2147*4882a593Smuzhiyun .info = snd_rme_spdif_format_info,
2148*4882a593Smuzhiyun .get = snd_rme_spdif_format_get,
2149*4882a593Smuzhiyun },
2150*4882a593Smuzhiyun {
2151*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2152*4882a593Smuzhiyun .name = "Sync Source",
2153*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
2154*4882a593Smuzhiyun .info = snd_rme_sync_source_info,
2155*4882a593Smuzhiyun .get = snd_rme_sync_source_get
2156*4882a593Smuzhiyun },
2157*4882a593Smuzhiyun {
2158*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2159*4882a593Smuzhiyun .name = "System Rate",
2160*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
2161*4882a593Smuzhiyun .info = snd_rme_rate_info,
2162*4882a593Smuzhiyun .get = snd_rme_rate_get,
2163*4882a593Smuzhiyun .private_value = SND_RME_DOMAIN_SYSTEM
2164*4882a593Smuzhiyun },
2165*4882a593Smuzhiyun {
2166*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2167*4882a593Smuzhiyun .name = "Current Frequency",
2168*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
2169*4882a593Smuzhiyun .info = snd_rme_rate_info,
2170*4882a593Smuzhiyun .get = snd_rme_current_freq_get
2171*4882a593Smuzhiyun }
2172*4882a593Smuzhiyun };
2173*4882a593Smuzhiyun
snd_rme_controls_create(struct usb_mixer_interface * mixer)2174*4882a593Smuzhiyun static int snd_rme_controls_create(struct usb_mixer_interface *mixer)
2175*4882a593Smuzhiyun {
2176*4882a593Smuzhiyun int err, i;
2177*4882a593Smuzhiyun
2178*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(snd_rme_controls); ++i) {
2179*4882a593Smuzhiyun err = add_single_ctl_with_resume(mixer, 0,
2180*4882a593Smuzhiyun NULL,
2181*4882a593Smuzhiyun &snd_rme_controls[i],
2182*4882a593Smuzhiyun NULL);
2183*4882a593Smuzhiyun if (err < 0)
2184*4882a593Smuzhiyun return err;
2185*4882a593Smuzhiyun }
2186*4882a593Smuzhiyun
2187*4882a593Smuzhiyun return 0;
2188*4882a593Smuzhiyun }
2189*4882a593Smuzhiyun
2190*4882a593Smuzhiyun /*
2191*4882a593Smuzhiyun * RME Babyface Pro (FS)
2192*4882a593Smuzhiyun *
2193*4882a593Smuzhiyun * These devices exposes a couple of DSP functions via request to EP0.
2194*4882a593Smuzhiyun * Switches are available via control registers, while routing is controlled
2195*4882a593Smuzhiyun * by controlling the volume on each possible crossing point.
2196*4882a593Smuzhiyun * Volume control is linear, from -inf (dec. 0) to +6dB (dec. 65536) with
2197*4882a593Smuzhiyun * 0dB being at dec. 32768.
2198*4882a593Smuzhiyun */
2199*4882a593Smuzhiyun enum {
2200*4882a593Smuzhiyun SND_BBFPRO_CTL_REG1 = 0,
2201*4882a593Smuzhiyun SND_BBFPRO_CTL_REG2
2202*4882a593Smuzhiyun };
2203*4882a593Smuzhiyun
2204*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG_MASK 1
2205*4882a593Smuzhiyun #define SND_BBFPRO_CTL_IDX_MASK 0xff
2206*4882a593Smuzhiyun #define SND_BBFPRO_CTL_IDX_SHIFT 1
2207*4882a593Smuzhiyun #define SND_BBFPRO_CTL_VAL_MASK 1
2208*4882a593Smuzhiyun #define SND_BBFPRO_CTL_VAL_SHIFT 9
2209*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG1_CLK_MASTER 0
2210*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG1_CLK_OPTICAL 1
2211*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG1_SPDIF_PRO 7
2212*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG1_SPDIF_EMPH 8
2213*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG1_SPDIF_OPTICAL 10
2214*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG2_48V_AN1 0
2215*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG2_48V_AN2 1
2216*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG2_SENS_IN3 2
2217*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG2_SENS_IN4 3
2218*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG2_PAD_AN1 4
2219*4882a593Smuzhiyun #define SND_BBFPRO_CTL_REG2_PAD_AN2 5
2220*4882a593Smuzhiyun
2221*4882a593Smuzhiyun #define SND_BBFPRO_MIXER_IDX_MASK 0x1ff
2222*4882a593Smuzhiyun #define SND_BBFPRO_MIXER_VAL_MASK 0x3ffff
2223*4882a593Smuzhiyun #define SND_BBFPRO_MIXER_VAL_SHIFT 9
2224*4882a593Smuzhiyun #define SND_BBFPRO_MIXER_VAL_MIN 0 // -inf
2225*4882a593Smuzhiyun #define SND_BBFPRO_MIXER_VAL_MAX 65536 // +6dB
2226*4882a593Smuzhiyun
2227*4882a593Smuzhiyun #define SND_BBFPRO_USBREQ_CTL_REG1 0x10
2228*4882a593Smuzhiyun #define SND_BBFPRO_USBREQ_CTL_REG2 0x17
2229*4882a593Smuzhiyun #define SND_BBFPRO_USBREQ_MIXER 0x12
2230*4882a593Smuzhiyun
snd_bbfpro_ctl_update(struct usb_mixer_interface * mixer,u8 reg,u8 index,u8 value)2231*4882a593Smuzhiyun static int snd_bbfpro_ctl_update(struct usb_mixer_interface *mixer, u8 reg,
2232*4882a593Smuzhiyun u8 index, u8 value)
2233*4882a593Smuzhiyun {
2234*4882a593Smuzhiyun int err;
2235*4882a593Smuzhiyun u16 usb_req, usb_idx, usb_val;
2236*4882a593Smuzhiyun struct snd_usb_audio *chip = mixer->chip;
2237*4882a593Smuzhiyun
2238*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
2239*4882a593Smuzhiyun if (err < 0)
2240*4882a593Smuzhiyun return err;
2241*4882a593Smuzhiyun
2242*4882a593Smuzhiyun if (reg == SND_BBFPRO_CTL_REG1) {
2243*4882a593Smuzhiyun usb_req = SND_BBFPRO_USBREQ_CTL_REG1;
2244*4882a593Smuzhiyun if (index == SND_BBFPRO_CTL_REG1_CLK_OPTICAL) {
2245*4882a593Smuzhiyun usb_idx = 3;
2246*4882a593Smuzhiyun usb_val = value ? 3 : 0;
2247*4882a593Smuzhiyun } else {
2248*4882a593Smuzhiyun usb_idx = 1 << index;
2249*4882a593Smuzhiyun usb_val = value ? usb_idx : 0;
2250*4882a593Smuzhiyun }
2251*4882a593Smuzhiyun } else {
2252*4882a593Smuzhiyun usb_req = SND_BBFPRO_USBREQ_CTL_REG2;
2253*4882a593Smuzhiyun usb_idx = 1 << index;
2254*4882a593Smuzhiyun usb_val = value ? usb_idx : 0;
2255*4882a593Smuzhiyun }
2256*4882a593Smuzhiyun
2257*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
2258*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0), usb_req,
2259*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
2260*4882a593Smuzhiyun usb_val, usb_idx, NULL, 0);
2261*4882a593Smuzhiyun
2262*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
2263*4882a593Smuzhiyun return err;
2264*4882a593Smuzhiyun }
2265*4882a593Smuzhiyun
snd_bbfpro_ctl_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)2266*4882a593Smuzhiyun static int snd_bbfpro_ctl_get(struct snd_kcontrol *kcontrol,
2267*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
2268*4882a593Smuzhiyun {
2269*4882a593Smuzhiyun u8 reg, idx, val;
2270*4882a593Smuzhiyun int pv;
2271*4882a593Smuzhiyun
2272*4882a593Smuzhiyun pv = kcontrol->private_value;
2273*4882a593Smuzhiyun reg = pv & SND_BBFPRO_CTL_REG_MASK;
2274*4882a593Smuzhiyun idx = (pv >> SND_BBFPRO_CTL_IDX_SHIFT) & SND_BBFPRO_CTL_IDX_MASK;
2275*4882a593Smuzhiyun val = kcontrol->private_value >> SND_BBFPRO_CTL_VAL_SHIFT;
2276*4882a593Smuzhiyun
2277*4882a593Smuzhiyun if ((reg == SND_BBFPRO_CTL_REG1 &&
2278*4882a593Smuzhiyun idx == SND_BBFPRO_CTL_REG1_CLK_OPTICAL) ||
2279*4882a593Smuzhiyun (reg == SND_BBFPRO_CTL_REG2 &&
2280*4882a593Smuzhiyun (idx == SND_BBFPRO_CTL_REG2_SENS_IN3 ||
2281*4882a593Smuzhiyun idx == SND_BBFPRO_CTL_REG2_SENS_IN4))) {
2282*4882a593Smuzhiyun ucontrol->value.enumerated.item[0] = val;
2283*4882a593Smuzhiyun } else {
2284*4882a593Smuzhiyun ucontrol->value.integer.value[0] = val;
2285*4882a593Smuzhiyun }
2286*4882a593Smuzhiyun return 0;
2287*4882a593Smuzhiyun }
2288*4882a593Smuzhiyun
snd_bbfpro_ctl_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)2289*4882a593Smuzhiyun static int snd_bbfpro_ctl_info(struct snd_kcontrol *kcontrol,
2290*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
2291*4882a593Smuzhiyun {
2292*4882a593Smuzhiyun u8 reg, idx;
2293*4882a593Smuzhiyun int pv;
2294*4882a593Smuzhiyun
2295*4882a593Smuzhiyun pv = kcontrol->private_value;
2296*4882a593Smuzhiyun reg = pv & SND_BBFPRO_CTL_REG_MASK;
2297*4882a593Smuzhiyun idx = (pv >> SND_BBFPRO_CTL_IDX_SHIFT) & SND_BBFPRO_CTL_IDX_MASK;
2298*4882a593Smuzhiyun
2299*4882a593Smuzhiyun if (reg == SND_BBFPRO_CTL_REG1 &&
2300*4882a593Smuzhiyun idx == SND_BBFPRO_CTL_REG1_CLK_OPTICAL) {
2301*4882a593Smuzhiyun static const char * const texts[2] = {
2302*4882a593Smuzhiyun "AutoSync",
2303*4882a593Smuzhiyun "Internal"
2304*4882a593Smuzhiyun };
2305*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1, 2, texts);
2306*4882a593Smuzhiyun } else if (reg == SND_BBFPRO_CTL_REG2 &&
2307*4882a593Smuzhiyun (idx == SND_BBFPRO_CTL_REG2_SENS_IN3 ||
2308*4882a593Smuzhiyun idx == SND_BBFPRO_CTL_REG2_SENS_IN4)) {
2309*4882a593Smuzhiyun static const char * const texts[2] = {
2310*4882a593Smuzhiyun "-10dBV",
2311*4882a593Smuzhiyun "+4dBu"
2312*4882a593Smuzhiyun };
2313*4882a593Smuzhiyun return snd_ctl_enum_info(uinfo, 1, 2, texts);
2314*4882a593Smuzhiyun }
2315*4882a593Smuzhiyun
2316*4882a593Smuzhiyun uinfo->count = 1;
2317*4882a593Smuzhiyun uinfo->value.integer.min = 0;
2318*4882a593Smuzhiyun uinfo->value.integer.max = 1;
2319*4882a593Smuzhiyun uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
2320*4882a593Smuzhiyun return 0;
2321*4882a593Smuzhiyun }
2322*4882a593Smuzhiyun
snd_bbfpro_ctl_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)2323*4882a593Smuzhiyun static int snd_bbfpro_ctl_put(struct snd_kcontrol *kcontrol,
2324*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
2325*4882a593Smuzhiyun {
2326*4882a593Smuzhiyun int err;
2327*4882a593Smuzhiyun u8 reg, idx;
2328*4882a593Smuzhiyun int old_value, pv, val;
2329*4882a593Smuzhiyun
2330*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
2331*4882a593Smuzhiyun struct usb_mixer_interface *mixer = list->mixer;
2332*4882a593Smuzhiyun
2333*4882a593Smuzhiyun pv = kcontrol->private_value;
2334*4882a593Smuzhiyun reg = pv & SND_BBFPRO_CTL_REG_MASK;
2335*4882a593Smuzhiyun idx = (pv >> SND_BBFPRO_CTL_IDX_SHIFT) & SND_BBFPRO_CTL_IDX_MASK;
2336*4882a593Smuzhiyun old_value = (pv >> SND_BBFPRO_CTL_VAL_SHIFT) & SND_BBFPRO_CTL_VAL_MASK;
2337*4882a593Smuzhiyun
2338*4882a593Smuzhiyun if ((reg == SND_BBFPRO_CTL_REG1 &&
2339*4882a593Smuzhiyun idx == SND_BBFPRO_CTL_REG1_CLK_OPTICAL) ||
2340*4882a593Smuzhiyun (reg == SND_BBFPRO_CTL_REG2 &&
2341*4882a593Smuzhiyun (idx == SND_BBFPRO_CTL_REG2_SENS_IN3 ||
2342*4882a593Smuzhiyun idx == SND_BBFPRO_CTL_REG2_SENS_IN4))) {
2343*4882a593Smuzhiyun val = ucontrol->value.enumerated.item[0];
2344*4882a593Smuzhiyun } else {
2345*4882a593Smuzhiyun val = ucontrol->value.integer.value[0];
2346*4882a593Smuzhiyun }
2347*4882a593Smuzhiyun
2348*4882a593Smuzhiyun if (val > 1)
2349*4882a593Smuzhiyun return -EINVAL;
2350*4882a593Smuzhiyun
2351*4882a593Smuzhiyun if (val == old_value)
2352*4882a593Smuzhiyun return 0;
2353*4882a593Smuzhiyun
2354*4882a593Smuzhiyun kcontrol->private_value = reg
2355*4882a593Smuzhiyun | ((idx & SND_BBFPRO_CTL_IDX_MASK) << SND_BBFPRO_CTL_IDX_SHIFT)
2356*4882a593Smuzhiyun | ((val & SND_BBFPRO_CTL_VAL_MASK) << SND_BBFPRO_CTL_VAL_SHIFT);
2357*4882a593Smuzhiyun
2358*4882a593Smuzhiyun err = snd_bbfpro_ctl_update(mixer, reg, idx, val);
2359*4882a593Smuzhiyun return err < 0 ? err : 1;
2360*4882a593Smuzhiyun }
2361*4882a593Smuzhiyun
snd_bbfpro_ctl_resume(struct usb_mixer_elem_list * list)2362*4882a593Smuzhiyun static int snd_bbfpro_ctl_resume(struct usb_mixer_elem_list *list)
2363*4882a593Smuzhiyun {
2364*4882a593Smuzhiyun u8 reg, idx;
2365*4882a593Smuzhiyun int value, pv;
2366*4882a593Smuzhiyun
2367*4882a593Smuzhiyun pv = list->kctl->private_value;
2368*4882a593Smuzhiyun reg = pv & SND_BBFPRO_CTL_REG_MASK;
2369*4882a593Smuzhiyun idx = (pv >> SND_BBFPRO_CTL_IDX_SHIFT) & SND_BBFPRO_CTL_IDX_MASK;
2370*4882a593Smuzhiyun value = (pv >> SND_BBFPRO_CTL_VAL_SHIFT) & SND_BBFPRO_CTL_VAL_MASK;
2371*4882a593Smuzhiyun
2372*4882a593Smuzhiyun return snd_bbfpro_ctl_update(list->mixer, reg, idx, value);
2373*4882a593Smuzhiyun }
2374*4882a593Smuzhiyun
snd_bbfpro_vol_update(struct usb_mixer_interface * mixer,u16 index,u32 value)2375*4882a593Smuzhiyun static int snd_bbfpro_vol_update(struct usb_mixer_interface *mixer, u16 index,
2376*4882a593Smuzhiyun u32 value)
2377*4882a593Smuzhiyun {
2378*4882a593Smuzhiyun struct snd_usb_audio *chip = mixer->chip;
2379*4882a593Smuzhiyun int err;
2380*4882a593Smuzhiyun u16 idx;
2381*4882a593Smuzhiyun u16 usb_idx, usb_val;
2382*4882a593Smuzhiyun u32 v;
2383*4882a593Smuzhiyun
2384*4882a593Smuzhiyun err = snd_usb_lock_shutdown(chip);
2385*4882a593Smuzhiyun if (err < 0)
2386*4882a593Smuzhiyun return err;
2387*4882a593Smuzhiyun
2388*4882a593Smuzhiyun idx = index & SND_BBFPRO_MIXER_IDX_MASK;
2389*4882a593Smuzhiyun // 18 bit linear volume, split so 2 bits end up in index.
2390*4882a593Smuzhiyun v = value & SND_BBFPRO_MIXER_VAL_MASK;
2391*4882a593Smuzhiyun usb_idx = idx | (v & 0x3) << 14;
2392*4882a593Smuzhiyun usb_val = (v >> 2) & 0xffff;
2393*4882a593Smuzhiyun
2394*4882a593Smuzhiyun err = snd_usb_ctl_msg(chip->dev,
2395*4882a593Smuzhiyun usb_sndctrlpipe(chip->dev, 0),
2396*4882a593Smuzhiyun SND_BBFPRO_USBREQ_MIXER,
2397*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR |
2398*4882a593Smuzhiyun USB_RECIP_DEVICE,
2399*4882a593Smuzhiyun usb_val, usb_idx, NULL, 0);
2400*4882a593Smuzhiyun
2401*4882a593Smuzhiyun snd_usb_unlock_shutdown(chip);
2402*4882a593Smuzhiyun return err;
2403*4882a593Smuzhiyun }
2404*4882a593Smuzhiyun
snd_bbfpro_vol_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)2405*4882a593Smuzhiyun static int snd_bbfpro_vol_get(struct snd_kcontrol *kcontrol,
2406*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
2407*4882a593Smuzhiyun {
2408*4882a593Smuzhiyun ucontrol->value.integer.value[0] =
2409*4882a593Smuzhiyun kcontrol->private_value >> SND_BBFPRO_MIXER_VAL_SHIFT;
2410*4882a593Smuzhiyun return 0;
2411*4882a593Smuzhiyun }
2412*4882a593Smuzhiyun
snd_bbfpro_vol_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)2413*4882a593Smuzhiyun static int snd_bbfpro_vol_info(struct snd_kcontrol *kcontrol,
2414*4882a593Smuzhiyun struct snd_ctl_elem_info *uinfo)
2415*4882a593Smuzhiyun {
2416*4882a593Smuzhiyun uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
2417*4882a593Smuzhiyun uinfo->count = 1;
2418*4882a593Smuzhiyun uinfo->value.integer.min = SND_BBFPRO_MIXER_VAL_MIN;
2419*4882a593Smuzhiyun uinfo->value.integer.max = SND_BBFPRO_MIXER_VAL_MAX;
2420*4882a593Smuzhiyun return 0;
2421*4882a593Smuzhiyun }
2422*4882a593Smuzhiyun
snd_bbfpro_vol_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)2423*4882a593Smuzhiyun static int snd_bbfpro_vol_put(struct snd_kcontrol *kcontrol,
2424*4882a593Smuzhiyun struct snd_ctl_elem_value *ucontrol)
2425*4882a593Smuzhiyun {
2426*4882a593Smuzhiyun int err;
2427*4882a593Smuzhiyun u16 idx;
2428*4882a593Smuzhiyun u32 new_val, old_value, uvalue;
2429*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kcontrol);
2430*4882a593Smuzhiyun struct usb_mixer_interface *mixer = list->mixer;
2431*4882a593Smuzhiyun
2432*4882a593Smuzhiyun uvalue = ucontrol->value.integer.value[0];
2433*4882a593Smuzhiyun idx = kcontrol->private_value & SND_BBFPRO_MIXER_IDX_MASK;
2434*4882a593Smuzhiyun old_value = kcontrol->private_value >> SND_BBFPRO_MIXER_VAL_SHIFT;
2435*4882a593Smuzhiyun
2436*4882a593Smuzhiyun if (uvalue > SND_BBFPRO_MIXER_VAL_MAX)
2437*4882a593Smuzhiyun return -EINVAL;
2438*4882a593Smuzhiyun
2439*4882a593Smuzhiyun if (uvalue == old_value)
2440*4882a593Smuzhiyun return 0;
2441*4882a593Smuzhiyun
2442*4882a593Smuzhiyun new_val = uvalue & SND_BBFPRO_MIXER_VAL_MASK;
2443*4882a593Smuzhiyun
2444*4882a593Smuzhiyun kcontrol->private_value = idx
2445*4882a593Smuzhiyun | (new_val << SND_BBFPRO_MIXER_VAL_SHIFT);
2446*4882a593Smuzhiyun
2447*4882a593Smuzhiyun err = snd_bbfpro_vol_update(mixer, idx, new_val);
2448*4882a593Smuzhiyun return err < 0 ? err : 1;
2449*4882a593Smuzhiyun }
2450*4882a593Smuzhiyun
snd_bbfpro_vol_resume(struct usb_mixer_elem_list * list)2451*4882a593Smuzhiyun static int snd_bbfpro_vol_resume(struct usb_mixer_elem_list *list)
2452*4882a593Smuzhiyun {
2453*4882a593Smuzhiyun int pv = list->kctl->private_value;
2454*4882a593Smuzhiyun u16 idx = pv & SND_BBFPRO_MIXER_IDX_MASK;
2455*4882a593Smuzhiyun u32 val = (pv >> SND_BBFPRO_MIXER_VAL_SHIFT)
2456*4882a593Smuzhiyun & SND_BBFPRO_MIXER_VAL_MASK;
2457*4882a593Smuzhiyun return snd_bbfpro_vol_update(list->mixer, idx, val);
2458*4882a593Smuzhiyun }
2459*4882a593Smuzhiyun
2460*4882a593Smuzhiyun // Predfine elements
2461*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_bbfpro_ctl_control = {
2462*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2463*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
2464*4882a593Smuzhiyun .index = 0,
2465*4882a593Smuzhiyun .info = snd_bbfpro_ctl_info,
2466*4882a593Smuzhiyun .get = snd_bbfpro_ctl_get,
2467*4882a593Smuzhiyun .put = snd_bbfpro_ctl_put
2468*4882a593Smuzhiyun };
2469*4882a593Smuzhiyun
2470*4882a593Smuzhiyun static const struct snd_kcontrol_new snd_bbfpro_vol_control = {
2471*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2472*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
2473*4882a593Smuzhiyun .index = 0,
2474*4882a593Smuzhiyun .info = snd_bbfpro_vol_info,
2475*4882a593Smuzhiyun .get = snd_bbfpro_vol_get,
2476*4882a593Smuzhiyun .put = snd_bbfpro_vol_put
2477*4882a593Smuzhiyun };
2478*4882a593Smuzhiyun
snd_bbfpro_ctl_add(struct usb_mixer_interface * mixer,u8 reg,u8 index,char * name)2479*4882a593Smuzhiyun static int snd_bbfpro_ctl_add(struct usb_mixer_interface *mixer, u8 reg,
2480*4882a593Smuzhiyun u8 index, char *name)
2481*4882a593Smuzhiyun {
2482*4882a593Smuzhiyun struct snd_kcontrol_new knew = snd_bbfpro_ctl_control;
2483*4882a593Smuzhiyun
2484*4882a593Smuzhiyun knew.name = name;
2485*4882a593Smuzhiyun knew.private_value = (reg & SND_BBFPRO_CTL_REG_MASK)
2486*4882a593Smuzhiyun | ((index & SND_BBFPRO_CTL_IDX_MASK)
2487*4882a593Smuzhiyun << SND_BBFPRO_CTL_IDX_SHIFT);
2488*4882a593Smuzhiyun
2489*4882a593Smuzhiyun return add_single_ctl_with_resume(mixer, 0, snd_bbfpro_ctl_resume,
2490*4882a593Smuzhiyun &knew, NULL);
2491*4882a593Smuzhiyun }
2492*4882a593Smuzhiyun
snd_bbfpro_vol_add(struct usb_mixer_interface * mixer,u16 index,char * name)2493*4882a593Smuzhiyun static int snd_bbfpro_vol_add(struct usb_mixer_interface *mixer, u16 index,
2494*4882a593Smuzhiyun char *name)
2495*4882a593Smuzhiyun {
2496*4882a593Smuzhiyun struct snd_kcontrol_new knew = snd_bbfpro_vol_control;
2497*4882a593Smuzhiyun
2498*4882a593Smuzhiyun knew.name = name;
2499*4882a593Smuzhiyun knew.private_value = index & SND_BBFPRO_MIXER_IDX_MASK;
2500*4882a593Smuzhiyun
2501*4882a593Smuzhiyun return add_single_ctl_with_resume(mixer, 0, snd_bbfpro_vol_resume,
2502*4882a593Smuzhiyun &knew, NULL);
2503*4882a593Smuzhiyun }
2504*4882a593Smuzhiyun
snd_bbfpro_controls_create(struct usb_mixer_interface * mixer)2505*4882a593Smuzhiyun static int snd_bbfpro_controls_create(struct usb_mixer_interface *mixer)
2506*4882a593Smuzhiyun {
2507*4882a593Smuzhiyun int err, i, o;
2508*4882a593Smuzhiyun char name[48];
2509*4882a593Smuzhiyun
2510*4882a593Smuzhiyun static const char * const input[] = {
2511*4882a593Smuzhiyun "AN1", "AN2", "IN3", "IN4", "AS1", "AS2", "ADAT3",
2512*4882a593Smuzhiyun "ADAT4", "ADAT5", "ADAT6", "ADAT7", "ADAT8"};
2513*4882a593Smuzhiyun
2514*4882a593Smuzhiyun static const char * const output[] = {
2515*4882a593Smuzhiyun "AN1", "AN2", "PH3", "PH4", "AS1", "AS2", "ADAT3", "ADAT4",
2516*4882a593Smuzhiyun "ADAT5", "ADAT6", "ADAT7", "ADAT8"};
2517*4882a593Smuzhiyun
2518*4882a593Smuzhiyun for (o = 0 ; o < 12 ; ++o) {
2519*4882a593Smuzhiyun for (i = 0 ; i < 12 ; ++i) {
2520*4882a593Smuzhiyun // Line routing
2521*4882a593Smuzhiyun snprintf(name, sizeof(name),
2522*4882a593Smuzhiyun "%s-%s-%s Playback Volume",
2523*4882a593Smuzhiyun (i < 2 ? "Mic" : "Line"),
2524*4882a593Smuzhiyun input[i], output[o]);
2525*4882a593Smuzhiyun err = snd_bbfpro_vol_add(mixer, (26 * o + i), name);
2526*4882a593Smuzhiyun if (err < 0)
2527*4882a593Smuzhiyun return err;
2528*4882a593Smuzhiyun
2529*4882a593Smuzhiyun // PCM routing... yes, it is output remapping
2530*4882a593Smuzhiyun snprintf(name, sizeof(name),
2531*4882a593Smuzhiyun "PCM-%s-%s Playback Volume",
2532*4882a593Smuzhiyun output[i], output[o]);
2533*4882a593Smuzhiyun err = snd_bbfpro_vol_add(mixer, (26 * o + 12 + i),
2534*4882a593Smuzhiyun name);
2535*4882a593Smuzhiyun if (err < 0)
2536*4882a593Smuzhiyun return err;
2537*4882a593Smuzhiyun }
2538*4882a593Smuzhiyun }
2539*4882a593Smuzhiyun
2540*4882a593Smuzhiyun // Control Reg 1
2541*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG1,
2542*4882a593Smuzhiyun SND_BBFPRO_CTL_REG1_CLK_OPTICAL,
2543*4882a593Smuzhiyun "Sample Clock Source");
2544*4882a593Smuzhiyun if (err < 0)
2545*4882a593Smuzhiyun return err;
2546*4882a593Smuzhiyun
2547*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG1,
2548*4882a593Smuzhiyun SND_BBFPRO_CTL_REG1_SPDIF_PRO,
2549*4882a593Smuzhiyun "IEC958 Pro Mask");
2550*4882a593Smuzhiyun if (err < 0)
2551*4882a593Smuzhiyun return err;
2552*4882a593Smuzhiyun
2553*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG1,
2554*4882a593Smuzhiyun SND_BBFPRO_CTL_REG1_SPDIF_EMPH,
2555*4882a593Smuzhiyun "IEC958 Emphasis");
2556*4882a593Smuzhiyun if (err < 0)
2557*4882a593Smuzhiyun return err;
2558*4882a593Smuzhiyun
2559*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG1,
2560*4882a593Smuzhiyun SND_BBFPRO_CTL_REG1_SPDIF_OPTICAL,
2561*4882a593Smuzhiyun "IEC958 Switch");
2562*4882a593Smuzhiyun if (err < 0)
2563*4882a593Smuzhiyun return err;
2564*4882a593Smuzhiyun
2565*4882a593Smuzhiyun // Control Reg 2
2566*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG2,
2567*4882a593Smuzhiyun SND_BBFPRO_CTL_REG2_48V_AN1,
2568*4882a593Smuzhiyun "Mic-AN1 48V");
2569*4882a593Smuzhiyun if (err < 0)
2570*4882a593Smuzhiyun return err;
2571*4882a593Smuzhiyun
2572*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG2,
2573*4882a593Smuzhiyun SND_BBFPRO_CTL_REG2_48V_AN2,
2574*4882a593Smuzhiyun "Mic-AN2 48V");
2575*4882a593Smuzhiyun if (err < 0)
2576*4882a593Smuzhiyun return err;
2577*4882a593Smuzhiyun
2578*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG2,
2579*4882a593Smuzhiyun SND_BBFPRO_CTL_REG2_SENS_IN3,
2580*4882a593Smuzhiyun "Line-IN3 Sens.");
2581*4882a593Smuzhiyun if (err < 0)
2582*4882a593Smuzhiyun return err;
2583*4882a593Smuzhiyun
2584*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG2,
2585*4882a593Smuzhiyun SND_BBFPRO_CTL_REG2_SENS_IN4,
2586*4882a593Smuzhiyun "Line-IN4 Sens.");
2587*4882a593Smuzhiyun if (err < 0)
2588*4882a593Smuzhiyun return err;
2589*4882a593Smuzhiyun
2590*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG2,
2591*4882a593Smuzhiyun SND_BBFPRO_CTL_REG2_PAD_AN1,
2592*4882a593Smuzhiyun "Mic-AN1 PAD");
2593*4882a593Smuzhiyun if (err < 0)
2594*4882a593Smuzhiyun return err;
2595*4882a593Smuzhiyun
2596*4882a593Smuzhiyun err = snd_bbfpro_ctl_add(mixer, SND_BBFPRO_CTL_REG2,
2597*4882a593Smuzhiyun SND_BBFPRO_CTL_REG2_PAD_AN2,
2598*4882a593Smuzhiyun "Mic-AN2 PAD");
2599*4882a593Smuzhiyun if (err < 0)
2600*4882a593Smuzhiyun return err;
2601*4882a593Smuzhiyun
2602*4882a593Smuzhiyun return 0;
2603*4882a593Smuzhiyun }
2604*4882a593Smuzhiyun
2605*4882a593Smuzhiyun /*
2606*4882a593Smuzhiyun * Pioneer DJ DJM Mixers
2607*4882a593Smuzhiyun *
2608*4882a593Smuzhiyun * These devices generally have options for soft-switching the playback and
2609*4882a593Smuzhiyun * capture sources in addition to the recording level. Although different
2610*4882a593Smuzhiyun * devices have different configurations, there seems to be canonical values
2611*4882a593Smuzhiyun * for specific capture/playback types: See the definitions of these below.
2612*4882a593Smuzhiyun *
2613*4882a593Smuzhiyun * The wValue is masked with the stereo channel number. e.g. Setting Ch2 to
2614*4882a593Smuzhiyun * capture phono would be 0x0203. Capture, playback and capture level have
2615*4882a593Smuzhiyun * different wIndexes.
2616*4882a593Smuzhiyun */
2617*4882a593Smuzhiyun
2618*4882a593Smuzhiyun // Capture types
2619*4882a593Smuzhiyun #define SND_DJM_CAP_LINE 0x00
2620*4882a593Smuzhiyun #define SND_DJM_CAP_CDLINE 0x01
2621*4882a593Smuzhiyun #define SND_DJM_CAP_DIGITAL 0x02
2622*4882a593Smuzhiyun #define SND_DJM_CAP_PHONO 0x03
2623*4882a593Smuzhiyun #define SND_DJM_CAP_PFADER 0x06
2624*4882a593Smuzhiyun #define SND_DJM_CAP_XFADERA 0x07
2625*4882a593Smuzhiyun #define SND_DJM_CAP_XFADERB 0x08
2626*4882a593Smuzhiyun #define SND_DJM_CAP_MIC 0x09
2627*4882a593Smuzhiyun #define SND_DJM_CAP_AUX 0x0d
2628*4882a593Smuzhiyun #define SND_DJM_CAP_RECOUT 0x0a
2629*4882a593Smuzhiyun #define SND_DJM_CAP_NONE 0x0f
2630*4882a593Smuzhiyun #define SND_DJM_CAP_CH1PFADER 0x11
2631*4882a593Smuzhiyun #define SND_DJM_CAP_CH2PFADER 0x12
2632*4882a593Smuzhiyun #define SND_DJM_CAP_CH3PFADER 0x13
2633*4882a593Smuzhiyun #define SND_DJM_CAP_CH4PFADER 0x14
2634*4882a593Smuzhiyun
2635*4882a593Smuzhiyun // Playback types
2636*4882a593Smuzhiyun #define SND_DJM_PB_CH1 0x00
2637*4882a593Smuzhiyun #define SND_DJM_PB_CH2 0x01
2638*4882a593Smuzhiyun #define SND_DJM_PB_AUX 0x04
2639*4882a593Smuzhiyun
2640*4882a593Smuzhiyun #define SND_DJM_WINDEX_CAP 0x8002
2641*4882a593Smuzhiyun #define SND_DJM_WINDEX_CAPLVL 0x8003
2642*4882a593Smuzhiyun #define SND_DJM_WINDEX_PB 0x8016
2643*4882a593Smuzhiyun
2644*4882a593Smuzhiyun // kcontrol->private_value layout
2645*4882a593Smuzhiyun #define SND_DJM_VALUE_MASK 0x0000ffff
2646*4882a593Smuzhiyun #define SND_DJM_GROUP_MASK 0x00ff0000
2647*4882a593Smuzhiyun #define SND_DJM_DEVICE_MASK 0xff000000
2648*4882a593Smuzhiyun #define SND_DJM_GROUP_SHIFT 16
2649*4882a593Smuzhiyun #define SND_DJM_DEVICE_SHIFT 24
2650*4882a593Smuzhiyun
2651*4882a593Smuzhiyun // device table index
2652*4882a593Smuzhiyun #define SND_DJM_250MK2_IDX 0x0
2653*4882a593Smuzhiyun #define SND_DJM_750_IDX 0x1
2654*4882a593Smuzhiyun #define SND_DJM_900NXS2_IDX 0x2
2655*4882a593Smuzhiyun
2656*4882a593Smuzhiyun
2657*4882a593Smuzhiyun #define SND_DJM_CTL(_name, suffix, _default_value, _windex) { \
2658*4882a593Smuzhiyun .name = _name, \
2659*4882a593Smuzhiyun .options = snd_djm_opts_##suffix, \
2660*4882a593Smuzhiyun .noptions = ARRAY_SIZE(snd_djm_opts_##suffix), \
2661*4882a593Smuzhiyun .default_value = _default_value, \
2662*4882a593Smuzhiyun .wIndex = _windex }
2663*4882a593Smuzhiyun
2664*4882a593Smuzhiyun #define SND_DJM_DEVICE(suffix) { \
2665*4882a593Smuzhiyun .controls = snd_djm_ctls_##suffix, \
2666*4882a593Smuzhiyun .ncontrols = ARRAY_SIZE(snd_djm_ctls_##suffix) }
2667*4882a593Smuzhiyun
2668*4882a593Smuzhiyun
2669*4882a593Smuzhiyun struct snd_djm_device {
2670*4882a593Smuzhiyun const char *name;
2671*4882a593Smuzhiyun const struct snd_djm_ctl *controls;
2672*4882a593Smuzhiyun size_t ncontrols;
2673*4882a593Smuzhiyun };
2674*4882a593Smuzhiyun
2675*4882a593Smuzhiyun struct snd_djm_ctl {
2676*4882a593Smuzhiyun const char *name;
2677*4882a593Smuzhiyun const u16 *options;
2678*4882a593Smuzhiyun size_t noptions;
2679*4882a593Smuzhiyun u16 default_value;
2680*4882a593Smuzhiyun u16 wIndex;
2681*4882a593Smuzhiyun };
2682*4882a593Smuzhiyun
snd_djm_get_label_caplevel(u16 wvalue)2683*4882a593Smuzhiyun static const char *snd_djm_get_label_caplevel(u16 wvalue)
2684*4882a593Smuzhiyun {
2685*4882a593Smuzhiyun switch (wvalue) {
2686*4882a593Smuzhiyun case 0x0000: return "-19dB";
2687*4882a593Smuzhiyun case 0x0100: return "-15dB";
2688*4882a593Smuzhiyun case 0x0200: return "-10dB";
2689*4882a593Smuzhiyun case 0x0300: return "-5dB";
2690*4882a593Smuzhiyun default: return NULL;
2691*4882a593Smuzhiyun }
2692*4882a593Smuzhiyun };
2693*4882a593Smuzhiyun
snd_djm_get_label_cap(u16 wvalue)2694*4882a593Smuzhiyun static const char *snd_djm_get_label_cap(u16 wvalue)
2695*4882a593Smuzhiyun {
2696*4882a593Smuzhiyun switch (wvalue & 0x00ff) {
2697*4882a593Smuzhiyun case SND_DJM_CAP_LINE: return "Control Tone LINE";
2698*4882a593Smuzhiyun case SND_DJM_CAP_CDLINE: return "Control Tone CD/LINE";
2699*4882a593Smuzhiyun case SND_DJM_CAP_DIGITAL: return "Control Tone DIGITAL";
2700*4882a593Smuzhiyun case SND_DJM_CAP_PHONO: return "Control Tone PHONO";
2701*4882a593Smuzhiyun case SND_DJM_CAP_PFADER: return "Post Fader";
2702*4882a593Smuzhiyun case SND_DJM_CAP_XFADERA: return "Cross Fader A";
2703*4882a593Smuzhiyun case SND_DJM_CAP_XFADERB: return "Cross Fader B";
2704*4882a593Smuzhiyun case SND_DJM_CAP_MIC: return "Mic";
2705*4882a593Smuzhiyun case SND_DJM_CAP_RECOUT: return "Rec Out";
2706*4882a593Smuzhiyun case SND_DJM_CAP_AUX: return "Aux";
2707*4882a593Smuzhiyun case SND_DJM_CAP_NONE: return "None";
2708*4882a593Smuzhiyun case SND_DJM_CAP_CH1PFADER: return "Post Fader Ch1";
2709*4882a593Smuzhiyun case SND_DJM_CAP_CH2PFADER: return "Post Fader Ch2";
2710*4882a593Smuzhiyun case SND_DJM_CAP_CH3PFADER: return "Post Fader Ch3";
2711*4882a593Smuzhiyun case SND_DJM_CAP_CH4PFADER: return "Post Fader Ch4";
2712*4882a593Smuzhiyun default: return NULL;
2713*4882a593Smuzhiyun }
2714*4882a593Smuzhiyun };
2715*4882a593Smuzhiyun
snd_djm_get_label_pb(u16 wvalue)2716*4882a593Smuzhiyun static const char *snd_djm_get_label_pb(u16 wvalue)
2717*4882a593Smuzhiyun {
2718*4882a593Smuzhiyun switch (wvalue & 0x00ff) {
2719*4882a593Smuzhiyun case SND_DJM_PB_CH1: return "Ch1";
2720*4882a593Smuzhiyun case SND_DJM_PB_CH2: return "Ch2";
2721*4882a593Smuzhiyun case SND_DJM_PB_AUX: return "Aux";
2722*4882a593Smuzhiyun default: return NULL;
2723*4882a593Smuzhiyun }
2724*4882a593Smuzhiyun };
2725*4882a593Smuzhiyun
snd_djm_get_label(u16 wvalue,u16 windex)2726*4882a593Smuzhiyun static const char *snd_djm_get_label(u16 wvalue, u16 windex)
2727*4882a593Smuzhiyun {
2728*4882a593Smuzhiyun switch (windex) {
2729*4882a593Smuzhiyun case SND_DJM_WINDEX_CAPLVL: return snd_djm_get_label_caplevel(wvalue);
2730*4882a593Smuzhiyun case SND_DJM_WINDEX_CAP: return snd_djm_get_label_cap(wvalue);
2731*4882a593Smuzhiyun case SND_DJM_WINDEX_PB: return snd_djm_get_label_pb(wvalue);
2732*4882a593Smuzhiyun default: return NULL;
2733*4882a593Smuzhiyun }
2734*4882a593Smuzhiyun };
2735*4882a593Smuzhiyun
2736*4882a593Smuzhiyun
2737*4882a593Smuzhiyun // DJM-250MK2
2738*4882a593Smuzhiyun static const u16 snd_djm_opts_cap_level[] = {
2739*4882a593Smuzhiyun 0x0000, 0x0100, 0x0200, 0x0300 };
2740*4882a593Smuzhiyun
2741*4882a593Smuzhiyun static const u16 snd_djm_opts_250mk2_cap1[] = {
2742*4882a593Smuzhiyun 0x0103, 0x0100, 0x0106, 0x0107, 0x0108, 0x0109, 0x010d, 0x010a };
2743*4882a593Smuzhiyun
2744*4882a593Smuzhiyun static const u16 snd_djm_opts_250mk2_cap2[] = {
2745*4882a593Smuzhiyun 0x0203, 0x0200, 0x0206, 0x0207, 0x0208, 0x0209, 0x020d, 0x020a };
2746*4882a593Smuzhiyun
2747*4882a593Smuzhiyun static const u16 snd_djm_opts_250mk2_cap3[] = {
2748*4882a593Smuzhiyun 0x030a, 0x0311, 0x0312, 0x0307, 0x0308, 0x0309, 0x030d };
2749*4882a593Smuzhiyun
2750*4882a593Smuzhiyun static const u16 snd_djm_opts_250mk2_pb1[] = { 0x0100, 0x0101, 0x0104 };
2751*4882a593Smuzhiyun static const u16 snd_djm_opts_250mk2_pb2[] = { 0x0200, 0x0201, 0x0204 };
2752*4882a593Smuzhiyun static const u16 snd_djm_opts_250mk2_pb3[] = { 0x0300, 0x0301, 0x0304 };
2753*4882a593Smuzhiyun
2754*4882a593Smuzhiyun static const struct snd_djm_ctl snd_djm_ctls_250mk2[] = {
2755*4882a593Smuzhiyun SND_DJM_CTL("Capture Level", cap_level, 0, SND_DJM_WINDEX_CAPLVL),
2756*4882a593Smuzhiyun SND_DJM_CTL("Ch1 Input", 250mk2_cap1, 2, SND_DJM_WINDEX_CAP),
2757*4882a593Smuzhiyun SND_DJM_CTL("Ch2 Input", 250mk2_cap2, 2, SND_DJM_WINDEX_CAP),
2758*4882a593Smuzhiyun SND_DJM_CTL("Ch3 Input", 250mk2_cap3, 0, SND_DJM_WINDEX_CAP),
2759*4882a593Smuzhiyun SND_DJM_CTL("Ch1 Output", 250mk2_pb1, 0, SND_DJM_WINDEX_PB),
2760*4882a593Smuzhiyun SND_DJM_CTL("Ch2 Output", 250mk2_pb2, 1, SND_DJM_WINDEX_PB),
2761*4882a593Smuzhiyun SND_DJM_CTL("Ch3 Output", 250mk2_pb3, 2, SND_DJM_WINDEX_PB)
2762*4882a593Smuzhiyun };
2763*4882a593Smuzhiyun
2764*4882a593Smuzhiyun
2765*4882a593Smuzhiyun // DJM-750
2766*4882a593Smuzhiyun static const u16 snd_djm_opts_750_cap1[] = {
2767*4882a593Smuzhiyun 0x0101, 0x0103, 0x0106, 0x0107, 0x0108, 0x0109, 0x010a, 0x010f };
2768*4882a593Smuzhiyun static const u16 snd_djm_opts_750_cap2[] = {
2769*4882a593Smuzhiyun 0x0200, 0x0201, 0x0206, 0x0207, 0x0208, 0x0209, 0x020a, 0x020f };
2770*4882a593Smuzhiyun static const u16 snd_djm_opts_750_cap3[] = {
2771*4882a593Smuzhiyun 0x0300, 0x0301, 0x0306, 0x0307, 0x0308, 0x0309, 0x030a, 0x030f };
2772*4882a593Smuzhiyun static const u16 snd_djm_opts_750_cap4[] = {
2773*4882a593Smuzhiyun 0x0401, 0x0403, 0x0406, 0x0407, 0x0408, 0x0409, 0x040a, 0x040f };
2774*4882a593Smuzhiyun
2775*4882a593Smuzhiyun static const struct snd_djm_ctl snd_djm_ctls_750[] = {
2776*4882a593Smuzhiyun SND_DJM_CTL("Capture Level", cap_level, 0, SND_DJM_WINDEX_CAPLVL),
2777*4882a593Smuzhiyun SND_DJM_CTL("Ch1 Input", 750_cap1, 2, SND_DJM_WINDEX_CAP),
2778*4882a593Smuzhiyun SND_DJM_CTL("Ch2 Input", 750_cap2, 2, SND_DJM_WINDEX_CAP),
2779*4882a593Smuzhiyun SND_DJM_CTL("Ch3 Input", 750_cap3, 0, SND_DJM_WINDEX_CAP),
2780*4882a593Smuzhiyun SND_DJM_CTL("Ch4 Input", 750_cap4, 0, SND_DJM_WINDEX_CAP)
2781*4882a593Smuzhiyun };
2782*4882a593Smuzhiyun
2783*4882a593Smuzhiyun
2784*4882a593Smuzhiyun // DJM-900NXS2
2785*4882a593Smuzhiyun static const u16 snd_djm_opts_900nxs2_cap1[] = {
2786*4882a593Smuzhiyun 0x0100, 0x0102, 0x0103, 0x0106, 0x0107, 0x0108, 0x0109, 0x010a };
2787*4882a593Smuzhiyun static const u16 snd_djm_opts_900nxs2_cap2[] = {
2788*4882a593Smuzhiyun 0x0200, 0x0202, 0x0203, 0x0206, 0x0207, 0x0208, 0x0209, 0x020a };
2789*4882a593Smuzhiyun static const u16 snd_djm_opts_900nxs2_cap3[] = {
2790*4882a593Smuzhiyun 0x0300, 0x0302, 0x0303, 0x0306, 0x0307, 0x0308, 0x0309, 0x030a };
2791*4882a593Smuzhiyun static const u16 snd_djm_opts_900nxs2_cap4[] = {
2792*4882a593Smuzhiyun 0x0400, 0x0402, 0x0403, 0x0406, 0x0407, 0x0408, 0x0409, 0x040a };
2793*4882a593Smuzhiyun static const u16 snd_djm_opts_900nxs2_cap5[] = {
2794*4882a593Smuzhiyun 0x0507, 0x0508, 0x0509, 0x050a, 0x0511, 0x0512, 0x0513, 0x0514 };
2795*4882a593Smuzhiyun
2796*4882a593Smuzhiyun static const struct snd_djm_ctl snd_djm_ctls_900nxs2[] = {
2797*4882a593Smuzhiyun SND_DJM_CTL("Capture Level", cap_level, 0, SND_DJM_WINDEX_CAPLVL),
2798*4882a593Smuzhiyun SND_DJM_CTL("Ch1 Input", 900nxs2_cap1, 2, SND_DJM_WINDEX_CAP),
2799*4882a593Smuzhiyun SND_DJM_CTL("Ch2 Input", 900nxs2_cap2, 2, SND_DJM_WINDEX_CAP),
2800*4882a593Smuzhiyun SND_DJM_CTL("Ch3 Input", 900nxs2_cap3, 2, SND_DJM_WINDEX_CAP),
2801*4882a593Smuzhiyun SND_DJM_CTL("Ch4 Input", 900nxs2_cap4, 2, SND_DJM_WINDEX_CAP),
2802*4882a593Smuzhiyun SND_DJM_CTL("Ch5 Input", 900nxs2_cap5, 3, SND_DJM_WINDEX_CAP)
2803*4882a593Smuzhiyun };
2804*4882a593Smuzhiyun
2805*4882a593Smuzhiyun
2806*4882a593Smuzhiyun static const struct snd_djm_device snd_djm_devices[] = {
2807*4882a593Smuzhiyun SND_DJM_DEVICE(250mk2),
2808*4882a593Smuzhiyun SND_DJM_DEVICE(750),
2809*4882a593Smuzhiyun SND_DJM_DEVICE(900nxs2)
2810*4882a593Smuzhiyun };
2811*4882a593Smuzhiyun
2812*4882a593Smuzhiyun
snd_djm_controls_info(struct snd_kcontrol * kctl,struct snd_ctl_elem_info * info)2813*4882a593Smuzhiyun static int snd_djm_controls_info(struct snd_kcontrol *kctl,
2814*4882a593Smuzhiyun struct snd_ctl_elem_info *info)
2815*4882a593Smuzhiyun {
2816*4882a593Smuzhiyun unsigned long private_value = kctl->private_value;
2817*4882a593Smuzhiyun u8 device_idx = (private_value & SND_DJM_DEVICE_MASK) >> SND_DJM_DEVICE_SHIFT;
2818*4882a593Smuzhiyun u8 ctl_idx = (private_value & SND_DJM_GROUP_MASK) >> SND_DJM_GROUP_SHIFT;
2819*4882a593Smuzhiyun const struct snd_djm_device *device = &snd_djm_devices[device_idx];
2820*4882a593Smuzhiyun const char *name;
2821*4882a593Smuzhiyun const struct snd_djm_ctl *ctl;
2822*4882a593Smuzhiyun size_t noptions;
2823*4882a593Smuzhiyun
2824*4882a593Smuzhiyun if (ctl_idx >= device->ncontrols)
2825*4882a593Smuzhiyun return -EINVAL;
2826*4882a593Smuzhiyun
2827*4882a593Smuzhiyun ctl = &device->controls[ctl_idx];
2828*4882a593Smuzhiyun noptions = ctl->noptions;
2829*4882a593Smuzhiyun if (info->value.enumerated.item >= noptions)
2830*4882a593Smuzhiyun info->value.enumerated.item = noptions - 1;
2831*4882a593Smuzhiyun
2832*4882a593Smuzhiyun name = snd_djm_get_label(ctl->options[info->value.enumerated.item],
2833*4882a593Smuzhiyun ctl->wIndex);
2834*4882a593Smuzhiyun if (!name)
2835*4882a593Smuzhiyun return -EINVAL;
2836*4882a593Smuzhiyun
2837*4882a593Smuzhiyun strlcpy(info->value.enumerated.name, name, sizeof(info->value.enumerated.name));
2838*4882a593Smuzhiyun info->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
2839*4882a593Smuzhiyun info->count = 1;
2840*4882a593Smuzhiyun info->value.enumerated.items = noptions;
2841*4882a593Smuzhiyun return 0;
2842*4882a593Smuzhiyun }
2843*4882a593Smuzhiyun
snd_djm_controls_update(struct usb_mixer_interface * mixer,u8 device_idx,u8 group,u16 value)2844*4882a593Smuzhiyun static int snd_djm_controls_update(struct usb_mixer_interface *mixer,
2845*4882a593Smuzhiyun u8 device_idx, u8 group, u16 value)
2846*4882a593Smuzhiyun {
2847*4882a593Smuzhiyun int err;
2848*4882a593Smuzhiyun const struct snd_djm_device *device = &snd_djm_devices[device_idx];
2849*4882a593Smuzhiyun
2850*4882a593Smuzhiyun if ((group >= device->ncontrols) || value >= device->controls[group].noptions)
2851*4882a593Smuzhiyun return -EINVAL;
2852*4882a593Smuzhiyun
2853*4882a593Smuzhiyun err = snd_usb_lock_shutdown(mixer->chip);
2854*4882a593Smuzhiyun if (err)
2855*4882a593Smuzhiyun return err;
2856*4882a593Smuzhiyun
2857*4882a593Smuzhiyun err = snd_usb_ctl_msg(
2858*4882a593Smuzhiyun mixer->chip->dev, usb_sndctrlpipe(mixer->chip->dev, 0),
2859*4882a593Smuzhiyun USB_REQ_SET_FEATURE,
2860*4882a593Smuzhiyun USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
2861*4882a593Smuzhiyun device->controls[group].options[value],
2862*4882a593Smuzhiyun device->controls[group].wIndex,
2863*4882a593Smuzhiyun NULL, 0);
2864*4882a593Smuzhiyun
2865*4882a593Smuzhiyun snd_usb_unlock_shutdown(mixer->chip);
2866*4882a593Smuzhiyun return err;
2867*4882a593Smuzhiyun }
2868*4882a593Smuzhiyun
snd_djm_controls_get(struct snd_kcontrol * kctl,struct snd_ctl_elem_value * elem)2869*4882a593Smuzhiyun static int snd_djm_controls_get(struct snd_kcontrol *kctl,
2870*4882a593Smuzhiyun struct snd_ctl_elem_value *elem)
2871*4882a593Smuzhiyun {
2872*4882a593Smuzhiyun elem->value.enumerated.item[0] = kctl->private_value & SND_DJM_VALUE_MASK;
2873*4882a593Smuzhiyun return 0;
2874*4882a593Smuzhiyun }
2875*4882a593Smuzhiyun
snd_djm_controls_put(struct snd_kcontrol * kctl,struct snd_ctl_elem_value * elem)2876*4882a593Smuzhiyun static int snd_djm_controls_put(struct snd_kcontrol *kctl, struct snd_ctl_elem_value *elem)
2877*4882a593Smuzhiyun {
2878*4882a593Smuzhiyun struct usb_mixer_elem_list *list = snd_kcontrol_chip(kctl);
2879*4882a593Smuzhiyun struct usb_mixer_interface *mixer = list->mixer;
2880*4882a593Smuzhiyun unsigned long private_value = kctl->private_value;
2881*4882a593Smuzhiyun
2882*4882a593Smuzhiyun u8 device = (private_value & SND_DJM_DEVICE_MASK) >> SND_DJM_DEVICE_SHIFT;
2883*4882a593Smuzhiyun u8 group = (private_value & SND_DJM_GROUP_MASK) >> SND_DJM_GROUP_SHIFT;
2884*4882a593Smuzhiyun u16 value = elem->value.enumerated.item[0];
2885*4882a593Smuzhiyun
2886*4882a593Smuzhiyun kctl->private_value = (((unsigned long)device << SND_DJM_DEVICE_SHIFT) |
2887*4882a593Smuzhiyun (group << SND_DJM_GROUP_SHIFT) |
2888*4882a593Smuzhiyun value);
2889*4882a593Smuzhiyun
2890*4882a593Smuzhiyun return snd_djm_controls_update(mixer, device, group, value);
2891*4882a593Smuzhiyun }
2892*4882a593Smuzhiyun
snd_djm_controls_resume(struct usb_mixer_elem_list * list)2893*4882a593Smuzhiyun static int snd_djm_controls_resume(struct usb_mixer_elem_list *list)
2894*4882a593Smuzhiyun {
2895*4882a593Smuzhiyun unsigned long private_value = list->kctl->private_value;
2896*4882a593Smuzhiyun u8 device = (private_value & SND_DJM_DEVICE_MASK) >> SND_DJM_DEVICE_SHIFT;
2897*4882a593Smuzhiyun u8 group = (private_value & SND_DJM_GROUP_MASK) >> SND_DJM_GROUP_SHIFT;
2898*4882a593Smuzhiyun u16 value = (private_value & SND_DJM_VALUE_MASK);
2899*4882a593Smuzhiyun
2900*4882a593Smuzhiyun return snd_djm_controls_update(list->mixer, device, group, value);
2901*4882a593Smuzhiyun }
2902*4882a593Smuzhiyun
snd_djm_controls_create(struct usb_mixer_interface * mixer,const u8 device_idx)2903*4882a593Smuzhiyun static int snd_djm_controls_create(struct usb_mixer_interface *mixer,
2904*4882a593Smuzhiyun const u8 device_idx)
2905*4882a593Smuzhiyun {
2906*4882a593Smuzhiyun int err, i;
2907*4882a593Smuzhiyun u16 value;
2908*4882a593Smuzhiyun
2909*4882a593Smuzhiyun const struct snd_djm_device *device = &snd_djm_devices[device_idx];
2910*4882a593Smuzhiyun
2911*4882a593Smuzhiyun struct snd_kcontrol_new knew = {
2912*4882a593Smuzhiyun .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2913*4882a593Smuzhiyun .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
2914*4882a593Smuzhiyun .index = 0,
2915*4882a593Smuzhiyun .info = snd_djm_controls_info,
2916*4882a593Smuzhiyun .get = snd_djm_controls_get,
2917*4882a593Smuzhiyun .put = snd_djm_controls_put
2918*4882a593Smuzhiyun };
2919*4882a593Smuzhiyun
2920*4882a593Smuzhiyun for (i = 0; i < device->ncontrols; i++) {
2921*4882a593Smuzhiyun value = device->controls[i].default_value;
2922*4882a593Smuzhiyun knew.name = device->controls[i].name;
2923*4882a593Smuzhiyun knew.private_value = (
2924*4882a593Smuzhiyun ((unsigned long)device_idx << SND_DJM_DEVICE_SHIFT) |
2925*4882a593Smuzhiyun (i << SND_DJM_GROUP_SHIFT) |
2926*4882a593Smuzhiyun value);
2927*4882a593Smuzhiyun err = snd_djm_controls_update(mixer, device_idx, i, value);
2928*4882a593Smuzhiyun if (err)
2929*4882a593Smuzhiyun return err;
2930*4882a593Smuzhiyun err = add_single_ctl_with_resume(mixer, 0, snd_djm_controls_resume,
2931*4882a593Smuzhiyun &knew, NULL);
2932*4882a593Smuzhiyun if (err)
2933*4882a593Smuzhiyun return err;
2934*4882a593Smuzhiyun }
2935*4882a593Smuzhiyun return 0;
2936*4882a593Smuzhiyun }
2937*4882a593Smuzhiyun
snd_usb_mixer_apply_create_quirk(struct usb_mixer_interface * mixer)2938*4882a593Smuzhiyun int snd_usb_mixer_apply_create_quirk(struct usb_mixer_interface *mixer)
2939*4882a593Smuzhiyun {
2940*4882a593Smuzhiyun int err = 0;
2941*4882a593Smuzhiyun
2942*4882a593Smuzhiyun err = snd_usb_soundblaster_remote_init(mixer);
2943*4882a593Smuzhiyun if (err < 0)
2944*4882a593Smuzhiyun return err;
2945*4882a593Smuzhiyun
2946*4882a593Smuzhiyun switch (mixer->chip->usb_id) {
2947*4882a593Smuzhiyun /* Tascam US-16x08 */
2948*4882a593Smuzhiyun case USB_ID(0x0644, 0x8047):
2949*4882a593Smuzhiyun err = snd_us16x08_controls_create(mixer);
2950*4882a593Smuzhiyun break;
2951*4882a593Smuzhiyun case USB_ID(0x041e, 0x3020):
2952*4882a593Smuzhiyun case USB_ID(0x041e, 0x3040):
2953*4882a593Smuzhiyun case USB_ID(0x041e, 0x3042):
2954*4882a593Smuzhiyun case USB_ID(0x041e, 0x30df):
2955*4882a593Smuzhiyun case USB_ID(0x041e, 0x3048):
2956*4882a593Smuzhiyun err = snd_audigy2nx_controls_create(mixer);
2957*4882a593Smuzhiyun if (err < 0)
2958*4882a593Smuzhiyun break;
2959*4882a593Smuzhiyun snd_card_ro_proc_new(mixer->chip->card, "audigy2nx",
2960*4882a593Smuzhiyun mixer, snd_audigy2nx_proc_read);
2961*4882a593Smuzhiyun break;
2962*4882a593Smuzhiyun
2963*4882a593Smuzhiyun /* EMU0204 */
2964*4882a593Smuzhiyun case USB_ID(0x041e, 0x3f19):
2965*4882a593Smuzhiyun err = snd_emu0204_controls_create(mixer);
2966*4882a593Smuzhiyun break;
2967*4882a593Smuzhiyun
2968*4882a593Smuzhiyun case USB_ID(0x0763, 0x2030): /* M-Audio Fast Track C400 */
2969*4882a593Smuzhiyun case USB_ID(0x0763, 0x2031): /* M-Audio Fast Track C400 */
2970*4882a593Smuzhiyun err = snd_c400_create_mixer(mixer);
2971*4882a593Smuzhiyun break;
2972*4882a593Smuzhiyun
2973*4882a593Smuzhiyun case USB_ID(0x0763, 0x2080): /* M-Audio Fast Track Ultra */
2974*4882a593Smuzhiyun case USB_ID(0x0763, 0x2081): /* M-Audio Fast Track Ultra 8R */
2975*4882a593Smuzhiyun err = snd_ftu_create_mixer(mixer);
2976*4882a593Smuzhiyun break;
2977*4882a593Smuzhiyun
2978*4882a593Smuzhiyun case USB_ID(0x0b05, 0x1739): /* ASUS Xonar U1 */
2979*4882a593Smuzhiyun case USB_ID(0x0b05, 0x1743): /* ASUS Xonar U1 (2) */
2980*4882a593Smuzhiyun case USB_ID(0x0b05, 0x17a0): /* ASUS Xonar U3 */
2981*4882a593Smuzhiyun err = snd_xonar_u1_controls_create(mixer);
2982*4882a593Smuzhiyun break;
2983*4882a593Smuzhiyun
2984*4882a593Smuzhiyun case USB_ID(0x0d8c, 0x0103): /* Audio Advantage Micro II */
2985*4882a593Smuzhiyun err = snd_microii_controls_create(mixer);
2986*4882a593Smuzhiyun break;
2987*4882a593Smuzhiyun
2988*4882a593Smuzhiyun case USB_ID(0x0dba, 0x1000): /* Digidesign Mbox 1 */
2989*4882a593Smuzhiyun err = snd_mbox1_create_sync_switch(mixer);
2990*4882a593Smuzhiyun break;
2991*4882a593Smuzhiyun
2992*4882a593Smuzhiyun case USB_ID(0x17cc, 0x1011): /* Traktor Audio 6 */
2993*4882a593Smuzhiyun err = snd_nativeinstruments_create_mixer(mixer,
2994*4882a593Smuzhiyun snd_nativeinstruments_ta6_mixers,
2995*4882a593Smuzhiyun ARRAY_SIZE(snd_nativeinstruments_ta6_mixers));
2996*4882a593Smuzhiyun break;
2997*4882a593Smuzhiyun
2998*4882a593Smuzhiyun case USB_ID(0x17cc, 0x1021): /* Traktor Audio 10 */
2999*4882a593Smuzhiyun err = snd_nativeinstruments_create_mixer(mixer,
3000*4882a593Smuzhiyun snd_nativeinstruments_ta10_mixers,
3001*4882a593Smuzhiyun ARRAY_SIZE(snd_nativeinstruments_ta10_mixers));
3002*4882a593Smuzhiyun break;
3003*4882a593Smuzhiyun
3004*4882a593Smuzhiyun case USB_ID(0x200c, 0x1018): /* Electrix Ebox-44 */
3005*4882a593Smuzhiyun /* detection is disabled in mixer_maps.c */
3006*4882a593Smuzhiyun err = snd_create_std_mono_table(mixer, ebox44_table);
3007*4882a593Smuzhiyun break;
3008*4882a593Smuzhiyun
3009*4882a593Smuzhiyun case USB_ID(0x1235, 0x8012): /* Focusrite Scarlett 6i6 */
3010*4882a593Smuzhiyun case USB_ID(0x1235, 0x8002): /* Focusrite Scarlett 8i6 */
3011*4882a593Smuzhiyun case USB_ID(0x1235, 0x8004): /* Focusrite Scarlett 18i6 */
3012*4882a593Smuzhiyun case USB_ID(0x1235, 0x8014): /* Focusrite Scarlett 18i8 */
3013*4882a593Smuzhiyun case USB_ID(0x1235, 0x800c): /* Focusrite Scarlett 18i20 */
3014*4882a593Smuzhiyun err = snd_scarlett_controls_create(mixer);
3015*4882a593Smuzhiyun break;
3016*4882a593Smuzhiyun
3017*4882a593Smuzhiyun case USB_ID(0x1235, 0x8203): /* Focusrite Scarlett 6i6 2nd Gen */
3018*4882a593Smuzhiyun case USB_ID(0x1235, 0x8204): /* Focusrite Scarlett 18i8 2nd Gen */
3019*4882a593Smuzhiyun case USB_ID(0x1235, 0x8201): /* Focusrite Scarlett 18i20 2nd Gen */
3020*4882a593Smuzhiyun err = snd_scarlett_gen2_init(mixer);
3021*4882a593Smuzhiyun break;
3022*4882a593Smuzhiyun
3023*4882a593Smuzhiyun case USB_ID(0x041e, 0x323b): /* Creative Sound Blaster E1 */
3024*4882a593Smuzhiyun err = snd_soundblaster_e1_switch_create(mixer);
3025*4882a593Smuzhiyun break;
3026*4882a593Smuzhiyun case USB_ID(0x0bda, 0x4014): /* Dell WD15 dock */
3027*4882a593Smuzhiyun err = dell_dock_mixer_init(mixer);
3028*4882a593Smuzhiyun break;
3029*4882a593Smuzhiyun
3030*4882a593Smuzhiyun case USB_ID(0x2a39, 0x3fd2): /* RME ADI-2 Pro */
3031*4882a593Smuzhiyun case USB_ID(0x2a39, 0x3fd3): /* RME ADI-2 DAC */
3032*4882a593Smuzhiyun case USB_ID(0x2a39, 0x3fd4): /* RME */
3033*4882a593Smuzhiyun err = snd_rme_controls_create(mixer);
3034*4882a593Smuzhiyun break;
3035*4882a593Smuzhiyun
3036*4882a593Smuzhiyun case USB_ID(0x194f, 0x010c): /* Presonus Studio 1810c */
3037*4882a593Smuzhiyun err = snd_sc1810_init_mixer(mixer);
3038*4882a593Smuzhiyun break;
3039*4882a593Smuzhiyun case USB_ID(0x2a39, 0x3fb0): /* RME Babyface Pro FS */
3040*4882a593Smuzhiyun err = snd_bbfpro_controls_create(mixer);
3041*4882a593Smuzhiyun break;
3042*4882a593Smuzhiyun case USB_ID(0x2b73, 0x0017): /* Pioneer DJ DJM-250MK2 */
3043*4882a593Smuzhiyun err = snd_djm_controls_create(mixer, SND_DJM_250MK2_IDX);
3044*4882a593Smuzhiyun break;
3045*4882a593Smuzhiyun case USB_ID(0x08e4, 0x017f): /* Pioneer DJ DJM-750 */
3046*4882a593Smuzhiyun err = snd_djm_controls_create(mixer, SND_DJM_750_IDX);
3047*4882a593Smuzhiyun break;
3048*4882a593Smuzhiyun case USB_ID(0x2b73, 0x000a): /* Pioneer DJ DJM-900NXS2 */
3049*4882a593Smuzhiyun err = snd_djm_controls_create(mixer, SND_DJM_900NXS2_IDX);
3050*4882a593Smuzhiyun break;
3051*4882a593Smuzhiyun }
3052*4882a593Smuzhiyun
3053*4882a593Smuzhiyun return err;
3054*4882a593Smuzhiyun }
3055*4882a593Smuzhiyun
3056*4882a593Smuzhiyun #ifdef CONFIG_PM
snd_usb_mixer_resume_quirk(struct usb_mixer_interface * mixer)3057*4882a593Smuzhiyun void snd_usb_mixer_resume_quirk(struct usb_mixer_interface *mixer)
3058*4882a593Smuzhiyun {
3059*4882a593Smuzhiyun switch (mixer->chip->usb_id) {
3060*4882a593Smuzhiyun case USB_ID(0x0bda, 0x4014): /* Dell WD15 dock */
3061*4882a593Smuzhiyun dell_dock_mixer_init(mixer);
3062*4882a593Smuzhiyun break;
3063*4882a593Smuzhiyun }
3064*4882a593Smuzhiyun }
3065*4882a593Smuzhiyun #endif
3066*4882a593Smuzhiyun
snd_usb_mixer_rc_memory_change(struct usb_mixer_interface * mixer,int unitid)3067*4882a593Smuzhiyun void snd_usb_mixer_rc_memory_change(struct usb_mixer_interface *mixer,
3068*4882a593Smuzhiyun int unitid)
3069*4882a593Smuzhiyun {
3070*4882a593Smuzhiyun if (!mixer->rc_cfg)
3071*4882a593Smuzhiyun return;
3072*4882a593Smuzhiyun /* unit ids specific to Extigy/Audigy 2 NX: */
3073*4882a593Smuzhiyun switch (unitid) {
3074*4882a593Smuzhiyun case 0: /* remote control */
3075*4882a593Smuzhiyun mixer->rc_urb->dev = mixer->chip->dev;
3076*4882a593Smuzhiyun usb_submit_urb(mixer->rc_urb, GFP_ATOMIC);
3077*4882a593Smuzhiyun break;
3078*4882a593Smuzhiyun case 4: /* digital in jack */
3079*4882a593Smuzhiyun case 7: /* line in jacks */
3080*4882a593Smuzhiyun case 19: /* speaker out jacks */
3081*4882a593Smuzhiyun case 20: /* headphones out jack */
3082*4882a593Smuzhiyun break;
3083*4882a593Smuzhiyun /* live24ext: 4 = line-in jack */
3084*4882a593Smuzhiyun case 3: /* hp-out jack (may actuate Mute) */
3085*4882a593Smuzhiyun if (mixer->chip->usb_id == USB_ID(0x041e, 0x3040) ||
3086*4882a593Smuzhiyun mixer->chip->usb_id == USB_ID(0x041e, 0x3048))
3087*4882a593Smuzhiyun snd_usb_mixer_notify_id(mixer, mixer->rc_cfg->mute_mixer_id);
3088*4882a593Smuzhiyun break;
3089*4882a593Smuzhiyun default:
3090*4882a593Smuzhiyun usb_audio_dbg(mixer->chip, "memory change in unknown unit %d\n", unitid);
3091*4882a593Smuzhiyun break;
3092*4882a593Smuzhiyun }
3093*4882a593Smuzhiyun }
3094*4882a593Smuzhiyun
snd_dragonfly_quirk_db_scale(struct usb_mixer_interface * mixer,struct usb_mixer_elem_info * cval,struct snd_kcontrol * kctl)3095*4882a593Smuzhiyun static void snd_dragonfly_quirk_db_scale(struct usb_mixer_interface *mixer,
3096*4882a593Smuzhiyun struct usb_mixer_elem_info *cval,
3097*4882a593Smuzhiyun struct snd_kcontrol *kctl)
3098*4882a593Smuzhiyun {
3099*4882a593Smuzhiyun /* Approximation using 10 ranges based on output measurement on hw v1.2.
3100*4882a593Smuzhiyun * This seems close to the cubic mapping e.g. alsamixer uses. */
3101*4882a593Smuzhiyun static const DECLARE_TLV_DB_RANGE(scale,
3102*4882a593Smuzhiyun 0, 1, TLV_DB_MINMAX_ITEM(-5300, -4970),
3103*4882a593Smuzhiyun 2, 5, TLV_DB_MINMAX_ITEM(-4710, -4160),
3104*4882a593Smuzhiyun 6, 7, TLV_DB_MINMAX_ITEM(-3884, -3710),
3105*4882a593Smuzhiyun 8, 14, TLV_DB_MINMAX_ITEM(-3443, -2560),
3106*4882a593Smuzhiyun 15, 16, TLV_DB_MINMAX_ITEM(-2475, -2324),
3107*4882a593Smuzhiyun 17, 19, TLV_DB_MINMAX_ITEM(-2228, -2031),
3108*4882a593Smuzhiyun 20, 26, TLV_DB_MINMAX_ITEM(-1910, -1393),
3109*4882a593Smuzhiyun 27, 31, TLV_DB_MINMAX_ITEM(-1322, -1032),
3110*4882a593Smuzhiyun 32, 40, TLV_DB_MINMAX_ITEM(-968, -490),
3111*4882a593Smuzhiyun 41, 50, TLV_DB_MINMAX_ITEM(-441, 0),
3112*4882a593Smuzhiyun );
3113*4882a593Smuzhiyun
3114*4882a593Smuzhiyun if (cval->min == 0 && cval->max == 50) {
3115*4882a593Smuzhiyun usb_audio_info(mixer->chip, "applying DragonFly dB scale quirk (0-50 variant)\n");
3116*4882a593Smuzhiyun kctl->tlv.p = scale;
3117*4882a593Smuzhiyun kctl->vd[0].access |= SNDRV_CTL_ELEM_ACCESS_TLV_READ;
3118*4882a593Smuzhiyun kctl->vd[0].access &= ~SNDRV_CTL_ELEM_ACCESS_TLV_CALLBACK;
3119*4882a593Smuzhiyun
3120*4882a593Smuzhiyun } else if (cval->min == 0 && cval->max <= 1000) {
3121*4882a593Smuzhiyun /* Some other clearly broken DragonFly variant.
3122*4882a593Smuzhiyun * At least a 0..53 variant (hw v1.0) exists.
3123*4882a593Smuzhiyun */
3124*4882a593Smuzhiyun usb_audio_info(mixer->chip, "ignoring too narrow dB range on a DragonFly device");
3125*4882a593Smuzhiyun kctl->vd[0].access &= ~SNDRV_CTL_ELEM_ACCESS_TLV_CALLBACK;
3126*4882a593Smuzhiyun }
3127*4882a593Smuzhiyun }
3128*4882a593Smuzhiyun
snd_usb_mixer_fu_apply_quirk(struct usb_mixer_interface * mixer,struct usb_mixer_elem_info * cval,int unitid,struct snd_kcontrol * kctl)3129*4882a593Smuzhiyun void snd_usb_mixer_fu_apply_quirk(struct usb_mixer_interface *mixer,
3130*4882a593Smuzhiyun struct usb_mixer_elem_info *cval, int unitid,
3131*4882a593Smuzhiyun struct snd_kcontrol *kctl)
3132*4882a593Smuzhiyun {
3133*4882a593Smuzhiyun switch (mixer->chip->usb_id) {
3134*4882a593Smuzhiyun case USB_ID(0x21b4, 0x0081): /* AudioQuest DragonFly */
3135*4882a593Smuzhiyun if (unitid == 7 && cval->control == UAC_FU_VOLUME)
3136*4882a593Smuzhiyun snd_dragonfly_quirk_db_scale(mixer, cval, kctl);
3137*4882a593Smuzhiyun break;
3138*4882a593Smuzhiyun /* lowest playback value is muted on some devices */
3139*4882a593Smuzhiyun case USB_ID(0x0d8c, 0x000c): /* C-Media */
3140*4882a593Smuzhiyun case USB_ID(0x0d8c, 0x0014): /* C-Media */
3141*4882a593Smuzhiyun case USB_ID(0x19f7, 0x0003): /* RODE NT-USB */
3142*4882a593Smuzhiyun if (strstr(kctl->id.name, "Playback"))
3143*4882a593Smuzhiyun cval->min_mute = 1;
3144*4882a593Smuzhiyun break;
3145*4882a593Smuzhiyun }
3146*4882a593Smuzhiyun }
3147*4882a593Smuzhiyun
3148