Re: [PATCH V2 2/2] ASoC: fsl_mqs: Add MQS component driver

From: Nicolin Chen
Date: Mon Sep 16 2019 - 20:08:25 EST


On Fri, Sep 13, 2019 at 05:42:14PM +0800, Shengjiu Wang wrote:
> MQS (medium quality sound), is used to generate medium quality
> audio via a standard digital output pin. It can be used to
> connect stereo speakers or headphones simply via power amplifier
> stages without an additional DAC chip. It only accepts 2-channel,
> LSB-valid 16bit, MSB shift-out first, frame sync asserting with
> the first bit of the frame, data shifted with the posedge of
> bit clock, 44.1 kHz or 48 kHz signals from SAI1 in left justified
> format; and it provides the SNR target as no more than 20dB for
> the signals below 10 kHz. The signals above 10 kHz will have
> worse THD+N values.
>
> MQS provides only simple audio reproduction. No internal pop,
> click or distortion artifact reduction methods are provided.
>
> The MQS receives the audio data from the SAI1 Tx section.
>
> Signed-off-by: Shengjiu Wang <shengjiu.wang@xxxxxxx>

Acked-by: Nicolin Chen <nicoleotsuka@xxxxxxxxx>

> ---
> Changes in v2
> - use devm_platform_ioremap_resource
>
> sound/soc/fsl/Kconfig | 10 ++
> sound/soc/fsl/Makefile | 2 +
> sound/soc/fsl/fsl_mqs.c | 333 ++++++++++++++++++++++++++++++++++++++++
> 3 files changed, 345 insertions(+)
> create mode 100644 sound/soc/fsl/fsl_mqs.c
>
> diff --git a/sound/soc/fsl/Kconfig b/sound/soc/fsl/Kconfig
> index aa99c008a925..65e8cd4be930 100644
> --- a/sound/soc/fsl/Kconfig
> +++ b/sound/soc/fsl/Kconfig
> @@ -25,6 +25,16 @@ config SND_SOC_FSL_SAI
> This option is only useful for out-of-tree drivers since
> in-tree drivers select it automatically.
>
> +config SND_SOC_FSL_MQS
> + tristate "Medium Quality Sound (MQS) module support"
> + depends on SND_SOC_FSL_SAI
> + select REGMAP_MMIO
> + help
> + Say Y if you want to add Medium Quality Sound (MQS)
> + support for the Freescale CPUs.
> + This option is only useful for out-of-tree drivers since
> + in-tree drivers select it automatically.
> +
> config SND_SOC_FSL_AUDMIX
> tristate "Audio Mixer (AUDMIX) module support"
> select REGMAP_MMIO
> diff --git a/sound/soc/fsl/Makefile b/sound/soc/fsl/Makefile
> index c0dd04422fe9..8cde88c72d93 100644
> --- a/sound/soc/fsl/Makefile
> +++ b/sound/soc/fsl/Makefile
> @@ -23,6 +23,7 @@ snd-soc-fsl-esai-objs := fsl_esai.o
> snd-soc-fsl-micfil-objs := fsl_micfil.o
> snd-soc-fsl-utils-objs := fsl_utils.o
> snd-soc-fsl-dma-objs := fsl_dma.o
> +snd-soc-fsl-mqs-objs := fsl_mqs.o
>
> obj-$(CONFIG_SND_SOC_FSL_AUDMIX) += snd-soc-fsl-audmix.o
> obj-$(CONFIG_SND_SOC_FSL_ASOC_CARD) += snd-soc-fsl-asoc-card.o
> @@ -33,6 +34,7 @@ obj-$(CONFIG_SND_SOC_FSL_SPDIF) += snd-soc-fsl-spdif.o
> obj-$(CONFIG_SND_SOC_FSL_ESAI) += snd-soc-fsl-esai.o
> obj-$(CONFIG_SND_SOC_FSL_MICFIL) += snd-soc-fsl-micfil.o
> obj-$(CONFIG_SND_SOC_FSL_UTILS) += snd-soc-fsl-utils.o
> +obj-$(CONFIG_SND_SOC_FSL_MQS) += snd-soc-fsl-mqs.o
> obj-$(CONFIG_SND_SOC_POWERPC_DMA) += snd-soc-fsl-dma.o
>
> # MPC5200 Platform Support
> diff --git a/sound/soc/fsl/fsl_mqs.c b/sound/soc/fsl/fsl_mqs.c
> new file mode 100644
> index 000000000000..c1619a553514
> --- /dev/null
> +++ b/sound/soc/fsl/fsl_mqs.c
> @@ -0,0 +1,333 @@
> +// SPDX-License-Identifier: GPL-2.0
> +//
> +// ALSA SoC IMX MQS driver
> +//
> +// Copyright (C) 2014-2015 Freescale Semiconductor, Inc.
> +// Copyright 2019 NXP
> +
> +#include <linux/clk.h>
> +#include <linux/module.h>
> +#include <linux/moduleparam.h>
> +#include <linux/mfd/syscon.h>
> +#include <linux/mfd/syscon/imx6q-iomuxc-gpr.h>
> +#include <linux/pm_runtime.h>
> +#include <linux/of.h>
> +#include <linux/pm.h>
> +#include <linux/slab.h>
> +#include <sound/soc.h>
> +#include <sound/pcm.h>
> +#include <sound/initval.h>
> +
> +#define REG_MQS_CTRL 0x00
> +
> +#define MQS_EN_MASK (0x1 << 28)
> +#define MQS_EN_SHIFT (28)
> +#define MQS_SW_RST_MASK (0x1 << 24)
> +#define MQS_SW_RST_SHIFT (24)
> +#define MQS_OVERSAMPLE_MASK (0x1 << 20)
> +#define MQS_OVERSAMPLE_SHIFT (20)
> +#define MQS_CLK_DIV_MASK (0xFF << 0)
> +#define MQS_CLK_DIV_SHIFT (0)
> +
> +/* codec private data */
> +struct fsl_mqs {
> + struct regmap *regmap;
> + struct clk *mclk;
> + struct clk *ipg;
> +
> + unsigned int reg_iomuxc_gpr2;
> + unsigned int reg_mqs_ctrl;
> + bool use_gpr;
> +};
> +
> +#define FSL_MQS_RATES (SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000)
> +#define FSL_MQS_FORMATS SNDRV_PCM_FMTBIT_S16_LE
> +
> +static int fsl_mqs_hw_params(struct snd_pcm_substream *substream,
> + struct snd_pcm_hw_params *params,
> + struct snd_soc_dai *dai)
> +{
> + struct snd_soc_component *component = dai->component;
> + struct fsl_mqs *mqs_priv = snd_soc_component_get_drvdata(component);
> + unsigned long mclk_rate;
> + int div, res;
> + int bclk, lrclk;
> +
> + mclk_rate = clk_get_rate(mqs_priv->mclk);
> + bclk = snd_soc_params_to_bclk(params);
> + lrclk = params_rate(params);
> +
> + /*
> + * mclk_rate / (oversample(32,64) * FS * 2 * divider ) = repeat_rate;
> + * if repeat_rate is 8, mqs can achieve better quality.
> + * oversample rate is fix to 32 currently.
> + */
> + div = mclk_rate / (32 * lrclk * 2 * 8);
> + res = mclk_rate % (32 * lrclk * 2 * 8);
> +
> + if (res == 0 && div > 0 && div <= 256) {
> + if (mqs_priv->use_gpr) {
> + regmap_update_bits(mqs_priv->regmap, IOMUXC_GPR2,
> + IMX6SX_GPR2_MQS_CLK_DIV_MASK,
> + (div - 1) << IMX6SX_GPR2_MQS_CLK_DIV_SHIFT);
> + regmap_update_bits(mqs_priv->regmap, IOMUXC_GPR2,
> + IMX6SX_GPR2_MQS_OVERSAMPLE_MASK, 0);
> + } else {
> + regmap_update_bits(mqs_priv->regmap, REG_MQS_CTRL,
> + MQS_CLK_DIV_MASK,
> + (div - 1) << MQS_CLK_DIV_SHIFT);
> + regmap_update_bits(mqs_priv->regmap, REG_MQS_CTRL,
> + MQS_OVERSAMPLE_MASK, 0);
> + }
> + } else {
> + dev_err(component->dev, "can't get proper divider\n");
> + }
> +
> + return 0;
> +}
> +
> +static int fsl_mqs_set_dai_fmt(struct snd_soc_dai *dai, unsigned int fmt)
> +{
> + /* Only LEFT_J & SLAVE mode is supported. */
> + switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
> + case SND_SOC_DAIFMT_LEFT_J:
> + break;
> + default:
> + return -EINVAL;
> + }
> +
> + switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
> + case SND_SOC_DAIFMT_NB_NF:
> + break;
> + default:
> + return -EINVAL;
> + }
> +
> + switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
> + case SND_SOC_DAIFMT_CBS_CFS:
> + break;
> + default:
> + return -EINVAL;
> + }
> +
> + return 0;
> +}
> +
> +static int fsl_mqs_startup(struct snd_pcm_substream *substream,
> + struct snd_soc_dai *dai)
> +{
> + struct snd_soc_component *component = dai->component;
> + struct fsl_mqs *mqs_priv = snd_soc_component_get_drvdata(component);
> +
> + if (mqs_priv->use_gpr)
> + regmap_update_bits(mqs_priv->regmap, IOMUXC_GPR2,
> + IMX6SX_GPR2_MQS_EN_MASK,
> + 1 << IMX6SX_GPR2_MQS_EN_SHIFT);
> + else
> + regmap_update_bits(mqs_priv->regmap, REG_MQS_CTRL,
> + MQS_EN_MASK,
> + 1 << MQS_EN_SHIFT);
> + return 0;
> +}
> +
> +static void fsl_mqs_shutdown(struct snd_pcm_substream *substream,
> + struct snd_soc_dai *dai)
> +{
> + struct snd_soc_component *component = dai->component;
> + struct fsl_mqs *mqs_priv = snd_soc_component_get_drvdata(component);
> +
> + if (mqs_priv->use_gpr)
> + regmap_update_bits(mqs_priv->regmap, IOMUXC_GPR2,
> + IMX6SX_GPR2_MQS_EN_MASK, 0);
> + else
> + regmap_update_bits(mqs_priv->regmap, REG_MQS_CTRL,
> + MQS_EN_MASK, 0);
> +}
> +
> +const static struct snd_soc_component_driver soc_codec_fsl_mqs = {
> + .idle_bias_on = 1,
> + .non_legacy_dai_naming = 1,
> +};
> +
> +static const struct snd_soc_dai_ops fsl_mqs_dai_ops = {
> + .startup = fsl_mqs_startup,
> + .shutdown = fsl_mqs_shutdown,
> + .hw_params = fsl_mqs_hw_params,
> + .set_fmt = fsl_mqs_set_dai_fmt,
> +};
> +
> +static struct snd_soc_dai_driver fsl_mqs_dai = {
> + .name = "fsl-mqs-dai",
> + .playback = {
> + .stream_name = "Playback",
> + .channels_min = 2,
> + .channels_max = 2,
> + .rates = FSL_MQS_RATES,
> + .formats = FSL_MQS_FORMATS,
> + },
> + .ops = &fsl_mqs_dai_ops,
> +};
> +
> +static const struct regmap_config fsl_mqs_regmap_config = {
> + .reg_bits = 32,
> + .reg_stride = 4,
> + .val_bits = 32,
> + .max_register = REG_MQS_CTRL,
> + .cache_type = REGCACHE_NONE,
> +};
> +
> +static int fsl_mqs_probe(struct platform_device *pdev)
> +{
> + struct device_node *np = pdev->dev.of_node;
> + struct device_node *gpr_np = 0;
> + struct fsl_mqs *mqs_priv;
> + void __iomem *regs;
> + int ret = 0;
> +
> + mqs_priv = devm_kzalloc(&pdev->dev, sizeof(*mqs_priv), GFP_KERNEL);
> + if (!mqs_priv)
> + return -ENOMEM;
> +
> + /* On i.MX6sx the MQS control register is in GPR domain
> + * But in i.MX8QM/i.MX8QXP the control register is moved
> + * to its own domain.
> + */
> + if (of_device_is_compatible(np, "fsl,imx8qm-mqs"))
> + mqs_priv->use_gpr = false;
> + else
> + mqs_priv->use_gpr = true;
> +
> + if (mqs_priv->use_gpr) {
> + gpr_np = of_parse_phandle(np, "gpr", 0);
> + if (IS_ERR(gpr_np)) {
> + dev_err(&pdev->dev, "failed to get gpr node by phandle\n");
> + ret = PTR_ERR(gpr_np);
> + goto out;
> + }
> +
> + mqs_priv->regmap = syscon_node_to_regmap(gpr_np);
> + if (IS_ERR(mqs_priv->regmap)) {
> + dev_err(&pdev->dev, "failed to get gpr regmap\n");
> + ret = PTR_ERR(mqs_priv->regmap);
> + goto out;
> + }
> + } else {
> + regs = devm_platform_ioremap_resource(pdev, 0);
> + if (IS_ERR(regs))
> + return PTR_ERR(regs);
> +
> + mqs_priv->regmap = devm_regmap_init_mmio_clk(&pdev->dev,
> + "core",
> + regs,
> + &fsl_mqs_regmap_config);
> + if (IS_ERR(mqs_priv->regmap)) {
> + dev_err(&pdev->dev, "failed to init regmap: %ld\n",
> + PTR_ERR(mqs_priv->regmap));
> + return PTR_ERR(mqs_priv->regmap);
> + }
> +
> + mqs_priv->ipg = devm_clk_get(&pdev->dev, "core");
> + if (IS_ERR(mqs_priv->ipg)) {
> + dev_err(&pdev->dev, "failed to get the clock: %ld\n",
> + PTR_ERR(mqs_priv->ipg));
> + goto out;
> + }
> + }
> +
> + mqs_priv->mclk = devm_clk_get(&pdev->dev, "mclk");
> + if (IS_ERR(mqs_priv->mclk)) {
> + dev_err(&pdev->dev, "failed to get the clock: %ld\n",
> + PTR_ERR(mqs_priv->mclk));
> + goto out;
> + }
> +
> + dev_set_drvdata(&pdev->dev, mqs_priv);
> + pm_runtime_enable(&pdev->dev);
> +
> + return devm_snd_soc_register_component(&pdev->dev, &soc_codec_fsl_mqs,
> + &fsl_mqs_dai, 1);
> +out:
> + if (!IS_ERR(gpr_np))
> + of_node_put(gpr_np);
> +
> + return ret;
> +}
> +
> +static int fsl_mqs_remove(struct platform_device *pdev)
> +{
> + pm_runtime_disable(&pdev->dev);
> + return 0;
> +}
> +
> +#ifdef CONFIG_PM
> +static int fsl_mqs_runtime_resume(struct device *dev)
> +{
> + struct fsl_mqs *mqs_priv = dev_get_drvdata(dev);
> +
> + if (mqs_priv->ipg)
> + clk_prepare_enable(mqs_priv->ipg);
> +
> + if (mqs_priv->mclk)
> + clk_prepare_enable(mqs_priv->mclk);
> +
> + if (mqs_priv->use_gpr)
> + regmap_write(mqs_priv->regmap, IOMUXC_GPR2,
> + mqs_priv->reg_iomuxc_gpr2);
> + else
> + regmap_write(mqs_priv->regmap, REG_MQS_CTRL,
> + mqs_priv->reg_mqs_ctrl);
> + return 0;
> +}
> +
> +static int fsl_mqs_runtime_suspend(struct device *dev)
> +{
> + struct fsl_mqs *mqs_priv = dev_get_drvdata(dev);
> +
> + if (mqs_priv->use_gpr)
> + regmap_read(mqs_priv->regmap, IOMUXC_GPR2,
> + &mqs_priv->reg_iomuxc_gpr2);
> + else
> + regmap_read(mqs_priv->regmap, REG_MQS_CTRL,
> + &mqs_priv->reg_mqs_ctrl);
> +
> + if (mqs_priv->mclk)
> + clk_disable_unprepare(mqs_priv->mclk);
> +
> + if (mqs_priv->ipg)
> + clk_disable_unprepare(mqs_priv->ipg);
> +
> + return 0;
> +}
> +#endif
> +
> +static const struct dev_pm_ops fsl_mqs_pm_ops = {
> + SET_RUNTIME_PM_OPS(fsl_mqs_runtime_suspend,
> + fsl_mqs_runtime_resume,
> + NULL)
> + SET_SYSTEM_SLEEP_PM_OPS(pm_runtime_force_suspend,
> + pm_runtime_force_resume)
> +};
> +
> +static const struct of_device_id fsl_mqs_dt_ids[] = {
> + { .compatible = "fsl,imx8qm-mqs", },
> + { .compatible = "fsl,imx6sx-mqs", },
> + {}
> +};
> +MODULE_DEVICE_TABLE(of, fsl_mqs_dt_ids);
> +
> +static struct platform_driver fsl_mqs_driver = {
> + .probe = fsl_mqs_probe,
> + .remove = fsl_mqs_remove,
> + .driver = {
> + .name = "fsl-mqs",
> + .of_match_table = fsl_mqs_dt_ids,
> + .pm = &fsl_mqs_pm_ops,
> + },
> +};
> +
> +module_platform_driver(fsl_mqs_driver);
> +
> +MODULE_AUTHOR("Shengjiu Wang <Shengjiu.Wang@xxxxxxx>");
> +MODULE_DESCRIPTION("MQS codec driver");
> +MODULE_LICENSE("GPL v2");
> +MODULE_ALIAS("platform: fsl-mqs");
> --
> 2.21.0
>