mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-04 04:06:26 +00:00
ALSA: usb-audio: Support multiple control interfaces
Registering Numark Party Mix II fails with error 'bogus bTerminalLink 1'. The problem stems from the driver not being able to find input/output terminals required to configure audio streaming. The information about those terminals is stored in AudioControl Interface. Numark device contains 2 AudioControl Interfaces and the driver checks only one of them. According to the USB standard, a device can have multiple audio functions, each represented by Audio Interface Collection. Every audio function is considered to be closed box and will contain unique AudioControl Interface and zero or more AudioStreaming and MIDIStreaming Interfaces. The Numark device adheres to the standard and defines two audio functions: - MIDIStreaming function - AudioStreaming function It starts with MIDI function, followed by the audio function. The driver saves the first AudioControl Interface in `snd_usb_audio` structure associated with the entire device. It then attempts to use this interface to query for terminals and clocks. However, this fails because the correct information is stored in the second AudioControl Interface, defined in the second Audio Interface Collection. This patch introduces a structure holding association between each MIDI/Audio Interface and its corresponding AudioControl Interface, instead of relying on AudioControl Interface defined for the entire device. This structure is populated during usb probing phase and leveraged later when querying for terminals and when sending USB requests. Alternative solutions considered include: - defining a quirk for Numark where the order of interface is manually changed, or terminals are hardcoded in the driver. This solution would have fixed only this model, though it seems that device is USB compliant, and it also seems that other devices from this company may be affected. What's more, it looks like products from other manufacturers have similar problems, i.e. Rane One DJ console - keeping a list of all AudioControl Interfaces and querying all of them to find required information. That would have solved my problem and have low probability of breaking other devices, as we would always start with the same logic of querying first AudioControl Interface. This solution would not have followed the standard though. This patch preserves the `snd_usb_audio.ctrl_intf` variable, which holds the first AudioControl Interface, and uses it as a fallback when some interfaces are not parsed correctly and lack an associated AudioControl Interface, i.e., when configured via quirks. Link: https://bugzilla.kernel.org/show_bug.cgi?id=217865 Signed-off-by: Karol Kosik <k.kosik@outlook.com> Link: https://patch.msgid.link/AS8P190MB1285893F4735C8B32AD3886BEC852@AS8P190MB1285.EURP190.PROD.OUTLOOK.COM Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
parent
ebfb5a57ca
commit
6aa8700150
@ -206,6 +206,8 @@ static int snd_usb_create_stream(struct snd_usb_audio *chip, int ctrlif, int int
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
snd_usb_add_ctrl_interface_link(chip, interface, ctrlif);
|
||||||
|
|
||||||
if (! snd_usb_parse_audio_interface(chip, interface)) {
|
if (! snd_usb_parse_audio_interface(chip, interface)) {
|
||||||
usb_set_interface(dev, interface, 0); /* reset the current interface */
|
usb_set_interface(dev, interface, 0); /* reset the current interface */
|
||||||
return usb_driver_claim_interface(&usb_audio_driver, iface,
|
return usb_driver_claim_interface(&usb_audio_driver, iface,
|
||||||
|
@ -76,11 +76,14 @@ static bool validate_clock_multiplier(void *p, int id, int proto)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#define DEFINE_FIND_HELPER(name, obj, validator, type2, type3) \
|
#define DEFINE_FIND_HELPER(name, obj, validator, type2, type3) \
|
||||||
static obj *name(struct snd_usb_audio *chip, int id, int proto) \
|
static obj *name(struct snd_usb_audio *chip, int id, \
|
||||||
|
const struct audioformat *fmt) \
|
||||||
{ \
|
{ \
|
||||||
return find_uac_clock_desc(chip->ctrl_intf, id, validator, \
|
struct usb_host_interface *ctrl_intf = \
|
||||||
proto == UAC_VERSION_3 ? (type3) : (type2), \
|
snd_usb_find_ctrl_interface(chip, fmt->iface); \
|
||||||
proto); \
|
return find_uac_clock_desc(ctrl_intf, id, validator, \
|
||||||
|
fmt->protocol == UAC_VERSION_3 ? (type3) : (type2), \
|
||||||
|
fmt->protocol); \
|
||||||
}
|
}
|
||||||
|
|
||||||
DEFINE_FIND_HELPER(snd_usb_find_clock_source,
|
DEFINE_FIND_HELPER(snd_usb_find_clock_source,
|
||||||
@ -93,16 +96,19 @@ DEFINE_FIND_HELPER(snd_usb_find_clock_multiplier,
|
|||||||
union uac23_clock_multiplier_desc, validate_clock_multiplier,
|
union uac23_clock_multiplier_desc, validate_clock_multiplier,
|
||||||
UAC2_CLOCK_MULTIPLIER, UAC3_CLOCK_MULTIPLIER);
|
UAC2_CLOCK_MULTIPLIER, UAC3_CLOCK_MULTIPLIER);
|
||||||
|
|
||||||
static int uac_clock_selector_get_val(struct snd_usb_audio *chip, int selector_id)
|
static int uac_clock_selector_get_val(struct snd_usb_audio *chip,
|
||||||
|
int selector_id, int iface_no)
|
||||||
{
|
{
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
unsigned char buf;
|
unsigned char buf;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
ctrl_intf = snd_usb_find_ctrl_interface(chip, iface_no);
|
||||||
ret = snd_usb_ctl_msg(chip->dev, usb_rcvctrlpipe(chip->dev, 0),
|
ret = snd_usb_ctl_msg(chip->dev, usb_rcvctrlpipe(chip->dev, 0),
|
||||||
UAC2_CS_CUR,
|
UAC2_CS_CUR,
|
||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
||||||
UAC2_CX_CLOCK_SELECTOR << 8,
|
UAC2_CX_CLOCK_SELECTOR << 8,
|
||||||
snd_usb_ctrl_intf(chip) | (selector_id << 8),
|
snd_usb_ctrl_intf(ctrl_intf) | (selector_id << 8),
|
||||||
&buf, sizeof(buf));
|
&buf, sizeof(buf));
|
||||||
|
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
@ -111,16 +117,18 @@ static int uac_clock_selector_get_val(struct snd_usb_audio *chip, int selector_i
|
|||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int uac_clock_selector_set_val(struct snd_usb_audio *chip, int selector_id,
|
static int uac_clock_selector_set_val(struct snd_usb_audio *chip,
|
||||||
unsigned char pin)
|
int selector_id, unsigned char pin, int iface_no)
|
||||||
{
|
{
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
ctrl_intf = snd_usb_find_ctrl_interface(chip, iface_no);
|
||||||
ret = snd_usb_ctl_msg(chip->dev, usb_sndctrlpipe(chip->dev, 0),
|
ret = snd_usb_ctl_msg(chip->dev, usb_sndctrlpipe(chip->dev, 0),
|
||||||
UAC2_CS_CUR,
|
UAC2_CS_CUR,
|
||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_OUT,
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_OUT,
|
||||||
UAC2_CX_CLOCK_SELECTOR << 8,
|
UAC2_CX_CLOCK_SELECTOR << 8,
|
||||||
snd_usb_ctrl_intf(chip) | (selector_id << 8),
|
snd_usb_ctrl_intf(ctrl_intf) | (selector_id << 8),
|
||||||
&pin, sizeof(pin));
|
&pin, sizeof(pin));
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
@ -132,7 +140,7 @@ static int uac_clock_selector_set_val(struct snd_usb_audio *chip, int selector_i
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = uac_clock_selector_get_val(chip, selector_id);
|
ret = uac_clock_selector_get_val(chip, selector_id, iface_no);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
@ -155,8 +163,10 @@ static bool uac_clock_source_is_valid_quirk(struct snd_usb_audio *chip,
|
|||||||
unsigned char data;
|
unsigned char data;
|
||||||
struct usb_device *dev = chip->dev;
|
struct usb_device *dev = chip->dev;
|
||||||
union uac23_clock_source_desc *cs_desc;
|
union uac23_clock_source_desc *cs_desc;
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
|
|
||||||
cs_desc = snd_usb_find_clock_source(chip, source_id, fmt->protocol);
|
ctrl_intf = snd_usb_find_ctrl_interface(chip, fmt->iface);
|
||||||
|
cs_desc = snd_usb_find_clock_source(chip, source_id, fmt);
|
||||||
if (!cs_desc)
|
if (!cs_desc)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
@ -191,7 +201,7 @@ static bool uac_clock_source_is_valid_quirk(struct snd_usb_audio *chip,
|
|||||||
err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_CUR,
|
err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_CUR,
|
||||||
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
||||||
UAC2_CS_CONTROL_CLOCK_VALID << 8,
|
UAC2_CS_CONTROL_CLOCK_VALID << 8,
|
||||||
snd_usb_ctrl_intf(chip) | (source_id << 8),
|
snd_usb_ctrl_intf(ctrl_intf) | (source_id << 8),
|
||||||
&data, sizeof(data));
|
&data, sizeof(data));
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
dev_warn(&dev->dev,
|
dev_warn(&dev->dev,
|
||||||
@ -217,8 +227,10 @@ static bool uac_clock_source_is_valid(struct snd_usb_audio *chip,
|
|||||||
struct usb_device *dev = chip->dev;
|
struct usb_device *dev = chip->dev;
|
||||||
u32 bmControls;
|
u32 bmControls;
|
||||||
union uac23_clock_source_desc *cs_desc;
|
union uac23_clock_source_desc *cs_desc;
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
|
|
||||||
cs_desc = snd_usb_find_clock_source(chip, source_id, fmt->protocol);
|
ctrl_intf = snd_usb_find_ctrl_interface(chip, fmt->iface);
|
||||||
|
cs_desc = snd_usb_find_clock_source(chip, source_id, fmt);
|
||||||
if (!cs_desc)
|
if (!cs_desc)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
@ -235,7 +247,7 @@ static bool uac_clock_source_is_valid(struct snd_usb_audio *chip,
|
|||||||
err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_CUR,
|
err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_CUR,
|
||||||
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
||||||
UAC2_CS_CONTROL_CLOCK_VALID << 8,
|
UAC2_CS_CONTROL_CLOCK_VALID << 8,
|
||||||
snd_usb_ctrl_intf(chip) | (source_id << 8),
|
snd_usb_ctrl_intf(ctrl_intf) | (source_id << 8),
|
||||||
&data, sizeof(data));
|
&data, sizeof(data));
|
||||||
|
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
@ -274,7 +286,7 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* first, see if the ID we're looking at is a clock source already */
|
/* first, see if the ID we're looking at is a clock source already */
|
||||||
source = snd_usb_find_clock_source(chip, entity_id, proto);
|
source = snd_usb_find_clock_source(chip, entity_id, fmt);
|
||||||
if (source) {
|
if (source) {
|
||||||
entity_id = GET_VAL(source, proto, bClockID);
|
entity_id = GET_VAL(source, proto, bClockID);
|
||||||
if (validate && !uac_clock_source_is_valid(chip, fmt,
|
if (validate && !uac_clock_source_is_valid(chip, fmt,
|
||||||
@ -287,7 +299,7 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip,
|
|||||||
return entity_id;
|
return entity_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
selector = snd_usb_find_clock_selector(chip, entity_id, proto);
|
selector = snd_usb_find_clock_selector(chip, entity_id, fmt);
|
||||||
if (selector) {
|
if (selector) {
|
||||||
pins = GET_VAL(selector, proto, bNrInPins);
|
pins = GET_VAL(selector, proto, bNrInPins);
|
||||||
clock_id = GET_VAL(selector, proto, bClockID);
|
clock_id = GET_VAL(selector, proto, bClockID);
|
||||||
@ -317,7 +329,7 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip,
|
|||||||
|
|
||||||
/* the entity ID we are looking at is a selector.
|
/* the entity ID we are looking at is a selector.
|
||||||
* find out what it currently selects */
|
* find out what it currently selects */
|
||||||
ret = uac_clock_selector_get_val(chip, clock_id);
|
ret = uac_clock_selector_get_val(chip, clock_id, fmt->iface);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
if (!chip->autoclock)
|
if (!chip->autoclock)
|
||||||
return ret;
|
return ret;
|
||||||
@ -346,7 +358,7 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip,
|
|||||||
if (chip->quirk_flags & QUIRK_FLAG_SKIP_CLOCK_SELECTOR ||
|
if (chip->quirk_flags & QUIRK_FLAG_SKIP_CLOCK_SELECTOR ||
|
||||||
!writeable)
|
!writeable)
|
||||||
return ret;
|
return ret;
|
||||||
err = uac_clock_selector_set_val(chip, entity_id, cur);
|
err = uac_clock_selector_set_val(chip, entity_id, cur, fmt->iface);
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
if (pins == 1) {
|
if (pins == 1) {
|
||||||
usb_audio_dbg(chip,
|
usb_audio_dbg(chip,
|
||||||
@ -377,7 +389,7 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip,
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
err = uac_clock_selector_set_val(chip, entity_id, i);
|
err = uac_clock_selector_set_val(chip, entity_id, i, fmt->iface);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@ -391,7 +403,7 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* FIXME: multipliers only act as pass-thru element for now */
|
/* FIXME: multipliers only act as pass-thru element for now */
|
||||||
multiplier = snd_usb_find_clock_multiplier(chip, entity_id, proto);
|
multiplier = snd_usb_find_clock_multiplier(chip, entity_id, fmt);
|
||||||
if (multiplier)
|
if (multiplier)
|
||||||
return __uac_clock_find_source(chip, fmt,
|
return __uac_clock_find_source(chip, fmt,
|
||||||
GET_VAL(multiplier, proto, bCSourceID),
|
GET_VAL(multiplier, proto, bCSourceID),
|
||||||
@ -491,11 +503,13 @@ static int get_sample_rate_v2v3(struct snd_usb_audio *chip, int iface,
|
|||||||
struct usb_device *dev = chip->dev;
|
struct usb_device *dev = chip->dev;
|
||||||
__le32 data;
|
__le32 data;
|
||||||
int err;
|
int err;
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
|
|
||||||
|
ctrl_intf = snd_usb_find_ctrl_interface(chip, iface);
|
||||||
err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_CUR,
|
err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_CUR,
|
||||||
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
||||||
UAC2_CS_CONTROL_SAM_FREQ << 8,
|
UAC2_CS_CONTROL_SAM_FREQ << 8,
|
||||||
snd_usb_ctrl_intf(chip) | (clock << 8),
|
snd_usb_ctrl_intf(ctrl_intf) | (clock << 8),
|
||||||
&data, sizeof(data));
|
&data, sizeof(data));
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
dev_warn(&dev->dev, "%d:%d: cannot get freq (v2/v3): err %d\n",
|
dev_warn(&dev->dev, "%d:%d: cannot get freq (v2/v3): err %d\n",
|
||||||
@ -524,8 +538,10 @@ int snd_usb_set_sample_rate_v2v3(struct snd_usb_audio *chip,
|
|||||||
__le32 data;
|
__le32 data;
|
||||||
int err;
|
int err;
|
||||||
union uac23_clock_source_desc *cs_desc;
|
union uac23_clock_source_desc *cs_desc;
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
|
|
||||||
cs_desc = snd_usb_find_clock_source(chip, clock, fmt->protocol);
|
ctrl_intf = snd_usb_find_ctrl_interface(chip, fmt->iface);
|
||||||
|
cs_desc = snd_usb_find_clock_source(chip, clock, fmt);
|
||||||
|
|
||||||
if (!cs_desc)
|
if (!cs_desc)
|
||||||
return 0;
|
return 0;
|
||||||
@ -544,7 +560,7 @@ int snd_usb_set_sample_rate_v2v3(struct snd_usb_audio *chip,
|
|||||||
err = snd_usb_ctl_msg(chip->dev, usb_sndctrlpipe(chip->dev, 0), UAC2_CS_CUR,
|
err = snd_usb_ctl_msg(chip->dev, usb_sndctrlpipe(chip->dev, 0), UAC2_CS_CUR,
|
||||||
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT,
|
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT,
|
||||||
UAC2_CS_CONTROL_SAM_FREQ << 8,
|
UAC2_CS_CONTROL_SAM_FREQ << 8,
|
||||||
snd_usb_ctrl_intf(chip) | (clock << 8),
|
snd_usb_ctrl_intf(ctrl_intf) | (clock << 8),
|
||||||
&data, sizeof(data));
|
&data, sizeof(data));
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return err;
|
return err;
|
||||||
|
@ -545,7 +545,9 @@ static int parse_audio_format_rates_v2v3(struct snd_usb_audio *chip,
|
|||||||
unsigned char tmp[2], *data;
|
unsigned char tmp[2], *data;
|
||||||
int nr_triplets, data_size, ret = 0, ret_l6;
|
int nr_triplets, data_size, ret = 0, ret_l6;
|
||||||
int clock = snd_usb_clock_find_source(chip, fp, false);
|
int clock = snd_usb_clock_find_source(chip, fp, false);
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
|
|
||||||
|
ctrl_intf = snd_usb_find_ctrl_interface(chip, fp->iface);
|
||||||
if (clock < 0) {
|
if (clock < 0) {
|
||||||
dev_err(&dev->dev,
|
dev_err(&dev->dev,
|
||||||
"%s(): unable to find clock source (clock %d)\n",
|
"%s(): unable to find clock source (clock %d)\n",
|
||||||
@ -557,7 +559,7 @@ static int parse_audio_format_rates_v2v3(struct snd_usb_audio *chip,
|
|||||||
ret = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_RANGE,
|
ret = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_RANGE,
|
||||||
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
||||||
UAC2_CS_CONTROL_SAM_FREQ << 8,
|
UAC2_CS_CONTROL_SAM_FREQ << 8,
|
||||||
snd_usb_ctrl_intf(chip) | (clock << 8),
|
snd_usb_ctrl_intf(ctrl_intf) | (clock << 8),
|
||||||
tmp, sizeof(tmp));
|
tmp, sizeof(tmp));
|
||||||
|
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
@ -592,7 +594,7 @@ static int parse_audio_format_rates_v2v3(struct snd_usb_audio *chip,
|
|||||||
ret = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_RANGE,
|
ret = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_RANGE,
|
||||||
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
||||||
UAC2_CS_CONTROL_SAM_FREQ << 8,
|
UAC2_CS_CONTROL_SAM_FREQ << 8,
|
||||||
snd_usb_ctrl_intf(chip) | (clock << 8),
|
snd_usb_ctrl_intf(ctrl_intf) | (clock << 8),
|
||||||
data, data_size);
|
data, data_size);
|
||||||
|
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
|
@ -130,3 +130,37 @@ snd_usb_get_host_interface(struct snd_usb_audio *chip, int ifnum, int altsetting
|
|||||||
return NULL;
|
return NULL;
|
||||||
return usb_altnum_to_altsetting(iface, altsetting);
|
return usb_altnum_to_altsetting(iface, altsetting);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int snd_usb_add_ctrl_interface_link(struct snd_usb_audio *chip, int ifnum,
|
||||||
|
int ctrlif)
|
||||||
|
{
|
||||||
|
struct usb_device *dev = chip->dev;
|
||||||
|
struct usb_host_interface *host_iface;
|
||||||
|
|
||||||
|
if (chip->num_intf_to_ctrl >= MAX_CARD_INTERFACES) {
|
||||||
|
dev_info(&dev->dev, "Too many interfaces assigned to the single USB-audio card\n");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* find audiocontrol interface */
|
||||||
|
host_iface = &usb_ifnum_to_if(dev, ctrlif)->altsetting[0];
|
||||||
|
|
||||||
|
chip->intf_to_ctrl[chip->num_intf_to_ctrl].interface = ifnum;
|
||||||
|
chip->intf_to_ctrl[chip->num_intf_to_ctrl].ctrl_intf = host_iface;
|
||||||
|
chip->num_intf_to_ctrl++;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct usb_host_interface *snd_usb_find_ctrl_interface(struct snd_usb_audio *chip,
|
||||||
|
int ifnum)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < chip->num_intf_to_ctrl; ++i)
|
||||||
|
if (chip->intf_to_ctrl[i].interface == ifnum)
|
||||||
|
return chip->intf_to_ctrl[i].ctrl_intf;
|
||||||
|
|
||||||
|
/* Fallback to first audiocontrol interface */
|
||||||
|
return chip->ctrl_intf;
|
||||||
|
}
|
||||||
|
@ -17,6 +17,12 @@ unsigned char snd_usb_parse_datainterval(struct snd_usb_audio *chip,
|
|||||||
struct usb_host_interface *
|
struct usb_host_interface *
|
||||||
snd_usb_get_host_interface(struct snd_usb_audio *chip, int ifnum, int altsetting);
|
snd_usb_get_host_interface(struct snd_usb_audio *chip, int ifnum, int altsetting);
|
||||||
|
|
||||||
|
int snd_usb_add_ctrl_interface_link(struct snd_usb_audio *chip, int ifnum,
|
||||||
|
int ctrlif);
|
||||||
|
|
||||||
|
struct usb_host_interface *snd_usb_find_ctrl_interface(struct snd_usb_audio *chip,
|
||||||
|
int ifnum);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* retrieve usb_interface descriptor from the host interface
|
* retrieve usb_interface descriptor from the host interface
|
||||||
* (conditional for compatibility with the older API)
|
* (conditional for compatibility with the older API)
|
||||||
@ -28,9 +34,9 @@ snd_usb_get_host_interface(struct snd_usb_audio *chip, int ifnum, int altsetting
|
|||||||
|
|
||||||
#define snd_usb_get_speed(dev) ((dev)->speed)
|
#define snd_usb_get_speed(dev) ((dev)->speed)
|
||||||
|
|
||||||
static inline int snd_usb_ctrl_intf(struct snd_usb_audio *chip)
|
static inline int snd_usb_ctrl_intf(struct usb_host_interface *ctrl_intf)
|
||||||
{
|
{
|
||||||
return get_iface_desc(chip->ctrl_intf)->bInterfaceNumber;
|
return get_iface_desc(ctrl_intf)->bInterfaceNumber;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* in validate.c */
|
/* in validate.c */
|
||||||
|
@ -728,7 +728,7 @@ static int get_cluster_channels_v3(struct mixer_build *state, unsigned int clust
|
|||||||
UAC3_CS_REQ_HIGH_CAPABILITY_DESCRIPTOR,
|
UAC3_CS_REQ_HIGH_CAPABILITY_DESCRIPTOR,
|
||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
||||||
cluster_id,
|
cluster_id,
|
||||||
snd_usb_ctrl_intf(state->chip),
|
snd_usb_ctrl_intf(state->mixer->hostif),
|
||||||
&c_header, sizeof(c_header));
|
&c_header, sizeof(c_header));
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
@ -1043,7 +1043,7 @@ static int snd_ftu_eff_switch_init(struct usb_mixer_interface *mixer,
|
|||||||
err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC_GET_CUR,
|
err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC_GET_CUR,
|
||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
||||||
pval & 0xff00,
|
pval & 0xff00,
|
||||||
snd_usb_ctrl_intf(mixer->chip) | ((pval & 0xff) << 8),
|
snd_usb_ctrl_intf(mixer->hostif) | ((pval & 0xff) << 8),
|
||||||
value, 2);
|
value, 2);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return err;
|
return err;
|
||||||
@ -1077,7 +1077,7 @@ static int snd_ftu_eff_switch_update(struct usb_mixer_elem_list *list)
|
|||||||
UAC_SET_CUR,
|
UAC_SET_CUR,
|
||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_OUT,
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_OUT,
|
||||||
pval & 0xff00,
|
pval & 0xff00,
|
||||||
snd_usb_ctrl_intf(chip) | ((pval & 0xff) << 8),
|
snd_usb_ctrl_intf(list->mixer->hostif) | ((pval & 0xff) << 8),
|
||||||
value, 2);
|
value, 2);
|
||||||
snd_usb_unlock_shutdown(chip);
|
snd_usb_unlock_shutdown(chip);
|
||||||
return err;
|
return err;
|
||||||
@ -2115,24 +2115,25 @@ static int dell_dock_mixer_create(struct usb_mixer_interface *mixer)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void dell_dock_init_vol(struct snd_usb_audio *chip, int ch, int id)
|
static void dell_dock_init_vol(struct usb_mixer_interface *mixer, int ch, int id)
|
||||||
{
|
{
|
||||||
|
struct snd_usb_audio *chip = mixer->chip;
|
||||||
u16 buf = 0;
|
u16 buf = 0;
|
||||||
|
|
||||||
snd_usb_ctl_msg(chip->dev, usb_sndctrlpipe(chip->dev, 0), UAC_SET_CUR,
|
snd_usb_ctl_msg(chip->dev, usb_sndctrlpipe(chip->dev, 0), UAC_SET_CUR,
|
||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_OUT,
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_OUT,
|
||||||
(UAC_FU_VOLUME << 8) | ch,
|
(UAC_FU_VOLUME << 8) | ch,
|
||||||
snd_usb_ctrl_intf(chip) | (id << 8),
|
snd_usb_ctrl_intf(mixer->hostif) | (id << 8),
|
||||||
&buf, 2);
|
&buf, 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int dell_dock_mixer_init(struct usb_mixer_interface *mixer)
|
static int dell_dock_mixer_init(struct usb_mixer_interface *mixer)
|
||||||
{
|
{
|
||||||
/* fix to 0dB playback volumes */
|
/* fix to 0dB playback volumes */
|
||||||
dell_dock_init_vol(mixer->chip, 1, 16);
|
dell_dock_init_vol(mixer, 1, 16);
|
||||||
dell_dock_init_vol(mixer->chip, 2, 16);
|
dell_dock_init_vol(mixer, 2, 16);
|
||||||
dell_dock_init_vol(mixer->chip, 1, 19);
|
dell_dock_init_vol(mixer, 1, 19);
|
||||||
dell_dock_init_vol(mixer->chip, 2, 19);
|
dell_dock_init_vol(mixer, 2, 19);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -460,7 +460,7 @@ static int scarlett_ctl_meter_get(struct snd_kcontrol *kctl,
|
|||||||
struct snd_usb_audio *chip = elem->head.mixer->chip;
|
struct snd_usb_audio *chip = elem->head.mixer->chip;
|
||||||
unsigned char buf[2 * MAX_CHANNELS] = {0, };
|
unsigned char buf[2 * MAX_CHANNELS] = {0, };
|
||||||
int wValue = (elem->control << 8) | elem->idx_off;
|
int wValue = (elem->control << 8) | elem->idx_off;
|
||||||
int idx = snd_usb_ctrl_intf(chip) | (elem->head.id << 8);
|
int idx = snd_usb_ctrl_intf(elem->head.mixer->hostif) | (elem->head.id << 8);
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
err = snd_usb_ctl_msg(chip->dev,
|
err = snd_usb_ctl_msg(chip->dev,
|
||||||
@ -1002,7 +1002,7 @@ int snd_scarlett_controls_create(struct usb_mixer_interface *mixer)
|
|||||||
err = snd_usb_ctl_msg(mixer->chip->dev,
|
err = snd_usb_ctl_msg(mixer->chip->dev,
|
||||||
usb_sndctrlpipe(mixer->chip->dev, 0), UAC2_CS_CUR,
|
usb_sndctrlpipe(mixer->chip->dev, 0), UAC2_CS_CUR,
|
||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS |
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS |
|
||||||
USB_DIR_OUT, 0x0100, snd_usb_ctrl_intf(mixer->chip) |
|
USB_DIR_OUT, 0x0100, snd_usb_ctrl_intf(mixer->hostif) |
|
||||||
(0x29 << 8), sample_rate_buffer, 4);
|
(0x29 << 8), sample_rate_buffer, 4);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return err;
|
return err;
|
||||||
|
@ -40,6 +40,7 @@ snd_usb_find_power_domain(struct usb_host_interface *ctrl_iface,
|
|||||||
le16_to_cpu(pd_desc->waRecoveryTime1);
|
le16_to_cpu(pd_desc->waRecoveryTime1);
|
||||||
pd->pd_d2d0_rec =
|
pd->pd_d2d0_rec =
|
||||||
le16_to_cpu(pd_desc->waRecoveryTime2);
|
le16_to_cpu(pd_desc->waRecoveryTime2);
|
||||||
|
pd->ctrl_iface = ctrl_iface;
|
||||||
return pd;
|
return pd;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -57,7 +58,7 @@ int snd_usb_power_domain_set(struct snd_usb_audio *chip,
|
|||||||
unsigned char current_state;
|
unsigned char current_state;
|
||||||
int err, idx;
|
int err, idx;
|
||||||
|
|
||||||
idx = snd_usb_ctrl_intf(chip) | (pd->pd_id << 8);
|
idx = snd_usb_ctrl_intf(pd->ctrl_iface) | (pd->pd_id << 8);
|
||||||
|
|
||||||
err = snd_usb_ctl_msg(chip->dev, usb_rcvctrlpipe(chip->dev, 0),
|
err = snd_usb_ctl_msg(chip->dev, usb_rcvctrlpipe(chip->dev, 0),
|
||||||
UAC2_CS_CUR,
|
UAC2_CS_CUR,
|
||||||
|
@ -6,6 +6,7 @@ struct snd_usb_power_domain {
|
|||||||
int pd_id; /* UAC3 Power Domain ID */
|
int pd_id; /* UAC3 Power Domain ID */
|
||||||
int pd_d1d0_rec; /* D1 to D0 recovery time */
|
int pd_d1d0_rec; /* D1 to D0 recovery time */
|
||||||
int pd_d2d0_rec; /* D2 to D0 recovery time */
|
int pd_d2d0_rec; /* D2 to D0 recovery time */
|
||||||
|
struct usb_host_interface *ctrl_iface; /* Control interface */
|
||||||
};
|
};
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
|
@ -713,10 +713,13 @@ snd_usb_get_audioformat_uac12(struct snd_usb_audio *chip,
|
|||||||
struct usb_device *dev = chip->dev;
|
struct usb_device *dev = chip->dev;
|
||||||
struct uac_format_type_i_continuous_descriptor *fmt;
|
struct uac_format_type_i_continuous_descriptor *fmt;
|
||||||
unsigned int num_channels = 0, chconfig = 0;
|
unsigned int num_channels = 0, chconfig = 0;
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
struct audioformat *fp;
|
struct audioformat *fp;
|
||||||
int clock = 0;
|
int clock = 0;
|
||||||
u64 format;
|
u64 format;
|
||||||
|
|
||||||
|
ctrl_intf = snd_usb_find_ctrl_interface(chip, iface_no);
|
||||||
|
|
||||||
/* get audio formats */
|
/* get audio formats */
|
||||||
if (protocol == UAC_VERSION_1) {
|
if (protocol == UAC_VERSION_1) {
|
||||||
struct uac1_as_header_descriptor *as =
|
struct uac1_as_header_descriptor *as =
|
||||||
@ -740,7 +743,7 @@ snd_usb_get_audioformat_uac12(struct snd_usb_audio *chip,
|
|||||||
|
|
||||||
format = le16_to_cpu(as->wFormatTag); /* remember the format value */
|
format = le16_to_cpu(as->wFormatTag); /* remember the format value */
|
||||||
|
|
||||||
iterm = snd_usb_find_input_terminal_descriptor(chip->ctrl_intf,
|
iterm = snd_usb_find_input_terminal_descriptor(ctrl_intf,
|
||||||
as->bTerminalLink,
|
as->bTerminalLink,
|
||||||
protocol);
|
protocol);
|
||||||
if (iterm) {
|
if (iterm) {
|
||||||
@ -776,7 +779,7 @@ snd_usb_get_audioformat_uac12(struct snd_usb_audio *chip,
|
|||||||
* lookup the terminal associated to this interface
|
* lookup the terminal associated to this interface
|
||||||
* to extract the clock
|
* to extract the clock
|
||||||
*/
|
*/
|
||||||
input_term = snd_usb_find_input_terminal_descriptor(chip->ctrl_intf,
|
input_term = snd_usb_find_input_terminal_descriptor(ctrl_intf,
|
||||||
as->bTerminalLink,
|
as->bTerminalLink,
|
||||||
protocol);
|
protocol);
|
||||||
if (input_term) {
|
if (input_term) {
|
||||||
@ -786,7 +789,7 @@ snd_usb_get_audioformat_uac12(struct snd_usb_audio *chip,
|
|||||||
goto found_clock;
|
goto found_clock;
|
||||||
}
|
}
|
||||||
|
|
||||||
output_term = snd_usb_find_output_terminal_descriptor(chip->ctrl_intf,
|
output_term = snd_usb_find_output_terminal_descriptor(ctrl_intf,
|
||||||
as->bTerminalLink,
|
as->bTerminalLink,
|
||||||
protocol);
|
protocol);
|
||||||
if (output_term) {
|
if (output_term) {
|
||||||
@ -870,6 +873,7 @@ snd_usb_get_audioformat_uac3(struct snd_usb_audio *chip,
|
|||||||
struct uac3_cluster_header_descriptor *cluster;
|
struct uac3_cluster_header_descriptor *cluster;
|
||||||
struct uac3_as_header_descriptor *as = NULL;
|
struct uac3_as_header_descriptor *as = NULL;
|
||||||
struct uac3_hc_descriptor_header hc_header;
|
struct uac3_hc_descriptor_header hc_header;
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
struct snd_pcm_chmap_elem *chmap;
|
struct snd_pcm_chmap_elem *chmap;
|
||||||
struct snd_usb_power_domain *pd;
|
struct snd_usb_power_domain *pd;
|
||||||
unsigned char badd_profile;
|
unsigned char badd_profile;
|
||||||
@ -881,6 +885,7 @@ snd_usb_get_audioformat_uac3(struct snd_usb_audio *chip,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
badd_profile = chip->badd_profile;
|
badd_profile = chip->badd_profile;
|
||||||
|
ctrl_intf = snd_usb_find_ctrl_interface(chip, iface_no);
|
||||||
|
|
||||||
if (badd_profile >= UAC3_FUNCTION_SUBCLASS_GENERIC_IO) {
|
if (badd_profile >= UAC3_FUNCTION_SUBCLASS_GENERIC_IO) {
|
||||||
unsigned int maxpacksize =
|
unsigned int maxpacksize =
|
||||||
@ -966,7 +971,7 @@ snd_usb_get_audioformat_uac3(struct snd_usb_audio *chip,
|
|||||||
UAC3_CS_REQ_HIGH_CAPABILITY_DESCRIPTOR,
|
UAC3_CS_REQ_HIGH_CAPABILITY_DESCRIPTOR,
|
||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
||||||
cluster_id,
|
cluster_id,
|
||||||
snd_usb_ctrl_intf(chip),
|
snd_usb_ctrl_intf(ctrl_intf),
|
||||||
&hc_header, sizeof(hc_header));
|
&hc_header, sizeof(hc_header));
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return ERR_PTR(err);
|
return ERR_PTR(err);
|
||||||
@ -990,7 +995,7 @@ snd_usb_get_audioformat_uac3(struct snd_usb_audio *chip,
|
|||||||
UAC3_CS_REQ_HIGH_CAPABILITY_DESCRIPTOR,
|
UAC3_CS_REQ_HIGH_CAPABILITY_DESCRIPTOR,
|
||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
||||||
cluster_id,
|
cluster_id,
|
||||||
snd_usb_ctrl_intf(chip),
|
snd_usb_ctrl_intf(ctrl_intf),
|
||||||
cluster, wLength);
|
cluster, wLength);
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
kfree(cluster);
|
kfree(cluster);
|
||||||
@ -1011,7 +1016,7 @@ snd_usb_get_audioformat_uac3(struct snd_usb_audio *chip,
|
|||||||
* lookup the terminal associated to this interface
|
* lookup the terminal associated to this interface
|
||||||
* to extract the clock
|
* to extract the clock
|
||||||
*/
|
*/
|
||||||
input_term = snd_usb_find_input_terminal_descriptor(chip->ctrl_intf,
|
input_term = snd_usb_find_input_terminal_descriptor(ctrl_intf,
|
||||||
as->bTerminalLink,
|
as->bTerminalLink,
|
||||||
UAC_VERSION_3);
|
UAC_VERSION_3);
|
||||||
if (input_term) {
|
if (input_term) {
|
||||||
@ -1019,7 +1024,7 @@ snd_usb_get_audioformat_uac3(struct snd_usb_audio *chip,
|
|||||||
goto found_clock;
|
goto found_clock;
|
||||||
}
|
}
|
||||||
|
|
||||||
output_term = snd_usb_find_output_terminal_descriptor(chip->ctrl_intf,
|
output_term = snd_usb_find_output_terminal_descriptor(ctrl_intf,
|
||||||
as->bTerminalLink,
|
as->bTerminalLink,
|
||||||
UAC_VERSION_3);
|
UAC_VERSION_3);
|
||||||
if (output_term) {
|
if (output_term) {
|
||||||
@ -1068,7 +1073,7 @@ snd_usb_get_audioformat_uac3(struct snd_usb_audio *chip,
|
|||||||
UAC_VERSION_3,
|
UAC_VERSION_3,
|
||||||
iface_no);
|
iface_no);
|
||||||
|
|
||||||
pd = snd_usb_find_power_domain(chip->ctrl_intf,
|
pd = snd_usb_find_power_domain(ctrl_intf,
|
||||||
as->bTerminalLink);
|
as->bTerminalLink);
|
||||||
|
|
||||||
/* ok, let's parse further... */
|
/* ok, let's parse further... */
|
||||||
|
@ -21,6 +21,15 @@ struct media_intf_devnode;
|
|||||||
|
|
||||||
#define MAX_CARD_INTERFACES 16
|
#define MAX_CARD_INTERFACES 16
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Structure holding assosiation between Audio Control Interface
|
||||||
|
* and given Streaming or Midi Interface.
|
||||||
|
*/
|
||||||
|
struct snd_intf_to_ctrl {
|
||||||
|
u8 interface;
|
||||||
|
struct usb_host_interface *ctrl_intf;
|
||||||
|
};
|
||||||
|
|
||||||
struct snd_usb_audio {
|
struct snd_usb_audio {
|
||||||
int index;
|
int index;
|
||||||
struct usb_device *dev;
|
struct usb_device *dev;
|
||||||
@ -63,6 +72,9 @@ struct snd_usb_audio {
|
|||||||
struct usb_host_interface *ctrl_intf; /* the audio control interface */
|
struct usb_host_interface *ctrl_intf; /* the audio control interface */
|
||||||
struct media_device *media_dev;
|
struct media_device *media_dev;
|
||||||
struct media_intf_devnode *ctl_intf_media_devnode;
|
struct media_intf_devnode *ctl_intf_media_devnode;
|
||||||
|
|
||||||
|
unsigned int num_intf_to_ctrl;
|
||||||
|
struct snd_intf_to_ctrl intf_to_ctrl[MAX_CARD_INTERFACES];
|
||||||
};
|
};
|
||||||
|
|
||||||
#define USB_AUDIO_IFACE_UNUSED ((void *)-1L)
|
#define USB_AUDIO_IFACE_UNUSED ((void *)-1L)
|
||||||
|
Loading…
Reference in New Issue
Block a user