mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2026-05-16 09:02:21 -04:00
ASoC: fsl: fix mixer-test failures
Shengjiu Wang <shengjiu.wang@nxp.com> says: Fix mixer-test failures. Mostly the issues are that event is not generated in put operation.
This commit is contained in:
@@ -54,6 +54,9 @@ static int fsl_easrc_iec958_put_bits(struct snd_kcontrol *kcontrol,
|
||||
unsigned int regval = ucontrol->value.integer.value[0];
|
||||
int ret;
|
||||
|
||||
if (regval < EASRC_WIDTH_16_BIT || regval > EASRC_WIDTH_24_BIT)
|
||||
return -EINVAL;
|
||||
|
||||
ret = (easrc_priv->bps_iec958[mc->regbase] != regval);
|
||||
|
||||
easrc_priv->bps_iec958[mc->regbase] = regval;
|
||||
@@ -70,22 +73,52 @@ static int fsl_easrc_iec958_get_bits(struct snd_kcontrol *kcontrol,
|
||||
struct soc_mreg_control *mc =
|
||||
(struct soc_mreg_control *)kcontrol->private_value;
|
||||
|
||||
ucontrol->value.enumerated.item[0] = easrc_priv->bps_iec958[mc->regbase];
|
||||
ucontrol->value.integer.value[0] = easrc_priv->bps_iec958[mc->regbase];
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int fsl_easrc_iec958_info(struct snd_kcontrol *kcontrol,
|
||||
struct snd_ctl_elem_info *uinfo)
|
||||
{
|
||||
uinfo->type = SNDRV_CTL_ELEM_TYPE_IEC958;
|
||||
uinfo->count = 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int fsl_easrc_get_reg(struct snd_kcontrol *kcontrol,
|
||||
struct snd_ctl_elem_value *ucontrol)
|
||||
{
|
||||
struct snd_soc_component *component = snd_kcontrol_chip(kcontrol);
|
||||
struct soc_mreg_control *mc =
|
||||
(struct soc_mreg_control *)kcontrol->private_value;
|
||||
unsigned int regval;
|
||||
struct fsl_asrc *easrc = snd_soc_component_get_drvdata(component);
|
||||
unsigned int *regval = (unsigned int *)ucontrol->value.iec958.status;
|
||||
int ret;
|
||||
|
||||
regval = snd_soc_component_read(component, mc->regbase);
|
||||
ret = regmap_read(easrc->regmap, REG_EASRC_CS0(mc->regbase), ®val[0]);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ucontrol->value.integer.value[0] = regval;
|
||||
ret = regmap_read(easrc->regmap, REG_EASRC_CS1(mc->regbase), ®val[1]);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = regmap_read(easrc->regmap, REG_EASRC_CS2(mc->regbase), ®val[2]);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = regmap_read(easrc->regmap, REG_EASRC_CS3(mc->regbase), ®val[3]);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = regmap_read(easrc->regmap, REG_EASRC_CS4(mc->regbase), ®val[4]);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = regmap_read(easrc->regmap, REG_EASRC_CS5(mc->regbase), ®val[5]);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -97,22 +130,62 @@ static int fsl_easrc_set_reg(struct snd_kcontrol *kcontrol,
|
||||
struct soc_mreg_control *mc =
|
||||
(struct soc_mreg_control *)kcontrol->private_value;
|
||||
struct fsl_asrc *easrc = snd_soc_component_get_drvdata(component);
|
||||
unsigned int regval = ucontrol->value.integer.value[0];
|
||||
bool changed;
|
||||
unsigned int *regval = (unsigned int *)ucontrol->value.iec958.status;
|
||||
bool changed, changed_all = false;
|
||||
int ret;
|
||||
|
||||
ret = regmap_update_bits_check(easrc->regmap, mc->regbase,
|
||||
GENMASK(31, 0), regval, &changed);
|
||||
if (ret != 0)
|
||||
ret = pm_runtime_resume_and_get(component->dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return changed;
|
||||
ret = regmap_update_bits_check(easrc->regmap, REG_EASRC_CS0(mc->regbase),
|
||||
GENMASK(31, 0), regval[0], &changed);
|
||||
if (ret != 0)
|
||||
goto err;
|
||||
changed_all |= changed;
|
||||
|
||||
ret = regmap_update_bits_check(easrc->regmap, REG_EASRC_CS1(mc->regbase),
|
||||
GENMASK(31, 0), regval[1], &changed);
|
||||
if (ret != 0)
|
||||
goto err;
|
||||
changed_all |= changed;
|
||||
|
||||
ret = regmap_update_bits_check(easrc->regmap, REG_EASRC_CS2(mc->regbase),
|
||||
GENMASK(31, 0), regval[2], &changed);
|
||||
if (ret != 0)
|
||||
goto err;
|
||||
changed_all |= changed;
|
||||
|
||||
ret = regmap_update_bits_check(easrc->regmap, REG_EASRC_CS3(mc->regbase),
|
||||
GENMASK(31, 0), regval[3], &changed);
|
||||
if (ret != 0)
|
||||
goto err;
|
||||
changed_all |= changed;
|
||||
|
||||
ret = regmap_update_bits_check(easrc->regmap, REG_EASRC_CS4(mc->regbase),
|
||||
GENMASK(31, 0), regval[4], &changed);
|
||||
if (ret != 0)
|
||||
goto err;
|
||||
changed_all |= changed;
|
||||
|
||||
ret = regmap_update_bits_check(easrc->regmap, REG_EASRC_CS5(mc->regbase),
|
||||
GENMASK(31, 0), regval[5], &changed);
|
||||
if (ret != 0)
|
||||
goto err;
|
||||
changed_all |= changed;
|
||||
err:
|
||||
pm_runtime_put_autosuspend(component->dev);
|
||||
|
||||
if (ret != 0)
|
||||
return ret;
|
||||
else
|
||||
return changed_all;
|
||||
}
|
||||
|
||||
#define SOC_SINGLE_REG_RW(xname, xreg) \
|
||||
{ .iface = SNDRV_CTL_ELEM_IFACE_PCM, .name = (xname), \
|
||||
.access = SNDRV_CTL_ELEM_ACCESS_READWRITE, \
|
||||
.info = snd_soc_info_xr_sx, .get = fsl_easrc_get_reg, \
|
||||
.info = fsl_easrc_iec958_info, .get = fsl_easrc_get_reg, \
|
||||
.put = fsl_easrc_set_reg, \
|
||||
.private_value = (unsigned long)&(struct soc_mreg_control) \
|
||||
{ .regbase = xreg, .regcount = 1, .nbits = 32, \
|
||||
@@ -143,30 +216,10 @@ static const struct snd_kcontrol_new fsl_easrc_snd_controls[] = {
|
||||
SOC_SINGLE_VAL_RW("Context 2 IEC958 Bits Per Sample", 2),
|
||||
SOC_SINGLE_VAL_RW("Context 3 IEC958 Bits Per Sample", 3),
|
||||
|
||||
SOC_SINGLE_REG_RW("Context 0 IEC958 CS0", REG_EASRC_CS0(0)),
|
||||
SOC_SINGLE_REG_RW("Context 1 IEC958 CS0", REG_EASRC_CS0(1)),
|
||||
SOC_SINGLE_REG_RW("Context 2 IEC958 CS0", REG_EASRC_CS0(2)),
|
||||
SOC_SINGLE_REG_RW("Context 3 IEC958 CS0", REG_EASRC_CS0(3)),
|
||||
SOC_SINGLE_REG_RW("Context 0 IEC958 CS1", REG_EASRC_CS1(0)),
|
||||
SOC_SINGLE_REG_RW("Context 1 IEC958 CS1", REG_EASRC_CS1(1)),
|
||||
SOC_SINGLE_REG_RW("Context 2 IEC958 CS1", REG_EASRC_CS1(2)),
|
||||
SOC_SINGLE_REG_RW("Context 3 IEC958 CS1", REG_EASRC_CS1(3)),
|
||||
SOC_SINGLE_REG_RW("Context 0 IEC958 CS2", REG_EASRC_CS2(0)),
|
||||
SOC_SINGLE_REG_RW("Context 1 IEC958 CS2", REG_EASRC_CS2(1)),
|
||||
SOC_SINGLE_REG_RW("Context 2 IEC958 CS2", REG_EASRC_CS2(2)),
|
||||
SOC_SINGLE_REG_RW("Context 3 IEC958 CS2", REG_EASRC_CS2(3)),
|
||||
SOC_SINGLE_REG_RW("Context 0 IEC958 CS3", REG_EASRC_CS3(0)),
|
||||
SOC_SINGLE_REG_RW("Context 1 IEC958 CS3", REG_EASRC_CS3(1)),
|
||||
SOC_SINGLE_REG_RW("Context 2 IEC958 CS3", REG_EASRC_CS3(2)),
|
||||
SOC_SINGLE_REG_RW("Context 3 IEC958 CS3", REG_EASRC_CS3(3)),
|
||||
SOC_SINGLE_REG_RW("Context 0 IEC958 CS4", REG_EASRC_CS4(0)),
|
||||
SOC_SINGLE_REG_RW("Context 1 IEC958 CS4", REG_EASRC_CS4(1)),
|
||||
SOC_SINGLE_REG_RW("Context 2 IEC958 CS4", REG_EASRC_CS4(2)),
|
||||
SOC_SINGLE_REG_RW("Context 3 IEC958 CS4", REG_EASRC_CS4(3)),
|
||||
SOC_SINGLE_REG_RW("Context 0 IEC958 CS5", REG_EASRC_CS5(0)),
|
||||
SOC_SINGLE_REG_RW("Context 1 IEC958 CS5", REG_EASRC_CS5(1)),
|
||||
SOC_SINGLE_REG_RW("Context 2 IEC958 CS5", REG_EASRC_CS5(2)),
|
||||
SOC_SINGLE_REG_RW("Context 3 IEC958 CS5", REG_EASRC_CS5(3)),
|
||||
SOC_SINGLE_REG_RW("Context 0 IEC958 CS", 0),
|
||||
SOC_SINGLE_REG_RW("Context 1 IEC958 CS", 1),
|
||||
SOC_SINGLE_REG_RW("Context 2 IEC958 CS", 2),
|
||||
SOC_SINGLE_REG_RW("Context 3 IEC958 CS", 3),
|
||||
};
|
||||
|
||||
/*
|
||||
|
||||
@@ -210,15 +210,23 @@ static int micfil_range_set(struct snd_kcontrol *kcontrol,
|
||||
(struct soc_mixer_control *)kcontrol->private_value;
|
||||
unsigned int shift = mc->shift;
|
||||
int max_range, new_range;
|
||||
int ret;
|
||||
|
||||
new_range = ucontrol->value.integer.value[0];
|
||||
max_range = micfil_get_max_range(micfil);
|
||||
if (new_range > max_range)
|
||||
dev_warn(&micfil->pdev->dev, "range makes channel %d data unreliable\n", shift / 4);
|
||||
|
||||
regmap_update_bits(micfil->regmap, REG_MICFIL_OUT_CTRL, 0xF << shift, new_range << shift);
|
||||
ret = pm_runtime_resume_and_get(cmpnt->dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
ret = snd_soc_component_update_bits(cmpnt, REG_MICFIL_OUT_CTRL, 0xF << shift,
|
||||
new_range << shift);
|
||||
|
||||
pm_runtime_put_autosuspend(cmpnt->dev);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int micfil_set_quality(struct fsl_micfil *micfil)
|
||||
@@ -281,10 +289,34 @@ static int micfil_quality_set(struct snd_kcontrol *kcontrol,
|
||||
{
|
||||
struct snd_soc_component *cmpnt = snd_kcontrol_chip(kcontrol);
|
||||
struct fsl_micfil *micfil = snd_soc_component_get_drvdata(cmpnt);
|
||||
int val = ucontrol->value.integer.value[0];
|
||||
bool change = false;
|
||||
int old_val;
|
||||
int ret;
|
||||
|
||||
micfil->quality = ucontrol->value.integer.value[0];
|
||||
if (val < QUALITY_HIGH || val > QUALITY_VLOW2)
|
||||
return -EINVAL;
|
||||
|
||||
return micfil_set_quality(micfil);
|
||||
if (micfil->quality != val) {
|
||||
ret = pm_runtime_resume_and_get(cmpnt->dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
old_val = micfil->quality;
|
||||
micfil->quality = val;
|
||||
ret = micfil_set_quality(micfil);
|
||||
|
||||
pm_runtime_put_autosuspend(cmpnt->dev);
|
||||
|
||||
if (ret) {
|
||||
micfil->quality = old_val;
|
||||
return ret;
|
||||
}
|
||||
|
||||
change = true;
|
||||
}
|
||||
|
||||
return change;
|
||||
}
|
||||
|
||||
static const char * const micfil_hwvad_enable[] = {
|
||||
@@ -343,6 +375,10 @@ static int micfil_put_dc_remover_state(struct snd_kcontrol *kcontrol,
|
||||
if (val < 0 || val > 3)
|
||||
return -EINVAL;
|
||||
|
||||
ret = pm_runtime_resume_and_get(comp->dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
micfil->dc_remover = val;
|
||||
|
||||
/* Calculate total value for all channels */
|
||||
@@ -352,10 +388,10 @@ static int micfil_put_dc_remover_state(struct snd_kcontrol *kcontrol,
|
||||
/* Update DC Remover mode for all channels */
|
||||
ret = snd_soc_component_update_bits(comp, REG_MICFIL_DC_CTRL,
|
||||
MICFIL_DC_CTRL_CONFIG, reg_val);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
pm_runtime_put_autosuspend(comp->dev);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int micfil_get_dc_remover_state(struct snd_kcontrol *kcontrol,
|
||||
@@ -377,10 +413,15 @@ static int hwvad_put_enable(struct snd_kcontrol *kcontrol,
|
||||
unsigned int *item = ucontrol->value.enumerated.item;
|
||||
struct fsl_micfil *micfil = snd_soc_component_get_drvdata(comp);
|
||||
int val = snd_soc_enum_item_to_val(e, item[0]);
|
||||
bool change = false;
|
||||
|
||||
if (val < 0 || val > 1)
|
||||
return -EINVAL;
|
||||
|
||||
change = (micfil->vad_enabled != val);
|
||||
micfil->vad_enabled = val;
|
||||
|
||||
return 0;
|
||||
return change;
|
||||
}
|
||||
|
||||
static int hwvad_get_enable(struct snd_kcontrol *kcontrol,
|
||||
@@ -402,13 +443,18 @@ static int hwvad_put_init_mode(struct snd_kcontrol *kcontrol,
|
||||
unsigned int *item = ucontrol->value.enumerated.item;
|
||||
struct fsl_micfil *micfil = snd_soc_component_get_drvdata(comp);
|
||||
int val = snd_soc_enum_item_to_val(e, item[0]);
|
||||
bool change = false;
|
||||
|
||||
if (val < MICFIL_HWVAD_ENVELOPE_MODE || val > MICFIL_HWVAD_ENERGY_MODE)
|
||||
return -EINVAL;
|
||||
|
||||
/* 0 - Envelope-based Mode
|
||||
* 1 - Energy-based Mode
|
||||
*/
|
||||
change = (micfil->vad_init_mode != val);
|
||||
micfil->vad_init_mode = val;
|
||||
|
||||
return 0;
|
||||
return change;
|
||||
}
|
||||
|
||||
static int hwvad_get_init_mode(struct snd_kcontrol *kcontrol,
|
||||
@@ -503,7 +549,13 @@ static const struct snd_kcontrol_new fsl_micfil_snd_controls[] = {
|
||||
SOC_SINGLE("HWVAD ZCD Adjustment", REG_MICFIL_VAD0_ZCD, 8, 15, 0),
|
||||
SOC_SINGLE("HWVAD ZCD And Behavior Switch",
|
||||
REG_MICFIL_VAD0_ZCD, 4, 1, 0),
|
||||
SOC_SINGLE_BOOL_EXT("VAD Detected", 0, hwvad_detected, NULL),
|
||||
{
|
||||
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
||||
.access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
|
||||
.name = "VAD Detected",
|
||||
.info = snd_soc_info_bool_ext,
|
||||
.get = hwvad_detected,
|
||||
},
|
||||
};
|
||||
|
||||
static int fsl_micfil_use_verid(struct device *dev)
|
||||
|
||||
@@ -167,10 +167,17 @@ static int fsl_xcvr_arc_mode_put(struct snd_kcontrol *kcontrol,
|
||||
struct fsl_xcvr *xcvr = snd_soc_dai_get_drvdata(dai);
|
||||
struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
|
||||
unsigned int *item = ucontrol->value.enumerated.item;
|
||||
int val = snd_soc_enum_item_to_val(e, item[0]);
|
||||
int ret;
|
||||
|
||||
xcvr->arc_mode = snd_soc_enum_item_to_val(e, item[0]);
|
||||
if (val < 0 || val > 1)
|
||||
return -EINVAL;
|
||||
|
||||
return 0;
|
||||
ret = (xcvr->arc_mode != val);
|
||||
|
||||
xcvr->arc_mode = val;
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int fsl_xcvr_arc_mode_get(struct snd_kcontrol *kcontrol,
|
||||
@@ -270,10 +277,17 @@ static int fsl_xcvr_mode_put(struct snd_kcontrol *kcontrol,
|
||||
struct fsl_xcvr *xcvr = snd_soc_dai_get_drvdata(dai);
|
||||
struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
|
||||
unsigned int *item = ucontrol->value.enumerated.item;
|
||||
int val = snd_soc_enum_item_to_val(e, item[0]);
|
||||
struct snd_soc_card *card = dai->component->card;
|
||||
struct snd_soc_pcm_runtime *rtd;
|
||||
int ret;
|
||||
|
||||
xcvr->mode = snd_soc_enum_item_to_val(e, item[0]);
|
||||
if (val < FSL_XCVR_MODE_SPDIF || val > FSL_XCVR_MODE_EARC)
|
||||
return -EINVAL;
|
||||
|
||||
ret = (xcvr->mode != val);
|
||||
|
||||
xcvr->mode = val;
|
||||
|
||||
fsl_xcvr_activate_ctl(dai, fsl_xcvr_arc_mode_kctl.name,
|
||||
(xcvr->mode == FSL_XCVR_MODE_ARC));
|
||||
@@ -283,7 +297,7 @@ static int fsl_xcvr_mode_put(struct snd_kcontrol *kcontrol,
|
||||
rtd = snd_soc_get_pcm_runtime(card, card->dai_link);
|
||||
rtd->pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].substream_count =
|
||||
(xcvr->mode == FSL_XCVR_MODE_SPDIF ? 1 : 0);
|
||||
return 0;
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int fsl_xcvr_mode_get(struct snd_kcontrol *kcontrol,
|
||||
|
||||
Reference in New Issue
Block a user