Commit ca95c7bf authored by Takashi Iwai's avatar Takashi Iwai

ALSA: usb-audio: Fix parse of UAC2 Extension Units

Extension Unit (XU) is used to have a compatible layout with
Processing Unit (PU) on UAC1, and the usb-audio driver code assumed it
for parsing the descriptors.  Meanwhile, on UAC2, XU became slightly
incompatible with PU; namely, XU has a one-byte bmControls bitmap
while PU has two bytes bmControls bitmap.  This incompatibility
results in the read of a wrong address for the last iExtension field,
which ended up with an incorrect string for the mixer element name, as
recently reported for Focusrite Scarlett 18i20 device.

This patch corrects this misalignment by introducing a couple of new
macros and calling them depending on the descriptor type.

Fixes: 23caaf19 ("ALSA: usb-mixer: Add support for Audio Class v2.0")
Reported-by: default avatarStefan Sauer <ensonic@hora-obscura.de>
Cc: <stable@vger.kernel.org>
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent 33539936
...@@ -450,6 +450,43 @@ static inline __u8 *uac_processing_unit_specific(struct uac_processing_unit_desc ...@@ -450,6 +450,43 @@ static inline __u8 *uac_processing_unit_specific(struct uac_processing_unit_desc
} }
} }
/*
* Extension Unit (XU) has almost compatible layout with Processing Unit, but
* on UAC2, it has a different bmControls size (bControlSize); it's 1 byte for
* XU while 2 bytes for PU. The last iExtension field is a one-byte index as
* well as iProcessing field of PU.
*/
static inline __u8 uac_extension_unit_bControlSize(struct uac_processing_unit_descriptor *desc,
int protocol)
{
switch (protocol) {
case UAC_VERSION_1:
return desc->baSourceID[desc->bNrInPins + 4];
case UAC_VERSION_2:
return 1; /* in UAC2, this value is constant */
case UAC_VERSION_3:
return 4; /* in UAC3, this value is constant */
default:
return 1;
}
}
static inline __u8 uac_extension_unit_iExtension(struct uac_processing_unit_descriptor *desc,
int protocol)
{
__u8 control_size = uac_extension_unit_bControlSize(desc, protocol);
switch (protocol) {
case UAC_VERSION_1:
case UAC_VERSION_2:
default:
return *(uac_processing_unit_bmControls(desc, protocol)
+ control_size);
case UAC_VERSION_3:
return 0; /* UAC3 does not have this field */
}
}
/* 4.5.2 Class-Specific AS Interface Descriptor */ /* 4.5.2 Class-Specific AS Interface Descriptor */
struct uac1_as_header_descriptor { struct uac1_as_header_descriptor {
__u8 bLength; /* in bytes: 7 */ __u8 bLength; /* in bytes: 7 */
......
...@@ -2318,7 +2318,7 @@ static struct procunit_info extunits[] = { ...@@ -2318,7 +2318,7 @@ static struct procunit_info extunits[] = {
*/ */
static int build_audio_procunit(struct mixer_build *state, int unitid, static int build_audio_procunit(struct mixer_build *state, int unitid,
void *raw_desc, struct procunit_info *list, void *raw_desc, struct procunit_info *list,
char *name) bool extension_unit)
{ {
struct uac_processing_unit_descriptor *desc = raw_desc; struct uac_processing_unit_descriptor *desc = raw_desc;
int num_ins; int num_ins;
...@@ -2335,6 +2335,8 @@ static int build_audio_procunit(struct mixer_build *state, int unitid, ...@@ -2335,6 +2335,8 @@ static int build_audio_procunit(struct mixer_build *state, int unitid,
static struct procunit_info default_info = { static struct procunit_info default_info = {
0, NULL, default_value_info 0, NULL, default_value_info
}; };
const char *name = extension_unit ?
"Extension Unit" : "Processing Unit";
if (desc->bLength < 13) { if (desc->bLength < 13) {
usb_audio_err(state->chip, "invalid %s descriptor (id %d)\n", name, unitid); usb_audio_err(state->chip, "invalid %s descriptor (id %d)\n", name, unitid);
...@@ -2448,6 +2450,9 @@ static int build_audio_procunit(struct mixer_build *state, int unitid, ...@@ -2448,6 +2450,9 @@ static int build_audio_procunit(struct mixer_build *state, int unitid,
} else if (info->name) { } else if (info->name) {
strlcpy(kctl->id.name, info->name, sizeof(kctl->id.name)); strlcpy(kctl->id.name, info->name, sizeof(kctl->id.name));
} else { } else {
if (extension_unit)
nameid = uac_extension_unit_iExtension(desc, state->mixer->protocol);
else
nameid = uac_processing_unit_iProcessing(desc, state->mixer->protocol); nameid = uac_processing_unit_iProcessing(desc, state->mixer->protocol);
len = 0; len = 0;
if (nameid) if (nameid)
...@@ -2481,10 +2486,10 @@ static int parse_audio_processing_unit(struct mixer_build *state, int unitid, ...@@ -2481,10 +2486,10 @@ static int parse_audio_processing_unit(struct mixer_build *state, int unitid,
case UAC_VERSION_2: case UAC_VERSION_2:
default: default:
return build_audio_procunit(state, unitid, raw_desc, return build_audio_procunit(state, unitid, raw_desc,
procunits, "Processing Unit"); procunits, false);
case UAC_VERSION_3: case UAC_VERSION_3:
return build_audio_procunit(state, unitid, raw_desc, return build_audio_procunit(state, unitid, raw_desc,
uac3_procunits, "Processing Unit"); uac3_procunits, false);
} }
} }
...@@ -2495,8 +2500,7 @@ static int parse_audio_extension_unit(struct mixer_build *state, int unitid, ...@@ -2495,8 +2500,7 @@ static int parse_audio_extension_unit(struct mixer_build *state, int unitid,
* Note that we parse extension units with processing unit descriptors. * Note that we parse extension units with processing unit descriptors.
* That's ok as the layout is the same. * That's ok as the layout is the same.
*/ */
return build_audio_procunit(state, unitid, raw_desc, return build_audio_procunit(state, unitid, raw_desc, extunits, true);
extunits, "Extension Unit");
} }
/* /*
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment