{ CS35L41_AMP_DIG_VOL_CTRL,     0x0000A678 }, // AMP_VOL_PCM Mute
 };
 
-static int cs35l41_control_add(struct cs_dsp_coeff_ctl *cs_ctl)
+static void cs35l41_add_controls(struct cs35l41_hda *cs35l41)
 {
-       struct cs35l41_hda *cs35l41 = container_of(cs_ctl->dsp, struct cs35l41_hda, cs_dsp);
        struct hda_cs_dsp_ctl_info info;
 
        info.device_name = cs35l41->amp_name;
        info.fw_type = cs35l41->firmware_type;
        info.card = cs35l41->codec->card;
 
-       return hda_cs_dsp_control_add(cs_ctl, &info);
+       hda_cs_dsp_add_controls(&cs35l41->cs_dsp, &info);
 }
 
 static const struct cs_dsp_client_ops client_ops = {
-       .control_add = cs35l41_control_add,
        .control_remove = hda_cs_dsp_control_remove,
 };
 
        if (ret)
                goto err_release;
 
+       cs35l41_add_controls(cs35l41);
+
        ret = cs35l41_save_calibration(cs35l41);
 
 err_release:
 
        return out;
 }
 
-static int hda_cs_dsp_add_kcontrol(struct hda_cs_dsp_coeff_ctl *ctl, const char *name)
+static void hda_cs_dsp_add_kcontrol(struct hda_cs_dsp_coeff_ctl *ctl, const char *name)
 {
        struct cs_dsp_coeff_ctl *cs_ctl = ctl->cs_ctl;
        struct snd_kcontrol_new kcontrol = {0};
        if (cs_ctl->len > ADSP_MAX_STD_CTRL_SIZE) {
                dev_err(cs_ctl->dsp->dev, "KControl %s: length %zu exceeds maximum %d\n", name,
                        cs_ctl->len, ADSP_MAX_STD_CTRL_SIZE);
-               return -EINVAL;
+               return;
        }
 
        kcontrol.name = name;
        /* Save ctl inside private_data, ctl is owned by cs_dsp,
         * and will be freed when cs_dsp removes the control */
        kctl = snd_ctl_new1(&kcontrol, (void *)ctl);
-       if (!kctl) {
-               ret = -ENOMEM;
-               return ret;
-       }
+       if (!kctl)
+               return;
 
        ret = snd_ctl_add(ctl->card, kctl);
        if (ret) {
                dev_err(cs_ctl->dsp->dev, "Failed to add KControl %s = %d\n", kcontrol.name, ret);
-               return ret;
+               return;
        }
 
        dev_dbg(cs_ctl->dsp->dev, "Added KControl: %s\n", kcontrol.name);
        ctl->kctl = kctl;
-
-       return 0;
 }
 
-int hda_cs_dsp_control_add(struct cs_dsp_coeff_ctl *cs_ctl, struct hda_cs_dsp_ctl_info *info)
+static void hda_cs_dsp_control_add(struct cs_dsp_coeff_ctl *cs_ctl,
+                                  const struct hda_cs_dsp_ctl_info *info)
 {
        struct cs_dsp *cs_dsp = cs_ctl->dsp;
        char name[SNDRV_CTL_ELEM_ID_NAME_MAXLEN];
        const char *region_name;
        int ret;
 
-       if (cs_ctl->flags & WMFW_CTL_FLAG_SYS)
-               return 0;
-
        region_name = cs_dsp_mem_region_name(cs_ctl->alg_region.type);
        if (!region_name) {
-               dev_err(cs_dsp->dev, "Unknown region type: %d\n", cs_ctl->alg_region.type);
-               return -EINVAL;
+               dev_warn(cs_dsp->dev, "Unknown region type: %d\n", cs_ctl->alg_region.type);
+               return;
        }
 
        ret = scnprintf(name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN, "%s %s %.12s %x", info->device_name,
 
        ctl = kzalloc(sizeof(*ctl), GFP_KERNEL);
        if (!ctl)
-               return -ENOMEM;
+               return;
 
        ctl->cs_ctl = cs_ctl;
        ctl->card = info->card;
        cs_ctl->priv = ctl;
 
-       ret = hda_cs_dsp_add_kcontrol(ctl, name);
-       if (ret) {
-               dev_err(cs_dsp->dev, "Error (%d) adding control %s\n", ret, name);
-               kfree(ctl);
-               return ret;
-       }
+       hda_cs_dsp_add_kcontrol(ctl, name);
+}
 
-       return 0;
+void hda_cs_dsp_add_controls(struct cs_dsp *dsp, const struct hda_cs_dsp_ctl_info *info)
+{
+       struct cs_dsp_coeff_ctl *cs_ctl;
+
+       /*
+        * pwr_lock would cause mutex inversion with ALSA control lock compared
+        * to the get/put functions.
+        * It is safe to walk the list without holding a mutex because entries
+        * are persistent and only cs_dsp_power_up() or cs_dsp_remove() can
+        * change the list.
+        */
+       lockdep_assert_not_held(&dsp->pwr_lock);
+
+       list_for_each_entry(cs_ctl, &dsp->ctl_list, list) {
+               if (cs_ctl->flags & WMFW_CTL_FLAG_SYS)
+                       continue;
+
+               if (cs_ctl->priv)
+                       continue;
+
+               hda_cs_dsp_control_add(cs_ctl, info);
+       }
 }
-EXPORT_SYMBOL_NS_GPL(hda_cs_dsp_control_add, SND_HDA_CS_DSP_CONTROLS);
+EXPORT_SYMBOL_NS_GPL(hda_cs_dsp_add_controls, SND_HDA_CS_DSP_CONTROLS);
 
 void hda_cs_dsp_control_remove(struct cs_dsp_coeff_ctl *cs_ctl)
 {