ALSA: hda: Fix the control element identification for multiple codecs
authorJaroslav Kysela <perex@perex.cz>
Thu, 2 Feb 2023 09:20:13 +0000 (10:20 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 10 Mar 2023 08:33:20 +0000 (09:33 +0100)
[ Upstream commit d045bceff5a904bd79d71dede9f927c00ce4906f ]

Some motherboards have multiple HDA codecs connected to the serial bus.
The current code may create multiple mixer controls with the almost
identical identification.

The current code use id.device field from the control element structure
to store the codec address to avoid such clashes for multiple codecs.
Unfortunately, the user space do not handle this correctly. For mixer
controls, only name and index are used for the identifiers.

This patch fixes this problem to compose the index using the codec
address as an offset in case, when the control already exists. It is
really unlikely that one codec will create 10 similar controls.

This patch adds new kernel module parameter 'ctl_dev_id' to allow
select the old behaviour, too. The CONFIG_SND_HDA_CTL_DEV_ID Kconfig
option sets the default value.

BugLink: https://github.com/alsa-project/alsa-lib/issues/294
BugLink: https://github.com/alsa-project/alsa-lib/issues/205
Fixes: 54d174031576 ("[ALSA] hda-codec - Fix connection list parsing")
Fixes: 1afe206ab699 ("ALSA: hda - Try to find an empty control index when it's occupied")
Signed-off-by: Jaroslav Kysela <perex@perex.cz>
Link: https://lore.kernel.org/r/20230202092013.4066998-1-perex@perex.cz
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Sasha Levin <sashal@kernel.org>
include/sound/hda_codec.h
sound/pci/hda/Kconfig
sound/pci/hda/hda_codec.c
sound/pci/hda/hda_controller.c
sound/pci/hda/hda_controller.h
sound/pci/hda/hda_intel.c

index eba23daf2c290ba1aa2428e85cea287f6d44582b..bbb7805e85d8e2281e77592e087b2ea4ec404960 100644 (file)
@@ -259,6 +259,7 @@ struct hda_codec {
        unsigned int relaxed_resume:1;  /* don't resume forcibly for jack */
        unsigned int forced_resume:1; /* forced resume for jack */
        unsigned int no_stream_clean_at_suspend:1; /* do not clean streams at suspend */
+       unsigned int ctl_dev_id:1; /* old control element id build behaviour */
 
 #ifdef CONFIG_PM
        unsigned long power_on_acct;
index a8e8cf98befa1ccd41dddc584b41bc0dd0c9c606..d29d8372a3c0455d89da0d23f037ad94dca11fd2 100644 (file)
@@ -302,6 +302,20 @@ config SND_HDA_INTEL_HDMI_SILENT_STREAM
          This feature can impact power consumption as resources
          are kept reserved both at transmitter and receiver.
 
+config SND_HDA_CTL_DEV_ID
+       bool "Use the device identifier field for controls"
+       depends on SND_HDA_INTEL
+       help
+         Say Y to use the device identifier field for (mixer)
+         controls (old behaviour until this option is available).
+
+         When enabled, the multiple HDA codecs may set the device
+         field in control (mixer) element identifiers. The use
+         of this field is not recommended and defined for mixer controls.
+
+         The old behaviour (Y) is obsolete and will be removed. Consider
+         to not enable this option.
+
 endif
 
 endmenu
index 2e728aad677136470cbb990fa85df6c37e2d8b32..9f79c0ac2bda717a4b72024c42f9e939ebd90777 100644 (file)
@@ -3389,7 +3389,12 @@ int snd_hda_add_new_ctls(struct hda_codec *codec,
                        kctl = snd_ctl_new1(knew, codec);
                        if (!kctl)
                                return -ENOMEM;
-                       if (addr > 0)
+                       /* Do not use the id.device field for MIXER elements.
+                        * This field is for real device numbers (like PCM) but codecs
+                        * are hidden components from the user space view (unrelated
+                        * to the mixer element identification).
+                        */
+                       if (addr > 0 && codec->ctl_dev_id)
                                kctl->id.device = addr;
                        if (idx > 0)
                                kctl->id.index = idx;
@@ -3400,9 +3405,11 @@ int snd_hda_add_new_ctls(struct hda_codec *codec,
                         * the codec addr; if it still fails (or it's the
                         * primary codec), then try another control index
                         */
-                       if (!addr && codec->core.addr)
+                       if (!addr && codec->core.addr) {
                                addr = codec->core.addr;
-                       else if (!idx && !knew->index) {
+                               if (!codec->ctl_dev_id)
+                                       idx += 10 * addr;
+                       } else if (!idx && !knew->index) {
                                idx = find_empty_mixer_ctl_idx(codec,
                                                               knew->name, 0);
                                if (idx <= 0)
index 0ff286b7b66bee3965012e69074baba863479a70..083df287c1a487bb2651b1eeda41799bca35cdbf 100644 (file)
@@ -1231,6 +1231,7 @@ int azx_probe_codecs(struct azx *chip, unsigned int max_slots)
                                continue;
                        codec->jackpoll_interval = chip->jackpoll_interval;
                        codec->beep_mode = chip->beep_mode;
+                       codec->ctl_dev_id = chip->ctl_dev_id;
                        codecs++;
                }
        }
index f5bf295eb83078d2586a5985aaa1af4eb2840116..8556031bcd68e48c4ad1c6d4d4a094fce0b2a3fb 100644 (file)
@@ -124,6 +124,7 @@ struct azx {
        /* HD codec */
        int  codec_probe_mask; /* copied from probe_mask option */
        unsigned int beep_mode;
+       bool ctl_dev_id;
 
 #ifdef CONFIG_SND_HDA_PATCH_LOADER
        const struct firmware *fw;
index 87002670c0c925e94e55a764b8d54daf61ca9647..2dbc082076f69d2c45c73a259d9062e12b3956d0 100644 (file)
@@ -119,6 +119,7 @@ static bool beep_mode[SNDRV_CARDS] = {[0 ... (SNDRV_CARDS-1)] =
                                        CONFIG_SND_HDA_INPUT_BEEP_MODE};
 #endif
 static bool dmic_detect = 1;
+static bool ctl_dev_id = IS_ENABLED(CONFIG_SND_HDA_CTL_DEV_ID) ? 1 : 0;
 
 module_param_array(index, int, NULL, 0444);
 MODULE_PARM_DESC(index, "Index value for Intel HD audio interface.");
@@ -157,6 +158,8 @@ module_param(dmic_detect, bool, 0444);
 MODULE_PARM_DESC(dmic_detect, "Allow DSP driver selection (bypass this driver) "
                             "(0=off, 1=on) (default=1); "
                 "deprecated, use snd-intel-dspcfg.dsp_driver option instead");
+module_param(ctl_dev_id, bool, 0444);
+MODULE_PARM_DESC(ctl_dev_id, "Use control device identifier (based on codec address).");
 
 #ifdef CONFIG_PM
 static int param_set_xint(const char *val, const struct kernel_param *kp);
@@ -2278,6 +2281,8 @@ static int azx_probe_continue(struct azx *chip)
        chip->beep_mode = beep_mode[dev];
 #endif
 
+       chip->ctl_dev_id = ctl_dev_id;
+
        /* create codec instances */
        if (bus->codec_mask) {
                err = azx_probe_codecs(chip, azx_max_codecs[chip->driver_type]);