ALSA: hda/tas2781: remove sound controls in unbind

Remove sound controls in hda_unbind to make
module loadable after module unload.

Add a driver specific struct (tas2781_hda) to store
the controls.

This patch depends on patch:
ALSA: hda/tas2781: do not use regcache

Fixes: 5be27f1e3e ("ALSA: hda/tas2781: Add tas2781 HDA driver")
CC: stable@vger.kernel.org
Signed-off-by: Gergo Koteles <soyer@irl.hu>
Link: https://lore.kernel.org/r/362aa3e2f81b9259a3e5222f576bec5debfc5e88.1703204848.git.soyer@irl.hu
Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
Gergo Koteles 2023-12-22 01:34:48 +01:00 committed by Takashi Iwai
parent e7aa105657
commit 4e7914eb1d

View File

@ -65,6 +65,15 @@ enum calib_data {
CALIB_MAX CALIB_MAX
}; };
struct tas2781_hda {
struct device *dev;
struct tasdevice_priv *priv;
struct snd_kcontrol *dsp_prog_ctl;
struct snd_kcontrol *dsp_conf_ctl;
struct snd_kcontrol *prof_ctl;
struct snd_kcontrol *snd_ctls[3];
};
static int tas2781_get_i2c_res(struct acpi_resource *ares, void *data) static int tas2781_get_i2c_res(struct acpi_resource *ares, void *data)
{ {
struct tasdevice_priv *tas_priv = data; struct tasdevice_priv *tas_priv = data;
@ -125,26 +134,26 @@ static int tas2781_read_acpi(struct tasdevice_priv *p, const char *hid)
static void tas2781_hda_playback_hook(struct device *dev, int action) static void tas2781_hda_playback_hook(struct device *dev, int action)
{ {
struct tasdevice_priv *tas_priv = dev_get_drvdata(dev); struct tas2781_hda *tas_hda = dev_get_drvdata(dev);
dev_dbg(tas_priv->dev, "%s: action = %d\n", __func__, action); dev_dbg(tas_hda->dev, "%s: action = %d\n", __func__, action);
switch (action) { switch (action) {
case HDA_GEN_PCM_ACT_OPEN: case HDA_GEN_PCM_ACT_OPEN:
pm_runtime_get_sync(dev); pm_runtime_get_sync(dev);
mutex_lock(&tas_priv->codec_lock); mutex_lock(&tas_hda->priv->codec_lock);
tasdevice_tuning_switch(tas_priv, 0); tasdevice_tuning_switch(tas_hda->priv, 0);
mutex_unlock(&tas_priv->codec_lock); mutex_unlock(&tas_hda->priv->codec_lock);
break; break;
case HDA_GEN_PCM_ACT_CLOSE: case HDA_GEN_PCM_ACT_CLOSE:
mutex_lock(&tas_priv->codec_lock); mutex_lock(&tas_hda->priv->codec_lock);
tasdevice_tuning_switch(tas_priv, 1); tasdevice_tuning_switch(tas_hda->priv, 1);
mutex_unlock(&tas_priv->codec_lock); mutex_unlock(&tas_hda->priv->codec_lock);
pm_runtime_mark_last_busy(dev); pm_runtime_mark_last_busy(dev);
pm_runtime_put_autosuspend(dev); pm_runtime_put_autosuspend(dev);
break; break;
default: default:
dev_dbg(tas_priv->dev, "Playback action not supported: %d\n", dev_dbg(tas_hda->dev, "Playback action not supported: %d\n",
action); action);
break; break;
} }
@ -477,9 +486,28 @@ static int tas2781_save_calibration(struct tasdevice_priv *tas_priv)
return 0; return 0;
} }
static void tas2781_hda_remove_controls(struct tas2781_hda *tas_hda)
{
struct hda_codec *codec = tas_hda->priv->codec;
if (tas_hda->dsp_prog_ctl)
snd_ctl_remove(codec->card, tas_hda->dsp_prog_ctl);
if (tas_hda->dsp_conf_ctl)
snd_ctl_remove(codec->card, tas_hda->dsp_conf_ctl);
for (int i = ARRAY_SIZE(tas_hda->snd_ctls) - 1; i >= 0; i--)
if (tas_hda->snd_ctls[i])
snd_ctl_remove(codec->card, tas_hda->snd_ctls[i]);
if (tas_hda->prof_ctl)
snd_ctl_remove(codec->card, tas_hda->prof_ctl);
}
static void tasdev_fw_ready(const struct firmware *fmw, void *context) static void tasdev_fw_ready(const struct firmware *fmw, void *context)
{ {
struct tasdevice_priv *tas_priv = context; struct tasdevice_priv *tas_priv = context;
struct tas2781_hda *tas_hda = dev_get_drvdata(tas_priv->dev);
struct hda_codec *codec = tas_priv->codec; struct hda_codec *codec = tas_priv->codec;
int i, ret; int i, ret;
@ -490,8 +518,8 @@ static void tasdev_fw_ready(const struct firmware *fmw, void *context)
if (ret) if (ret)
goto out; goto out;
ret = snd_ctl_add(codec->card, tas_hda->prof_ctl = snd_ctl_new1(&tas2781_prof_ctrl, tas_priv);
snd_ctl_new1(&tas2781_prof_ctrl, tas_priv)); ret = snd_ctl_add(codec->card, tas_hda->prof_ctl);
if (ret) { if (ret) {
dev_err(tas_priv->dev, dev_err(tas_priv->dev,
"Failed to add KControl %s = %d\n", "Failed to add KControl %s = %d\n",
@ -500,8 +528,9 @@ static void tasdev_fw_ready(const struct firmware *fmw, void *context)
} }
for (i = 0; i < ARRAY_SIZE(tas2781_snd_controls); i++) { for (i = 0; i < ARRAY_SIZE(tas2781_snd_controls); i++) {
ret = snd_ctl_add(codec->card, tas_hda->snd_ctls[i] = snd_ctl_new1(&tas2781_snd_controls[i],
snd_ctl_new1(&tas2781_snd_controls[i], tas_priv)); tas_priv);
ret = snd_ctl_add(codec->card, tas_hda->snd_ctls[i]);
if (ret) { if (ret) {
dev_err(tas_priv->dev, dev_err(tas_priv->dev,
"Failed to add KControl %s = %d\n", "Failed to add KControl %s = %d\n",
@ -523,8 +552,9 @@ static void tasdev_fw_ready(const struct firmware *fmw, void *context)
goto out; goto out;
} }
ret = snd_ctl_add(codec->card, tas_hda->dsp_prog_ctl = snd_ctl_new1(&tas2781_dsp_prog_ctrl,
snd_ctl_new1(&tas2781_dsp_prog_ctrl, tas_priv)); tas_priv);
ret = snd_ctl_add(codec->card, tas_hda->dsp_prog_ctl);
if (ret) { if (ret) {
dev_err(tas_priv->dev, dev_err(tas_priv->dev,
"Failed to add KControl %s = %d\n", "Failed to add KControl %s = %d\n",
@ -532,8 +562,9 @@ static void tasdev_fw_ready(const struct firmware *fmw, void *context)
goto out; goto out;
} }
ret = snd_ctl_add(codec->card, tas_hda->dsp_conf_ctl = snd_ctl_new1(&tas2781_dsp_conf_ctrl,
snd_ctl_new1(&tas2781_dsp_conf_ctrl, tas_priv)); tas_priv);
ret = snd_ctl_add(codec->card, tas_hda->dsp_conf_ctl);
if (ret) { if (ret) {
dev_err(tas_priv->dev, dev_err(tas_priv->dev,
"Failed to add KControl %s = %d\n", "Failed to add KControl %s = %d\n",
@ -554,27 +585,27 @@ static void tasdev_fw_ready(const struct firmware *fmw, void *context)
tas2781_save_calibration(tas_priv); tas2781_save_calibration(tas_priv);
out: out:
mutex_unlock(&tas_priv->codec_lock); mutex_unlock(&tas_hda->priv->codec_lock);
if (fmw) if (fmw)
release_firmware(fmw); release_firmware(fmw);
pm_runtime_mark_last_busy(tas_priv->dev); pm_runtime_mark_last_busy(tas_hda->dev);
pm_runtime_put_autosuspend(tas_priv->dev); pm_runtime_put_autosuspend(tas_hda->dev);
} }
static int tas2781_hda_bind(struct device *dev, struct device *master, static int tas2781_hda_bind(struct device *dev, struct device *master,
void *master_data) void *master_data)
{ {
struct tasdevice_priv *tas_priv = dev_get_drvdata(dev); struct tas2781_hda *tas_hda = dev_get_drvdata(dev);
struct hda_component *comps = master_data; struct hda_component *comps = master_data;
struct hda_codec *codec; struct hda_codec *codec;
unsigned int subid; unsigned int subid;
int ret; int ret;
if (!comps || tas_priv->index < 0 || if (!comps || tas_hda->priv->index < 0 ||
tas_priv->index >= HDA_MAX_COMPONENTS) tas_hda->priv->index >= HDA_MAX_COMPONENTS)
return -EINVAL; return -EINVAL;
comps = &comps[tas_priv->index]; comps = &comps[tas_hda->priv->index];
if (comps->dev) if (comps->dev)
return -EBUSY; return -EBUSY;
@ -583,10 +614,10 @@ static int tas2781_hda_bind(struct device *dev, struct device *master,
switch (subid) { switch (subid) {
case 0x17aa: case 0x17aa:
tas_priv->catlog_id = LENOVO; tas_hda->priv->catlog_id = LENOVO;
break; break;
default: default:
tas_priv->catlog_id = OTHERS; tas_hda->priv->catlog_id = OTHERS;
break; break;
} }
@ -596,7 +627,7 @@ static int tas2781_hda_bind(struct device *dev, struct device *master,
strscpy(comps->name, dev_name(dev), sizeof(comps->name)); strscpy(comps->name, dev_name(dev), sizeof(comps->name));
ret = tascodec_init(tas_priv, codec, tasdev_fw_ready); ret = tascodec_init(tas_hda->priv, codec, tasdev_fw_ready);
if (!ret) if (!ret)
comps->playback_hook = tas2781_hda_playback_hook; comps->playback_hook = tas2781_hda_playback_hook;
@ -609,9 +640,9 @@ static int tas2781_hda_bind(struct device *dev, struct device *master,
static void tas2781_hda_unbind(struct device *dev, static void tas2781_hda_unbind(struct device *dev,
struct device *master, void *master_data) struct device *master, void *master_data)
{ {
struct tasdevice_priv *tas_priv = dev_get_drvdata(dev); struct tas2781_hda *tas_hda = dev_get_drvdata(dev);
struct hda_component *comps = master_data; struct hda_component *comps = master_data;
comps = &comps[tas_priv->index]; comps = &comps[tas_hda->priv->index];
if (comps->dev == dev) { if (comps->dev == dev) {
comps->dev = NULL; comps->dev = NULL;
@ -619,10 +650,12 @@ static void tas2781_hda_unbind(struct device *dev,
comps->playback_hook = NULL; comps->playback_hook = NULL;
} }
tasdevice_config_info_remove(tas_priv); tas2781_hda_remove_controls(tas_hda);
tasdevice_dsp_remove(tas_priv);
tas_priv->fw_state = TASDEVICE_DSP_FW_PENDING; tasdevice_config_info_remove(tas_hda->priv);
tasdevice_dsp_remove(tas_hda->priv);
tas_hda->priv->fw_state = TASDEVICE_DSP_FW_PENDING;
} }
static const struct component_ops tas2781_hda_comp_ops = { static const struct component_ops tas2781_hda_comp_ops = {
@ -632,21 +665,21 @@ static const struct component_ops tas2781_hda_comp_ops = {
static void tas2781_hda_remove(struct device *dev) static void tas2781_hda_remove(struct device *dev)
{ {
struct tasdevice_priv *tas_priv = dev_get_drvdata(dev); struct tas2781_hda *tas_hda = dev_get_drvdata(dev);
pm_runtime_get_sync(tas_priv->dev); pm_runtime_get_sync(tas_hda->dev);
pm_runtime_disable(tas_priv->dev); pm_runtime_disable(tas_hda->dev);
component_del(tas_priv->dev, &tas2781_hda_comp_ops); component_del(tas_hda->dev, &tas2781_hda_comp_ops);
pm_runtime_put_noidle(tas_priv->dev); pm_runtime_put_noidle(tas_hda->dev);
tasdevice_remove(tas_priv); tasdevice_remove(tas_hda->priv);
} }
static int tas2781_hda_i2c_probe(struct i2c_client *clt) static int tas2781_hda_i2c_probe(struct i2c_client *clt)
{ {
struct tasdevice_priv *tas_priv; struct tas2781_hda *tas_hda;
const char *device_name; const char *device_name;
int ret; int ret;
@ -655,37 +688,42 @@ static int tas2781_hda_i2c_probe(struct i2c_client *clt)
else else
return -ENODEV; return -ENODEV;
tas_priv = tasdevice_kzalloc(clt); tas_hda = devm_kzalloc(&clt->dev, sizeof(*tas_hda), GFP_KERNEL);
if (!tas_priv) if (!tas_hda)
return -ENOMEM; return -ENOMEM;
dev_set_drvdata(&clt->dev, tas_priv); dev_set_drvdata(&clt->dev, tas_hda);
tas_hda->dev = &clt->dev;
tas_priv->irq_info.irq = clt->irq; tas_hda->priv = tasdevice_kzalloc(clt);
ret = tas2781_read_acpi(tas_priv, device_name); if (!tas_hda->priv)
return -ENOMEM;
tas_hda->priv->irq_info.irq = clt->irq;
ret = tas2781_read_acpi(tas_hda->priv, device_name);
if (ret) if (ret)
return dev_err_probe(tas_priv->dev, ret, return dev_err_probe(tas_hda->dev, ret,
"Platform not supported\n"); "Platform not supported\n");
ret = tasdevice_init(tas_priv); ret = tasdevice_init(tas_hda->priv);
if (ret) if (ret)
goto err; goto err;
pm_runtime_set_autosuspend_delay(tas_priv->dev, 3000); pm_runtime_set_autosuspend_delay(tas_hda->dev, 3000);
pm_runtime_use_autosuspend(tas_priv->dev); pm_runtime_use_autosuspend(tas_hda->dev);
pm_runtime_mark_last_busy(tas_priv->dev); pm_runtime_mark_last_busy(tas_hda->dev);
pm_runtime_set_active(tas_priv->dev); pm_runtime_set_active(tas_hda->dev);
pm_runtime_get_noresume(tas_priv->dev); pm_runtime_get_noresume(tas_hda->dev);
pm_runtime_enable(tas_priv->dev); pm_runtime_enable(tas_hda->dev);
pm_runtime_put_autosuspend(tas_priv->dev); pm_runtime_put_autosuspend(tas_hda->dev);
tas2781_reset(tas_priv); tas2781_reset(tas_hda->priv);
ret = component_add(tas_priv->dev, &tas2781_hda_comp_ops); ret = component_add(tas_hda->dev, &tas2781_hda_comp_ops);
if (ret) { if (ret) {
dev_err(tas_priv->dev, "Register component failed: %d\n", ret); dev_err(tas_hda->dev, "Register component failed: %d\n", ret);
pm_runtime_disable(tas_priv->dev); pm_runtime_disable(tas_hda->dev);
} }
err: err:
@ -701,66 +739,65 @@ static void tas2781_hda_i2c_remove(struct i2c_client *clt)
static int tas2781_runtime_suspend(struct device *dev) static int tas2781_runtime_suspend(struct device *dev)
{ {
struct tasdevice_priv *tas_priv = dev_get_drvdata(dev); struct tas2781_hda *tas_hda = dev_get_drvdata(dev);
int i; int i;
dev_dbg(tas_priv->dev, "Runtime Suspend\n"); dev_dbg(tas_hda->dev, "Runtime Suspend\n");
mutex_lock(&tas_priv->codec_lock); mutex_lock(&tas_hda->priv->codec_lock);
if (tas_priv->playback_started) { if (tas_hda->priv->playback_started) {
tasdevice_tuning_switch(tas_priv, 1); tasdevice_tuning_switch(tas_hda->priv, 1);
tas_priv->playback_started = false; tas_hda->priv->playback_started = false;
} }
for (i = 0; i < tas_priv->ndev; i++) { for (i = 0; i < tas_hda->priv->ndev; i++) {
tas_priv->tasdevice[i].cur_book = -1; tas_hda->priv->tasdevice[i].cur_book = -1;
tas_priv->tasdevice[i].cur_prog = -1; tas_hda->priv->tasdevice[i].cur_prog = -1;
tas_priv->tasdevice[i].cur_conf = -1; tas_hda->priv->tasdevice[i].cur_conf = -1;
} }
mutex_unlock(&tas_hda->priv->codec_lock);
mutex_unlock(&tas_priv->codec_lock);
return 0; return 0;
} }
static int tas2781_runtime_resume(struct device *dev) static int tas2781_runtime_resume(struct device *dev)
{ {
struct tasdevice_priv *tas_priv = dev_get_drvdata(dev); struct tas2781_hda *tas_hda = dev_get_drvdata(dev);
unsigned long calib_data_sz = unsigned long calib_data_sz =
tas_priv->ndev * TASDEVICE_SPEAKER_CALIBRATION_SIZE; tas_hda->priv->ndev * TASDEVICE_SPEAKER_CALIBRATION_SIZE;
dev_dbg(tas_priv->dev, "Runtime Resume\n"); dev_dbg(tas_hda->dev, "Runtime Resume\n");
mutex_lock(&tas_priv->codec_lock); mutex_lock(&tas_hda->priv->codec_lock);
tasdevice_prmg_load(tas_priv, tas_priv->cur_prog); tasdevice_prmg_load(tas_hda->priv, tas_hda->priv->cur_prog);
/* If calibrated data occurs error, dsp will still works with default /* If calibrated data occurs error, dsp will still works with default
* calibrated data inside algo. * calibrated data inside algo.
*/ */
if (tas_priv->cali_data.total_sz > calib_data_sz) if (tas_hda->priv->cali_data.total_sz > calib_data_sz)
tas2781_apply_calib(tas_priv); tas2781_apply_calib(tas_hda->priv);
mutex_unlock(&tas_priv->codec_lock); mutex_unlock(&tas_hda->priv->codec_lock);
return 0; return 0;
} }
static int tas2781_system_suspend(struct device *dev) static int tas2781_system_suspend(struct device *dev)
{ {
struct tasdevice_priv *tas_priv = dev_get_drvdata(dev); struct tas2781_hda *tas_hda = dev_get_drvdata(dev);
int ret; int ret;
dev_dbg(tas_priv->dev, "System Suspend\n"); dev_dbg(tas_hda->priv->dev, "System Suspend\n");
ret = pm_runtime_force_suspend(dev); ret = pm_runtime_force_suspend(dev);
if (ret) if (ret)
return ret; return ret;
/* Shutdown chip before system suspend */ /* Shutdown chip before system suspend */
tasdevice_tuning_switch(tas_priv, 1); tasdevice_tuning_switch(tas_hda->priv, 1);
/* /*
* Reset GPIO may be shared, so cannot reset here. * Reset GPIO may be shared, so cannot reset here.
@ -771,33 +808,33 @@ static int tas2781_system_suspend(struct device *dev)
static int tas2781_system_resume(struct device *dev) static int tas2781_system_resume(struct device *dev)
{ {
struct tasdevice_priv *tas_priv = dev_get_drvdata(dev); struct tas2781_hda *tas_hda = dev_get_drvdata(dev);
unsigned long calib_data_sz = unsigned long calib_data_sz =
tas_priv->ndev * TASDEVICE_SPEAKER_CALIBRATION_SIZE; tas_hda->priv->ndev * TASDEVICE_SPEAKER_CALIBRATION_SIZE;
int i, ret; int i, ret;
dev_dbg(tas_priv->dev, "System Resume\n"); dev_info(tas_hda->priv->dev, "System Resume\n");
ret = pm_runtime_force_resume(dev); ret = pm_runtime_force_resume(dev);
if (ret) if (ret)
return ret; return ret;
mutex_lock(&tas_priv->codec_lock); mutex_lock(&tas_hda->priv->codec_lock);
for (i = 0; i < tas_priv->ndev; i++) { for (i = 0; i < tas_hda->priv->ndev; i++) {
tas_priv->tasdevice[i].cur_book = -1; tas_hda->priv->tasdevice[i].cur_book = -1;
tas_priv->tasdevice[i].cur_prog = -1; tas_hda->priv->tasdevice[i].cur_prog = -1;
tas_priv->tasdevice[i].cur_conf = -1; tas_hda->priv->tasdevice[i].cur_conf = -1;
} }
tas2781_reset(tas_priv); tas2781_reset(tas_hda->priv);
tasdevice_prmg_load(tas_priv, tas_priv->cur_prog); tasdevice_prmg_load(tas_hda->priv, tas_hda->priv->cur_prog);
/* If calibrated data occurs error, dsp will still work with default /* If calibrated data occurs error, dsp will still work with default
* calibrated data inside algo. * calibrated data inside algo.
*/ */
if (tas_priv->cali_data.total_sz > calib_data_sz) if (tas_hda->priv->cali_data.total_sz > calib_data_sz)
tas2781_apply_calib(tas_priv); tas2781_apply_calib(tas_hda->priv);
mutex_unlock(&tas_priv->codec_lock); mutex_unlock(&tas_hda->priv->codec_lock);
return 0; return 0;
} }