]> asedeno.scripts.mit.edu Git - linux.git/commitdiff
ALSA: usb-audio: Processing Unit controls parsing in UAC2
authorJorge Sanjuan <jorge.sanjuan@codethink.co.uk>
Wed, 11 Jul 2018 12:37:52 +0000 (13:37 +0100)
committerTakashi Iwai <tiwai@suse.de>
Mon, 16 Jul 2018 14:35:09 +0000 (16:35 +0200)
Current support for UAC2 Processing Units does the parsing
as one control per bit in the bitmap. However, the UAC2 spec
defines the controls as bit pairs where b01 means read-only
and b11 means read/write control.

This patch fixes that and uses the helper functions for checking
controls readability/writability when the control is defined as
bit pairs (UAC2 and UAC3).

Signed-off-by: Jorge Sanjuan <jorge.sanjuan@codethink.co.uk>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
sound/usb/mixer.c

index a51f2320a3dd24eabae63f0bc44f2e2e8f11a76f..bfb3484096a68b874d6f1fb7a6b551d7e7e60e3d 100644 (file)
@@ -2300,8 +2300,16 @@ static int build_audio_procunit(struct mixer_build *state, int unitid,
        for (valinfo = info->values; valinfo->control; valinfo++) {
                __u8 *controls = uac_processing_unit_bmControls(desc, state->mixer->protocol);
 
-               if (!(controls[valinfo->control / 8] & (1 << ((valinfo->control % 8) - 1))))
-                       continue;
+               if (state->mixer->protocol == UAC_VERSION_1) {
+                       if (!(controls[valinfo->control / 8] &
+                                       (1 << ((valinfo->control % 8) - 1))))
+                               continue;
+               } else { /* UAC_VERSION_2/3 */
+                       if (!uac_v2v3_control_is_readable(controls[valinfo->control / 8],
+                                                         valinfo->control))
+                               continue;
+               }
+
                map = find_map(state->map, unitid, valinfo->control);
                if (check_ignored_ctl(map))
                        continue;
@@ -2313,6 +2321,11 @@ static int build_audio_procunit(struct mixer_build *state, int unitid,
                cval->val_type = valinfo->val_type;
                cval->channels = 1;
 
+               if (state->mixer->protocol > UAC_VERSION_1 &&
+                   !uac_v2v3_control_is_writeable(controls[valinfo->control / 8],
+                                                  valinfo->control))
+                       cval->master_readonly = 1;
+
                /* get min/max values */
                if (type == UAC_PROCESS_UP_DOWNMIX && cval->control == UAC_UD_MODE_SELECT) {
                        __u8 *control_spec = uac_processing_unit_specific(desc, state->mixer->protocol);