mirror of https://gitee.com/openkylin/linux.git
ASoC: codecs: tas5720: add TAS5722 specific volume control
The TAS5722 supports modifying volume in 0.25dB steps (as opposed to 0.5dB steps on the TAS5720). Introduce a custom mixer control that allows taking advantage of this finer output volume granularity. Also add custom getters/setters for access as the TAS5722 digital volume controls are split over two registers. Signed-off-by: Andreas Dannenberg <dannenberg@ti.com> Signed-off-by: Andrew F. Davis <afd@ti.com> Signed-off-by: Mark Brown <broonie@kernel.org>
This commit is contained in:
parent
5fcb457ac2
commit
ec94c177bf
|
@ -485,15 +485,56 @@ static const DECLARE_TLV_DB_RANGE(dac_analog_tlv,
|
||||||
);
|
);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* DAC digital volumes. From -103.5 to 24 dB in 0.5 dB steps. Note that
|
* DAC digital volumes. From -103.5 to 24 dB in 0.5 dB or 0.25 dB steps
|
||||||
* setting the gain below -100 dB (register value <0x7) is effectively a MUTE
|
* depending on the device. Note that setting the gain below -100 dB
|
||||||
* as per device datasheet.
|
* (register value <0x7) is effectively a MUTE as per device datasheet.
|
||||||
|
*
|
||||||
|
* Note that for the TAS5722 the digital volume controls are actually split
|
||||||
|
* over two registers, so we need custom getters/setters for access.
|
||||||
*/
|
*/
|
||||||
static DECLARE_TLV_DB_SCALE(dac_tlv, -10350, 50, 0);
|
static DECLARE_TLV_DB_SCALE(tas5720_dac_tlv, -10350, 50, 0);
|
||||||
|
static DECLARE_TLV_DB_SCALE(tas5722_dac_tlv, -10350, 25, 0);
|
||||||
|
|
||||||
|
static int tas5722_volume_get(struct snd_kcontrol *kcontrol,
|
||||||
|
struct snd_ctl_elem_value *ucontrol)
|
||||||
|
{
|
||||||
|
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
|
||||||
|
unsigned int val;
|
||||||
|
|
||||||
|
snd_soc_component_read(component, TAS5720_VOLUME_CTRL_REG, &val);
|
||||||
|
ucontrol->value.integer.value[0] = val << 1;
|
||||||
|
|
||||||
|
snd_soc_component_read(component, TAS5722_DIGITAL_CTRL2_REG, &val);
|
||||||
|
ucontrol->value.integer.value[0] |= val & TAS5722_VOL_CONTROL_LSB;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int tas5722_volume_set(struct snd_kcontrol *kcontrol,
|
||||||
|
struct snd_ctl_elem_value *ucontrol)
|
||||||
|
{
|
||||||
|
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
|
||||||
|
unsigned int sel = ucontrol->value.integer.value[0];
|
||||||
|
|
||||||
|
snd_soc_component_write(component, TAS5720_VOLUME_CTRL_REG, sel >> 1);
|
||||||
|
snd_soc_component_update_bits(component, TAS5722_DIGITAL_CTRL2_REG,
|
||||||
|
TAS5722_VOL_CONTROL_LSB, sel);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static const struct snd_kcontrol_new tas5720_snd_controls[] = {
|
static const struct snd_kcontrol_new tas5720_snd_controls[] = {
|
||||||
SOC_SINGLE_TLV("Speaker Driver Playback Volume",
|
SOC_SINGLE_TLV("Speaker Driver Playback Volume",
|
||||||
TAS5720_VOLUME_CTRL_REG, 0, 0xff, 0, dac_tlv),
|
TAS5720_VOLUME_CTRL_REG, 0, 0xff, 0, tas5720_dac_tlv),
|
||||||
|
SOC_SINGLE_TLV("Speaker Driver Analog Gain", TAS5720_ANALOG_CTRL_REG,
|
||||||
|
TAS5720_ANALOG_GAIN_SHIFT, 3, 0, dac_analog_tlv),
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct snd_kcontrol_new tas5722_snd_controls[] = {
|
||||||
|
SOC_SINGLE_EXT_TLV("Speaker Driver Playback Volume",
|
||||||
|
0, 0, 511, 0,
|
||||||
|
tas5722_volume_get, tas5722_volume_set,
|
||||||
|
tas5722_dac_tlv),
|
||||||
SOC_SINGLE_TLV("Speaker Driver Analog Gain", TAS5720_ANALOG_CTRL_REG,
|
SOC_SINGLE_TLV("Speaker Driver Analog Gain", TAS5720_ANALOG_CTRL_REG,
|
||||||
TAS5720_ANALOG_GAIN_SHIFT, 3, 0, dac_analog_tlv),
|
TAS5720_ANALOG_GAIN_SHIFT, 3, 0, dac_analog_tlv),
|
||||||
};
|
};
|
||||||
|
@ -527,6 +568,23 @@ static const struct snd_soc_component_driver soc_component_dev_tas5720 = {
|
||||||
.non_legacy_dai_naming = 1,
|
.non_legacy_dai_naming = 1,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct snd_soc_component_driver soc_component_dev_tas5722 = {
|
||||||
|
.probe = tas5720_codec_probe,
|
||||||
|
.remove = tas5720_codec_remove,
|
||||||
|
.suspend = tas5720_suspend,
|
||||||
|
.resume = tas5720_resume,
|
||||||
|
.controls = tas5722_snd_controls,
|
||||||
|
.num_controls = ARRAY_SIZE(tas5722_snd_controls),
|
||||||
|
.dapm_widgets = tas5720_dapm_widgets,
|
||||||
|
.num_dapm_widgets = ARRAY_SIZE(tas5720_dapm_widgets),
|
||||||
|
.dapm_routes = tas5720_audio_map,
|
||||||
|
.num_dapm_routes = ARRAY_SIZE(tas5720_audio_map),
|
||||||
|
.idle_bias_on = 1,
|
||||||
|
.use_pmdown_time = 1,
|
||||||
|
.endianness = 1,
|
||||||
|
.non_legacy_dai_naming = 1,
|
||||||
|
};
|
||||||
|
|
||||||
/* PCM rates supported by the TAS5720 driver */
|
/* PCM rates supported by the TAS5720 driver */
|
||||||
#define TAS5720_RATES (SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000 |\
|
#define TAS5720_RATES (SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000 |\
|
||||||
SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000)
|
SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000)
|
||||||
|
@ -613,9 +671,23 @@ static int tas5720_probe(struct i2c_client *client,
|
||||||
|
|
||||||
dev_set_drvdata(dev, data);
|
dev_set_drvdata(dev, data);
|
||||||
|
|
||||||
ret = devm_snd_soc_register_component(&client->dev,
|
switch (id->driver_data) {
|
||||||
&soc_component_dev_tas5720,
|
case TAS5720:
|
||||||
tas5720_dai, ARRAY_SIZE(tas5720_dai));
|
ret = devm_snd_soc_register_component(&client->dev,
|
||||||
|
&soc_component_dev_tas5720,
|
||||||
|
tas5720_dai,
|
||||||
|
ARRAY_SIZE(tas5720_dai));
|
||||||
|
break;
|
||||||
|
case TAS5722:
|
||||||
|
ret = devm_snd_soc_register_component(&client->dev,
|
||||||
|
&soc_component_dev_tas5722,
|
||||||
|
tas5720_dai,
|
||||||
|
ARRAY_SIZE(tas5720_dai));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
dev_err(dev, "unexpected private driver data\n");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
dev_err(dev, "failed to register component: %d\n", ret);
|
dev_err(dev, "failed to register component: %d\n", ret);
|
||||||
return ret;
|
return ret;
|
||||||
|
|
Loading…
Reference in New Issue