Abandoning this patch as it some clock changes that is already fixed in Brent's older patch (handled it in a different way). Will make a separate patch for other fixes once the complete solution is ready. > > >> > > > > > This patch adds a dapm route to provide early mclk/sclk for for DMIC > > > on > > > SSP0(rt5514) and Headset on SSP1(rt5663). > > > > > > The sclk rate for both codecs is different which is taken care of in > > > the platform_clock_control().The platform has one mclk and one sclk. > > > Two variables sclk0 and sclk1 are used for the two codecs on > > > differnet ssp that use different clock rate. > > > > > > This change ensures the DMIC PCM port will return valid data > > > > > > Signed-off-by: Brent Lu <brent.lu@xxxxxxxxx> > > > Signed-off-by: Vamshi Krishna Gopal <vamshi.krishna.gopal@xxxxxxxxx> > > > Signed-off-by: Harsha Priya <harshapriya.n@xxxxxxxxx> > > Hi Harsha, > > > > Isn't it working? I tested it on a eve before upstreaming. Thanks. > > > > commit 15747a80207585fe942416025540c0ff34e2aef8 > > Author: Brent Lu <brent.lu@xxxxxxxxx> > > Date: Fri Oct 25 17:11:31 2019 +0800 > > > > ASoC: eve: implement set_bias_level function for rt5514 > > > > The first DMIC capture always fail (zero sequence data from PCM port) > > after using DSP hotwording function (i.e. Google assistant). > > > > This rt5514 codec requires to control mclk directly in the set_bias_level > > function. Implement this function in machine driver to control the > > ssp1_mclk clock explicitly could fix this issue. > > > > Signed-off-by: Brent Lu <brent.lu@xxxxxxxxx> > > Reviewed-by: Pierre-Louis Bossart <pierre-louis.bossart@xxxxxxxxxxxxxxx> > > Link: https://lore.kernel.org/r/1571994691-20199-1-git-send-email- > > brent.lu@xxxxxxxxx > > Signed-off-by: Mark Brown <broonie@xxxxxxxxxx> > Your patch is necessary and is being used. > But we found few issues during kernel uprev where we need this current patch > also to get it working > > > > Regards, > > Brent > > > > > --- > > > v1 -> v2: > > > - Only one mclk with same rate is used, so changed to using one > > > variable > > > - dropping ssp_ prefix from sclk variable names to make them sound right. > > > - removing a return statment that was not required > > > - fixed commit message accordingly > > > > > > .../soc/intel/boards/kbl_rt5663_rt5514_max98927.c | 146 > > > ++++++++++++++------- > > > 1 file changed, 97 insertions(+), 49 deletions(-) > > > > > > diff --git a/sound/soc/intel/boards/kbl_rt5663_rt5514_max98927.c > > > b/sound/soc/intel/boards/kbl_rt5663_rt5514_max98927.c > > > index b34cf6c..155f2b4 100644 > > > --- a/sound/soc/intel/boards/kbl_rt5663_rt5514_max98927.c > > > +++ b/sound/soc/intel/boards/kbl_rt5663_rt5514_max98927.c > > > @@ -54,7 +54,8 @@ struct kbl_codec_private { > > > struct list_head hdmi_pcm_list; > > > struct snd_soc_jack kabylake_hdmi[2]; > > > struct clk *mclk; > > > - struct clk *sclk; > > > + struct clk *sclk0; > > > + struct clk *sclk1; > > > }; > > > > > > enum { > > > @@ -77,13 +78,29 @@ static const struct snd_kcontrol_new > > > kabylake_controls[] = { }; > > > > > > static int platform_clock_control(struct snd_soc_dapm_widget *w, > > > - struct snd_kcontrol *k, int event) > > > + struct snd_kcontrol *k, int event, int ssp_num) > > > { > > > struct snd_soc_dapm_context *dapm = w->dapm; > > > struct snd_soc_card *card = dapm->card; > > > struct kbl_codec_private *priv = snd_soc_card_get_drvdata(card); > > > + struct clk *sclk; > > > + unsigned long sclk_rate; > > > int ret = 0; > > > > > > + switch (ssp_num) { > > > + case 0: > > > + sclk = priv->sclk0; > > > + sclk_rate = 6144000; > > > + break; > > > + case 1: > > > + sclk = priv->sclk1; > > > + sclk_rate = 3072000; > > > + break; > > > + default: > > > + dev_err(card->dev, "Invalid ssp_num %d\n", ssp_num); > > > + return -EINVAL; > > > + } > > > + > > > /* > > > * MCLK/SCLK need to be ON early for a successful synchronization of > > > * codec internal clock. And the clocks are turned off during @@ - > > > 91,38 +108,48 @@ static int platform_clock_control(struct > > > snd_soc_dapm_widget *w, > > > */ > > > switch (event) { > > > case SND_SOC_DAPM_PRE_PMU: > > > - /* Enable MCLK */ > > > ret = clk_set_rate(priv->mclk, 24000000); > > > if (ret < 0) { > > > - dev_err(card->dev, "Can't set rate for mclk, err: > > > %d\n", > > > - ret); > > > - return ret; > > > + dev_err(card->dev, "Can't set rate for mclk for ssp%d, > > > err: %d\n", > > > + ssp_num, ret); > > > + return ret; > > > } > > > > > > - ret = clk_prepare_enable(priv->mclk); > > > - if (ret < 0) { > > > - dev_err(card->dev, "Can't enable mclk, err: %d\n", > > > ret); > > > - return ret; > > > + if (!__clk_is_enabled(priv->mclk)) { > > > + /* Enable MCLK */ > > > + ret = clk_prepare_enable(priv->mclk); > > > + if (ret < 0) { > > > + dev_err(card->dev, "Can't enable mclk for > > > ssp%d, err: %d\n", > > > + ssp_num, ret); > > > + return ret; > > > + } > > > } > > > > > > - /* Enable SCLK */ > > > - ret = clk_set_rate(priv->sclk, 3072000); > > > + ret = clk_set_rate(sclk, sclk_rate); > > > if (ret < 0) { > > > - dev_err(card->dev, "Can't set rate for sclk, err: > > > %d\n", > > > - ret); > > > + dev_err(card->dev, "Can't set rate for sclk for ssp%d, > > > err: %d\n", > > > + ssp_num, ret); > > > clk_disable_unprepare(priv->mclk); > > > return ret; > > > } > > > > > > - ret = clk_prepare_enable(priv->sclk); > > > - if (ret < 0) { > > > - dev_err(card->dev, "Can't enable sclk, err: %d\n", > > > ret); > > > - clk_disable_unprepare(priv->mclk); > > > + if (!__clk_is_enabled(sclk)) { > > > + /* Enable SCLK */ > > > + ret = clk_prepare_enable(sclk); > > > + if (ret < 0) { > > > + dev_err(card->dev, "Can't enable sclk for > > > ssp%d, err: %d\n", > > > + ssp_num, ret); > > > + clk_disable_unprepare(priv->mclk); > > > + return ret; > > > + } > > > } > > > break; > > > case SND_SOC_DAPM_POST_PMD: > > > - clk_disable_unprepare(priv->mclk); > > > - clk_disable_unprepare(priv->sclk); > > > + if (__clk_is_enabled(priv->mclk)) > > > + clk_disable_unprepare(priv->mclk); > > > + > > > + if (__clk_is_enabled(sclk)) > > > + clk_disable_unprepare(sclk); > > > break; > > > default: > > > return 0; > > > @@ -131,6 +158,18 @@ static int platform_clock_control(struct > > > snd_soc_dapm_widget *w, > > > return 0; > > > } > > > > > > +static int platform_clock_control_ssp0(struct snd_soc_dapm_widget *w, > > > + struct snd_kcontrol *k, int event) { > > > + return platform_clock_control(w, k, event, 0); } > > > + > > > +static int platform_clock_control_ssp1(struct snd_soc_dapm_widget *w, > > > + struct snd_kcontrol *k, int event) { > > > + return platform_clock_control(w, k, event, 1); } > > > + > > > static const struct snd_soc_dapm_widget kabylake_widgets[] = { > > > SND_SOC_DAPM_HP("Headphone Jack", NULL), > > > SND_SOC_DAPM_MIC("Headset Mic", NULL), @@ -139,15 +178,17 > > @@ static > > > const struct snd_soc_dapm_widget kabylake_widgets[] = { > > > SND_SOC_DAPM_MIC("DMIC", NULL), > > > SND_SOC_DAPM_SPK("HDMI1", NULL), > > > SND_SOC_DAPM_SPK("HDMI2", NULL), > > > - SND_SOC_DAPM_SUPPLY("Platform Clock", SND_SOC_NOPM, 0, 0, > > > - platform_clock_control, SND_SOC_DAPM_PRE_PMU > > > | > > > + SND_SOC_DAPM_SUPPLY("Platform Clock SSP0", SND_SOC_NOPM, > > > 0, 0, > > > + platform_clock_control_ssp0, > > > SND_SOC_DAPM_PRE_PMU | > > > + SND_SOC_DAPM_POST_PMD), > > > + SND_SOC_DAPM_SUPPLY("Platform Clock SSP1", SND_SOC_NOPM, > > > 0, 0, > > > + platform_clock_control_ssp1, > > > SND_SOC_DAPM_PRE_PMU | > > > SND_SOC_DAPM_POST_PMD), > > > - > > > }; > > > > > > static const struct snd_soc_dapm_route kabylake_map[] = { > > > /* Headphones */ > > > - { "Headphone Jack", NULL, "Platform Clock" }, > > > + { "Headphone Jack", NULL, "Platform Clock SSP1" }, > > > { "Headphone Jack", NULL, "HPOL" }, > > > { "Headphone Jack", NULL, "HPOR" }, > > > > > > @@ -156,7 +197,7 @@ static const struct snd_soc_dapm_route > > > kabylake_map[] = { > > > { "Right Spk", NULL, "Right BE_OUT" }, > > > > > > /* other jacks */ > > > - { "Headset Mic", NULL, "Platform Clock" }, > > > + { "Headset Mic", NULL, "Platform Clock SSP1" }, > > > { "IN1P", NULL, "Headset Mic" }, > > > { "IN1N", NULL, "Headset Mic" }, > > > > > > @@ -180,6 +221,7 @@ static const struct snd_soc_dapm_route > > > kabylake_map[] = { > > > { "ssp0 Rx", NULL, "Right HiFi Capture" }, > > > > > > /* DMIC */ > > > + { "DMIC", NULL, "Platform Clock SSP0" }, > > > { "DMIC1L", NULL, "DMIC" }, > > > { "DMIC1R", NULL, "DMIC" }, > > > { "DMIC2L", NULL, "DMIC" }, > > > @@ -666,7 +708,7 @@ static int kabylake_set_bias_level(struct > > > snd_soc_card *card, > > > if (!component || strcmp(component->name, RT5514_DEV_NAME)) > > > return 0; > > > > > > - if (IS_ERR(priv->mclk)) > > > + if (IS_ERR(priv->mclk0)) > > > return 0; > > > > > > /* > > > @@ -757,6 +799,28 @@ static struct snd_soc_card kabylake_audio_card = > { > > > .late_probe = kabylake_card_late_probe, }; > > > > > > +static int kabylake_audio_clk_get(struct device *dev, const char *id, > > > + struct clk **clk) > > > +{ > > > + int ret = 0; > > > + > > > + if (!clk) > > > + return -EINVAL; > > > + > > > + *clk = devm_clk_get(dev, id); > > > + if (IS_ERR(*clk)) { > > > + ret = PTR_ERR(*clk); > > > + if (ret == -ENOENT) { > > > + dev_info(dev, "Failed to get %s, defer probe\n", id); > > > + return -EPROBE_DEFER; > > > + } > > > + > > > + dev_err(dev, "Failed to get %s with err:%d\n", id, ret); > > > + } > > > + > > > + return ret; > > > +} > > > + > > > static int kabylake_audio_probe(struct platform_device *pdev) { > > > struct kbl_codec_private *ctx; > > > @@ -777,33 +841,17 @@ static int kabylake_audio_probe(struct > > > platform_device *pdev) > > > dmic_constraints = mach->mach_params.dmic_num == 2 ? > > > &constraints_dmic_2ch : > > > &constraints_dmic_channels; > > > > > > - ctx->mclk = devm_clk_get(&pdev->dev, "ssp1_mclk"); > > > - if (IS_ERR(ctx->mclk)) { > > > - ret = PTR_ERR(ctx->mclk); > > > - if (ret == -ENOENT) { > > > - dev_info(&pdev->dev, > > > - "Failed to get ssp1_mclk, defer probe\n"); > > > - return -EPROBE_DEFER; > > > - } > > > - > > > - dev_err(&pdev->dev, "Failed to get ssp1_mclk with > > > err:%d\n", > > > - ret); > > > + ret = kabylake_audio_clk_get(&pdev->dev, "ssp0_sclk", &ctx->sclk0); > > > + if (ret != 0) > > > return ret; > > > - } > > > > > > - ctx->sclk = devm_clk_get(&pdev->dev, "ssp1_sclk"); > > > - if (IS_ERR(ctx->sclk)) { > > > - ret = PTR_ERR(ctx->sclk); > > > - if (ret == -ENOENT) { > > > - dev_info(&pdev->dev, > > > - "Failed to get ssp1_sclk, defer probe\n"); > > > - return -EPROBE_DEFER; > > > - } > > > + ret = kabylake_audio_clk_get(&pdev->dev, "ssp1_mclk", &ctx- > > > >mclk); > > > + if (ret != 0) > > > + return ret; > > > > > > - dev_err(&pdev->dev, "Failed to get ssp1_sclk with err:%d\n", > > > - ret); > > > + ret = kabylake_audio_clk_get(&pdev->dev, "ssp1_sclk", &ctx->sclk1); > > > + if (ret != 0) > > > return ret; > > > - } > > > > > > return devm_snd_soc_register_card(&pdev->dev, > > > &kabylake_audio_card); } > > > -- > > > 2.7.4