Commit 17c81d2f authored by Adam Thomson's avatar Adam Thomson Committed by Mark Brown

ASoC: da7219: Add delays to capture path to remove DC offset noise

On some platforms it has been noted that a pop noise can be
witnessed when capturing audio, mainly for first time after a
headset jack has been inserted. This is due to a DC offset in the
Mic PGA and so to avoid this delays are required when powering
up the capture path.

This commit rectifies the problem by adding delays post Mic PGA and
post Mixin PGA. The post Mic PGA delay is determined based on
Mic Bias voltage, and is only applied the first time after a
headset jack is inserted.
Signed-off-by: default avatarAdam Thomson <Adam.Thomson.Opensource@diasemi.com>
Signed-off-by: default avatarMark Brown <broonie@kernel.org>
parent 605391d0
...@@ -59,6 +59,7 @@ static void da7219_aad_btn_det_work(struct work_struct *work) ...@@ -59,6 +59,7 @@ static void da7219_aad_btn_det_work(struct work_struct *work)
container_of(work, struct da7219_aad_priv, btn_det_work); container_of(work, struct da7219_aad_priv, btn_det_work);
struct snd_soc_component *component = da7219_aad->component; struct snd_soc_component *component = da7219_aad->component;
struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component); struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
struct da7219_priv *da7219 = snd_soc_component_get_drvdata(component);
u8 statusa, micbias_ctrl; u8 statusa, micbias_ctrl;
bool micbias_up = false; bool micbias_up = false;
int retries = 0; int retries = 0;
...@@ -86,6 +87,8 @@ static void da7219_aad_btn_det_work(struct work_struct *work) ...@@ -86,6 +87,8 @@ static void da7219_aad_btn_det_work(struct work_struct *work)
if (retries >= DA7219_AAD_MICBIAS_CHK_RETRIES) if (retries >= DA7219_AAD_MICBIAS_CHK_RETRIES)
dev_warn(component->dev, "Mic bias status check timed out"); dev_warn(component->dev, "Mic bias status check timed out");
da7219->micbias_on_event = true;
/* /*
* Mic bias pulse required to enable mic, must be done before enabling * Mic bias pulse required to enable mic, must be done before enabling
* button detection to prevent erroneous button readings. * button detection to prevent erroneous button readings.
...@@ -439,6 +442,8 @@ static irqreturn_t da7219_aad_irq_thread(int irq, void *data) ...@@ -439,6 +442,8 @@ static irqreturn_t da7219_aad_irq_thread(int irq, void *data)
snd_soc_component_update_bits(component, DA7219_ACCDET_CONFIG_1, snd_soc_component_update_bits(component, DA7219_ACCDET_CONFIG_1,
DA7219_BUTTON_CONFIG_MASK, 0); DA7219_BUTTON_CONFIG_MASK, 0);
da7219->micbias_on_event = false;
/* Disable mic bias */ /* Disable mic bias */
snd_soc_dapm_disable_pin(dapm, "Mic Bias"); snd_soc_dapm_disable_pin(dapm, "Mic Bias");
snd_soc_dapm_sync(dapm); snd_soc_dapm_sync(dapm);
......
...@@ -768,6 +768,30 @@ static const struct snd_kcontrol_new da7219_st_out_filtr_mix_controls[] = { ...@@ -768,6 +768,30 @@ static const struct snd_kcontrol_new da7219_st_out_filtr_mix_controls[] = {
* DAPM Events * DAPM Events
*/ */
static int da7219_mic_pga_event(struct snd_soc_dapm_widget *w,
struct snd_kcontrol *kcontrol, int event)
{
struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm);
struct da7219_priv *da7219 = snd_soc_component_get_drvdata(component);
switch (event) {
case SND_SOC_DAPM_POST_PMU:
if (da7219->micbias_on_event) {
/*
* Delay only for first capture after bias enabled to
* avoid possible DC offset related noise.
*/
da7219->micbias_on_event = false;
msleep(da7219->mic_pga_delay);
}
break;
default:
break;
}
return 0;
}
static int da7219_dai_event(struct snd_soc_dapm_widget *w, static int da7219_dai_event(struct snd_soc_dapm_widget *w,
struct snd_kcontrol *kcontrol, int event) struct snd_kcontrol *kcontrol, int event)
{ {
...@@ -937,12 +961,12 @@ static const struct snd_soc_dapm_widget da7219_dapm_widgets[] = { ...@@ -937,12 +961,12 @@ static const struct snd_soc_dapm_widget da7219_dapm_widgets[] = {
SND_SOC_DAPM_INPUT("MIC"), SND_SOC_DAPM_INPUT("MIC"),
/* Input PGAs */ /* Input PGAs */
SND_SOC_DAPM_PGA("Mic PGA", DA7219_MIC_1_CTRL, SND_SOC_DAPM_PGA_E("Mic PGA", DA7219_MIC_1_CTRL,
DA7219_MIC_1_AMP_EN_SHIFT, DA7219_NO_INVERT, DA7219_MIC_1_AMP_EN_SHIFT, DA7219_NO_INVERT,
NULL, 0), NULL, 0, da7219_mic_pga_event, SND_SOC_DAPM_POST_PMU),
SND_SOC_DAPM_PGA("Mixin PGA", DA7219_MIXIN_L_CTRL, SND_SOC_DAPM_PGA_E("Mixin PGA", DA7219_MIXIN_L_CTRL,
DA7219_MIXIN_L_AMP_EN_SHIFT, DA7219_NO_INVERT, DA7219_MIXIN_L_AMP_EN_SHIFT, DA7219_NO_INVERT,
NULL, 0), NULL, 0, da7219_settling_event, SND_SOC_DAPM_POST_PMU),
/* Input Filters */ /* Input Filters */
SND_SOC_DAPM_ADC("ADC", NULL, DA7219_ADC_L_CTRL, DA7219_ADC_L_EN_SHIFT, SND_SOC_DAPM_ADC("ADC", NULL, DA7219_ADC_L_CTRL, DA7219_ADC_L_EN_SHIFT,
...@@ -1847,6 +1871,14 @@ static void da7219_handle_pdata(struct snd_soc_component *component) ...@@ -1847,6 +1871,14 @@ static void da7219_handle_pdata(struct snd_soc_component *component)
snd_soc_component_write(component, DA7219_MICBIAS_CTRL, micbias_lvl); snd_soc_component_write(component, DA7219_MICBIAS_CTRL, micbias_lvl);
/*
* Calculate delay required to compensate for DC offset in
* Mic PGA, based on Mic Bias voltage.
*/
da7219->mic_pga_delay = DA7219_MIC_PGA_BASE_DELAY +
(pdata->micbias_lvl *
DA7219_MIC_PGA_OFFSET_DELAY);
/* Mic */ /* Mic */
switch (pdata->mic_amp_in_sel) { switch (pdata->mic_amp_in_sel) {
case DA7219_MIC_AMP_IN_SEL_DIFF: case DA7219_MIC_AMP_IN_SEL_DIFF:
......
...@@ -783,6 +783,8 @@ ...@@ -783,6 +783,8 @@
/* Power up/down Delays */ /* Power up/down Delays */
#define DA7219_SETTLING_DELAY 40 #define DA7219_SETTLING_DELAY 40
#define DA7219_MIN_GAIN_DELAY 30 #define DA7219_MIN_GAIN_DELAY 30
#define DA7219_MIC_PGA_BASE_DELAY 100
#define DA7219_MIC_PGA_OFFSET_DELAY 40
enum da7219_clk_src { enum da7219_clk_src {
DA7219_CLKSRC_MCLK = 0, DA7219_CLKSRC_MCLK = 0,
...@@ -828,6 +830,8 @@ struct da7219_priv { ...@@ -828,6 +830,8 @@ struct da7219_priv {
bool master; bool master;
bool alc_en; bool alc_en;
bool micbias_on_event;
unsigned int mic_pga_delay;
u8 gain_ramp_ctrl; u8 gain_ramp_ctrl;
}; };
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment