[alsa-devel] [PATCH v2] ASoC: imx-wm8958: add imx-wm8958 machine driver
Nicolin Chen
nicoleotsuka at gmail.com
Fri Feb 26 21:52:09 CET 2016
On Fri, Feb 26, 2016 at 11:42:42AM +0800, Zidan Wang wrote:
> This is the initial imx-wm8958 device-tree-only machine driver working
> with fsl_sai driver. This sound card has three dai link, HIFI, VOICE
> and BT dai. HIFI dai link will support codec master and slave mode.
> VOICE an BT dai link have dummy cpu dai, and just support codec
> master mode.
Why VOICE and BT are using dummy cpu dai? Where can I find the
schematics for wm8958 from NXP website?
> diff --git a/Documentation/devicetree/bindings/sound/imx-audio-wm8958.txt
> +Example:
> +
> +sound {
> + compatible = "fsl,imx6ul-evk-wm8958",
> + "fsl,imx-audio-wm8958";
> + model = "wm8960-audio";
8960?
> diff --git a/arch/arm/configs/imx_v6_v7_defconfig
> +config SND_SOC_IMX_WM8958
> + tristate "SoC Audio support for i.MX boards with wm8958"
> + depends on OF && I2C
> + select MFD_WM8994
> + select SND_SOC_WM8994
> + select SND_SOC_IMX_PCM_DMA
> + select SND_SOC_FSL_SAI
> + select SND_SOC_FSL_UTILS
> + select SND_KCTL_JACK
The driver doesn't seem to use kcontrol....
> +static u32 imx_wm8958_rates[] = { 8000, 16000, 32000, 48000 };
> +
> +static struct snd_pcm_hw_constraint_list imx_wm8958_rate_constraints = {
> + .count = ARRAY_SIZE(imx_wm8958_rates),
> + .list = imx_wm8958_rates,
> +};
Where are these rate constraints coming from?
> +static int imx_wm8958_hw_params(struct snd_pcm_substream *substream,
> + struct snd_pcm_hw_params *params)
> +{
> + if (data->mclk_freq[id])
> + mclk_id = WM8994_FLL_SRC_MCLK(id);
> + else if (id == HIFI_DAI)
> + mclk_id = WM8994_FLL_SRC_MCLK2;
> + else
> + mclk_id = WM8994_FLL_SRC_MCLK1;
I can barely follow the MCLK assigning logic here...
Is it trying to use the alternative MCLK if the corresponding one
is unavailable? (Since id could be only 0 or 1). And what if both
are unavailable?
> + if (id == HIFI_DAI) {
> + /*
> + * Set GPIO1 pin function to reserve, so that DAC1 and ADC1
> + * using shared LRCLK from DACLRCK1.
> + */
> + snd_soc_update_bits(codec, WM8994_GPIO_1, 0x1f, 0x2);
I don't see any reverse operation against this GPIO configuration here.
So is it really an operation that has to be done every hw_params()?
> + } else if (id == VOICE_DAI) {
> + /*
> + * Set GPIO6 pin function to reserve, so that DAC2 and ADC2
> + * using shared LRCLK from DACLRCK2.
> + */
> + snd_soc_update_bits(codec, WM8994_GPIO_6, 0x1f, 0x2);
Same here.
> +static int imx_wm8958_set_bias_level(struct snd_soc_card *card,
> + struct snd_soc_dapm_context *dapm,
> + enum snd_soc_bias_level level)
> +{
> + struct snd_soc_pcm_runtime *rtd;
> + struct snd_soc_dai *codec_dai;
> + struct imx_wm8958_data *data = snd_soc_card_get_drvdata(card);
> + int ret, i;
> +
> + rtd = snd_soc_get_pcm_runtime(card, card->dai_link[0].name);
> + codec_dai = rtd->codec_dai;
> +
> + if (dapm->dev != codec_dai->dev)
> + return 0;
> +
> + switch (level) {
> + case SND_SOC_BIAS_STANDBY:
> + if (card->dapm.bias_level == SND_SOC_BIAS_OFF) {
Using
if (card->dapm.bias_level != SND_SOC_BIAS_OFF)
break;
may save one \t for each line below.
> + /* need to enable mclk to write/read wm8958 register */
> + for (i = 0; i < WM8958_MCLK_MAX; i++) {
> + if (!IS_ERR(data->mclk[i])) {
> + ret = clk_prepare_enable(data->mclk[i]);
> + if (ret)
> + dev_warn(card->dev,
> + "Failed to enable MCLK%d: %d\n",
> + i + 1, ret);
> + }
> + }
> + }
> + break;
> + default:
> + break;
> + }
> +
> + return 0;
> +}
> +
> +static int imx_wm8958_set_bias_level_post(struct snd_soc_card *card,
> + struct snd_soc_dapm_context *dapm,
> + enum snd_soc_bias_level level)
> +{
> + struct snd_soc_pcm_runtime *rtd;
> + struct snd_soc_dai *codec_dai;
> + struct imx_wm8958_data *data = snd_soc_card_get_drvdata(card);
> + int i;
> +
> + rtd = snd_soc_get_pcm_runtime(card, card->dai_link[0].name);
> + codec_dai = rtd->codec_dai;
> +
> + if (dapm->dev != codec_dai->dev)
> + return 0;
> +
> + switch (level) {
> + case SND_SOC_BIAS_OFF:
> + if (card->dapm.bias_level == SND_SOC_BIAS_STANDBY)
> + for (i = 0; i < WM8958_MCLK_MAX; i++) {
> + if (!IS_ERR(data->mclk[i]))
> + clk_disable_unprepare(data->mclk[i]);
The set_bias() funcs are merely en/disabling mclk. Would be possible to
do it in the codec driver?
> +static struct snd_soc_dai_link imx_wm8958_dai_link[] = {
> + [HIFI_DAI] = {
> + .name = "HiFi",
> + .stream_name = "HiFi",
> + .codec_name = "wm8994-codec",
> + .codec_dai_name = "wm8994-aif1",
> + .ops = &imx_hifi_ops,
> + .dai_fmt = SND_SOC_DAIFMT_I2S |
> + SND_SOC_DAIFMT_NB_NF,
> + },
> + [VOICE_DAI] = {
> + .name = "Voice",
> + .stream_name = "Voice",
> + .cpu_dai_name = "snd-soc-dummy-dai",
> + .codec_name = "wm8994-codec",
> + .codec_dai_name = "wm8994-aif2",
> + .platform_name = "snd-soc-dummy",
> + .ignore_pmdown_time = 1,
> + .ops = &imx_voice_ops,
> + .dai_fmt = SND_SOC_DAIFMT_I2S |
> + SND_SOC_DAIFMT_NB_NF |
> + SND_SOC_DAIFMT_CBM_CFM,
> + },
> + [BT_DAI] = {
> + .name = "Bluetooth",
> + .stream_name = "Bluetooth",
> + .cpu_dai_name = "snd-soc-dummy-dai",
> + .codec_name = "wm8994-codec",
> + .codec_dai_name = "wm8994-aif3",
> + .platform_name = "snd-soc-dummy",
> + .ignore_pmdown_time = 1,
> + },
A little curious...how is the BT gonna work for you?
> +};
> +
> +static int imx_wm8958_probe(struct platform_device *pdev)
> +{
> + for (i = 0; i < WM8958_MCLK_MAX; i++) {
> + sprintf(tmp, "mclk%d", i + 1);
> + data->mclk[i] = devm_clk_get(&codec_dev->dev, tmp);
> + if (IS_ERR(data->mclk[i])) {
> + ret = PTR_ERR(data->mclk[i]);
> + dev_err(&pdev->dev, "failed to get mclk%d clock: %d\n",
> + i + 1, ret);
What's the point to have ret = PTR_ERR without doing any error out?
> + } else {
> + data->mclk_freq[i] = clk_get_rate(data->mclk[i]);
> + }
> + }
Thanks
Nicolin
More information about the Alsa-devel
mailing list