ASoC: tegra: Add Tegra210 based AMX driver
authorSameer Pujar <spujar@nvidia.com>
Mon, 13 Sep 2021 16:42:16 +0000 (22:12 +0530)
committerMark Brown <broonie@kernel.org>
Mon, 20 Sep 2021 12:31:34 +0000 (13:31 +0100)
The Audio Multiplexer (AMX) block can multiplex up to four input streams
each of which can have maximum 16 channels and generate an output stream
with maximum 16 channels. A byte RAM helps to form an output frame by
any combination of bytes from the input frames.

This patch registers AMX driver with ASoC framework. The component driver
exposes DAPM widgets, routes and kcontrols for the device. The DAI driver
exposes AMX interfaces, which can be used to connect different components
in the ASoC layer. Makefile and Kconfig support is added to allow build
the driver. It can be enabled in the DT via "nvidia,tegra210-amx" for
Tegra210 and Tegra186. For Tegra194 and later, "nvidia,tegra194-amx" can
be used.

Signed-off-by: Sameer Pujar <spujar@nvidia.com>
Link: https://lore.kernel.org/r/1631551342-25469-9-git-send-email-spujar@nvidia.com
Signed-off-by: Mark Brown <broonie@kernel.org>
sound/soc/tegra/Kconfig
sound/soc/tegra/Makefile
sound/soc/tegra/tegra210_amx.c [new file with mode: 0644]
sound/soc/tegra/tegra210_amx.h [new file with mode: 0644]

index 102564eece2aab45fd307decd9c077ca6c247a65..54d8342d96880380a1baaac1b98d2aa09668bd69 100644 (file)
@@ -126,6 +126,15 @@ config SND_SOC_TEGRA210_SFC
          upto 2 channels (stereo).
          Say Y or M if you want to add support for Tegra210 SFC module.
 
+config SND_SOC_TEGRA210_AMX
+       tristate "Tegra210 AMX module"
+       help
+         Config to enable the Audio Multiplexer (AMX) which can multiplex
+         four input streams (each of up to 16 channels) and generate
+         output stream (of up to 16 channels). A byte RAM helps to form an
+         output frame by any combination of bytes from the input frames.
+         Say Y or M if you want to add support for Tegra210 AMX module.
+
 config SND_SOC_TEGRA_AUDIO_GRAPH_CARD
        tristate "Audio Graph Card based Tegra driver"
        depends on SND_AUDIO_GRAPH_CARD
index bb0cf3ad7aaaba454436dcaba10a6f016aa22ad1..549162b1a3cc6523edead5de142a5c3ccb6350bb 100644 (file)
@@ -15,6 +15,7 @@ snd-soc-tegra186-dspk-objs := tegra186_dspk.o
 snd-soc-tegra210-admaif-objs := tegra210_admaif.o
 snd-soc-tegra210-mvc-objs := tegra210_mvc.o
 snd-soc-tegra210-sfc-objs := tegra210_sfc.o
+snd-soc-tegra210-amx-objs := tegra210_amx.o
 
 obj-$(CONFIG_SND_SOC_TEGRA) += snd-soc-tegra-pcm.o
 obj-$(CONFIG_SND_SOC_TEGRA20_AC97) += snd-soc-tegra20-ac97.o
@@ -30,6 +31,7 @@ obj-$(CONFIG_SND_SOC_TEGRA186_DSPK) += snd-soc-tegra186-dspk.o
 obj-$(CONFIG_SND_SOC_TEGRA210_ADMAIF) += snd-soc-tegra210-admaif.o
 obj-$(CONFIG_SND_SOC_TEGRA210_MVC) += snd-soc-tegra210-mvc.o
 obj-$(CONFIG_SND_SOC_TEGRA210_SFC) += snd-soc-tegra210-sfc.o
+obj-$(CONFIG_SND_SOC_TEGRA210_AMX) += snd-soc-tegra210-amx.o
 
 # Tegra machine Support
 snd-soc-tegra-wm8903-objs := tegra_wm8903.o
diff --git a/sound/soc/tegra/tegra210_amx.c b/sound/soc/tegra/tegra210_amx.c
new file mode 100644 (file)
index 0000000..83176e1
--- /dev/null
@@ -0,0 +1,600 @@
+// SPDX-License-Identifier: GPL-2.0-only
+//
+// tegra210_amx.c - Tegra210 AMX driver
+//
+// Copyright (c) 2021 NVIDIA CORPORATION.  All rights reserved.
+
+#include <linux/clk.h>
+#include <linux/device.h>
+#include <linux/io.h>
+#include <linux/module.h>
+#include <linux/of.h>
+#include <linux/of_device.h>
+#include <linux/platform_device.h>
+#include <linux/pm_runtime.h>
+#include <linux/regmap.h>
+#include <sound/core.h>
+#include <sound/pcm.h>
+#include <sound/pcm_params.h>
+#include <sound/soc.h>
+
+#include "tegra210_amx.h"
+#include "tegra_cif.h"
+
+/*
+ * The counter is in terms of AHUB clock cycles. If a frame is not
+ * received within these clock cycles, the AMX input channel gets
+ * automatically disabled. For now the counter is calculated as a
+ * function of sample rate (8 kHz) and AHUB clock (49.152 MHz).
+ * If later an accurate number is needed, the counter needs to be
+ * calculated at runtime.
+ *
+ *     count = ahub_clk / sample_rate
+ */
+#define TEGRA194_MAX_FRAME_IDLE_COUNT  0x1800
+
+#define AMX_CH_REG(id, reg) ((reg) + ((id) * TEGRA210_AMX_AUDIOCIF_CH_STRIDE))
+
+static const struct reg_default tegra210_amx_reg_defaults[] = {
+       { TEGRA210_AMX_RX_INT_MASK, 0x0000000f},
+       { TEGRA210_AMX_RX1_CIF_CTRL, 0x00007000},
+       { TEGRA210_AMX_RX2_CIF_CTRL, 0x00007000},
+       { TEGRA210_AMX_RX3_CIF_CTRL, 0x00007000},
+       { TEGRA210_AMX_RX4_CIF_CTRL, 0x00007000},
+       { TEGRA210_AMX_TX_INT_MASK, 0x00000001},
+       { TEGRA210_AMX_TX_CIF_CTRL, 0x00007000},
+       { TEGRA210_AMX_CG, 0x1},
+       { TEGRA210_AMX_CFG_RAM_CTRL, 0x00004000},
+};
+
+static void tegra210_amx_write_map_ram(struct tegra210_amx *amx)
+{
+       int i;
+
+       regmap_write(amx->regmap, TEGRA210_AMX_CFG_RAM_CTRL,
+                    TEGRA210_AMX_CFG_RAM_CTRL_SEQ_ACCESS_EN |
+                    TEGRA210_AMX_CFG_RAM_CTRL_ADDR_INIT_EN |
+                    TEGRA210_AMX_CFG_RAM_CTRL_RW_WRITE);
+
+       for (i = 0; i < TEGRA210_AMX_RAM_DEPTH; i++)
+               regmap_write(amx->regmap, TEGRA210_AMX_CFG_RAM_DATA,
+                            amx->map[i]);
+
+       regmap_write(amx->regmap, TEGRA210_AMX_OUT_BYTE_EN0, amx->byte_mask[0]);
+       regmap_write(amx->regmap, TEGRA210_AMX_OUT_BYTE_EN1, amx->byte_mask[1]);
+}
+
+static int tegra210_amx_startup(struct snd_pcm_substream *substream,
+                               struct snd_soc_dai *dai)
+{
+       struct tegra210_amx *amx = snd_soc_dai_get_drvdata(dai);
+       unsigned int val;
+       int err;
+
+       /* Ensure if AMX is disabled */
+       err = regmap_read_poll_timeout(amx->regmap, TEGRA210_AMX_STATUS, val,
+                                      !(val & 0x1), 10, 10000);
+       if (err < 0) {
+               dev_err(dai->dev, "failed to stop AMX, err = %d\n", err);
+               return err;
+       }
+
+       /*
+        * Soft Reset: Below performs module soft reset which clears
+        * all FSM logic, flushes flow control of FIFO and resets the
+        * state register. It also brings module back to disabled
+        * state (without flushing the data in the pipe).
+        */
+       regmap_update_bits(amx->regmap, TEGRA210_AMX_SOFT_RESET,
+                          TEGRA210_AMX_SOFT_RESET_SOFT_RESET_MASK,
+                          TEGRA210_AMX_SOFT_RESET_SOFT_EN);
+
+       err = regmap_read_poll_timeout(amx->regmap, TEGRA210_AMX_SOFT_RESET,
+                                      val, !(val & 0x1), 10, 10000);
+       if (err < 0) {
+               dev_err(dai->dev, "failed to reset AMX, err = %d\n", err);
+               return err;
+       }
+
+       return 0;
+}
+
+static int __maybe_unused tegra210_amx_runtime_suspend(struct device *dev)
+{
+       struct tegra210_amx *amx = dev_get_drvdata(dev);
+
+       regcache_cache_only(amx->regmap, true);
+       regcache_mark_dirty(amx->regmap);
+
+       return 0;
+}
+
+static int __maybe_unused tegra210_amx_runtime_resume(struct device *dev)
+{
+       struct tegra210_amx *amx = dev_get_drvdata(dev);
+
+       regcache_cache_only(amx->regmap, false);
+       regcache_sync(amx->regmap);
+
+       regmap_update_bits(amx->regmap,
+               TEGRA210_AMX_CTRL,
+               TEGRA210_AMX_CTRL_RX_DEP_MASK,
+               TEGRA210_AMX_WAIT_ON_ANY << TEGRA210_AMX_CTRL_RX_DEP_SHIFT);
+
+       tegra210_amx_write_map_ram(amx);
+
+       return 0;
+}
+
+static int tegra210_amx_set_audio_cif(struct snd_soc_dai *dai,
+                                     struct snd_pcm_hw_params *params,
+                                     unsigned int reg)
+{
+       struct tegra210_amx *amx = snd_soc_dai_get_drvdata(dai);
+       int channels, audio_bits;
+       struct tegra_cif_conf cif_conf;
+
+       memset(&cif_conf, 0, sizeof(struct tegra_cif_conf));
+
+       channels = params_channels(params);
+
+       switch (params_format(params)) {
+       case SNDRV_PCM_FORMAT_S8:
+               audio_bits = TEGRA_ACIF_BITS_8;
+               break;
+       case SNDRV_PCM_FORMAT_S16_LE:
+               audio_bits = TEGRA_ACIF_BITS_16;
+               break;
+       case SNDRV_PCM_FORMAT_S32_LE:
+               audio_bits = TEGRA_ACIF_BITS_32;
+               break;
+       default:
+               return -EINVAL;
+       }
+
+       cif_conf.audio_ch = channels;
+       cif_conf.client_ch = channels;
+       cif_conf.audio_bits = audio_bits;
+       cif_conf.client_bits = audio_bits;
+
+       tegra_set_cif(amx->regmap, reg, &cif_conf);
+
+       return 0;
+}
+
+static int tegra210_amx_in_hw_params(struct snd_pcm_substream *substream,
+                                    struct snd_pcm_hw_params *params,
+                                    struct snd_soc_dai *dai)
+{
+       struct tegra210_amx *amx = snd_soc_dai_get_drvdata(dai);
+
+       if (amx->soc_data->auto_disable) {
+               regmap_write(amx->regmap,
+                            AMX_CH_REG(dai->id, TEGRA194_AMX_RX1_FRAME_PERIOD),
+                            TEGRA194_MAX_FRAME_IDLE_COUNT);
+               regmap_write(amx->regmap, TEGRA210_AMX_CYA, 1);
+       }
+
+       return tegra210_amx_set_audio_cif(dai, params,
+                       AMX_CH_REG(dai->id, TEGRA210_AMX_RX1_CIF_CTRL));
+}
+
+static int tegra210_amx_out_hw_params(struct snd_pcm_substream *substream,
+                                     struct snd_pcm_hw_params *params,
+                                     struct snd_soc_dai *dai)
+{
+       return tegra210_amx_set_audio_cif(dai, params,
+                                         TEGRA210_AMX_TX_CIF_CTRL);
+}
+
+static int tegra210_amx_get_byte_map(struct snd_kcontrol *kcontrol,
+                                    struct snd_ctl_elem_value *ucontrol)
+{
+       struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol);
+       struct soc_mixer_control *mc =
+               (struct soc_mixer_control *)kcontrol->private_value;
+       struct tegra210_amx *amx = snd_soc_component_get_drvdata(cmpnt);
+       unsigned char *bytes_map = (unsigned char *)&amx->map;
+       int reg = mc->reg;
+       int enabled;
+
+       if (reg > 31)
+               enabled = amx->byte_mask[1] & (1 << (reg - 32));
+       else
+               enabled = amx->byte_mask[0] & (1 << reg);
+
+       if (enabled)
+               ucontrol->value.integer.value[0] = bytes_map[reg];
+       else
+               ucontrol->value.integer.value[0] = 0;
+
+       return 0;
+}
+
+static int tegra210_amx_put_byte_map(struct snd_kcontrol *kcontrol,
+                                    struct snd_ctl_elem_value *ucontrol)
+{
+       struct soc_mixer_control *mc =
+               (struct soc_mixer_control *)kcontrol->private_value;
+       struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol);
+       struct tegra210_amx *amx = snd_soc_component_get_drvdata(cmpnt);
+       unsigned char *bytes_map = (unsigned char *)&amx->map;
+       int reg = mc->reg;
+       int value = ucontrol->value.integer.value[0];
+
+       if (value >= 0 && value <= 255) {
+               /* Update byte map and enable slot */
+               bytes_map[reg] = value;
+               if (reg > 31)
+                       amx->byte_mask[1] |= (1 << (reg - 32));
+               else
+                       amx->byte_mask[0] |= (1 << reg);
+       } else {
+               /* Reset byte map and disable slot */
+               bytes_map[reg] = 0;
+               if (reg > 31)
+                       amx->byte_mask[1] &= ~(1 << (reg - 32));
+               else
+                       amx->byte_mask[0] &= ~(1 << reg);
+       }
+
+       return 1;
+}
+
+static struct snd_soc_dai_ops tegra210_amx_out_dai_ops = {
+       .hw_params      = tegra210_amx_out_hw_params,
+       .startup        = tegra210_amx_startup,
+};
+
+static struct snd_soc_dai_ops tegra210_amx_in_dai_ops = {
+       .hw_params      = tegra210_amx_in_hw_params,
+};
+
+#define IN_DAI(id)                                             \
+       {                                                       \
+               .name = "AMX-RX-CIF" #id,                       \
+               .playback = {                                   \
+                       .stream_name = "RX" #id "-CIF-Playback",\
+                       .channels_min = 1,                      \
+                       .channels_max = 16,                     \
+                       .rates = SNDRV_PCM_RATE_8000_192000,    \
+                       .formats = SNDRV_PCM_FMTBIT_S8 |        \
+                                  SNDRV_PCM_FMTBIT_S16_LE |    \
+                                  SNDRV_PCM_FMTBIT_S32_LE,     \
+               },                                              \
+               .capture = {                                    \
+                       .stream_name = "RX" #id "-CIF-Capture", \
+                       .channels_min = 1,                      \
+                       .channels_max = 16,                     \
+                       .rates = SNDRV_PCM_RATE_8000_192000,    \
+                       .formats = SNDRV_PCM_FMTBIT_S8 |        \
+                                  SNDRV_PCM_FMTBIT_S16_LE |    \
+                                  SNDRV_PCM_FMTBIT_S32_LE,     \
+               },                                              \
+               .ops = &tegra210_amx_in_dai_ops,                \
+       }
+
+#define OUT_DAI                                                        \
+       {                                                       \
+               .name = "AMX-TX-CIF",                           \
+               .playback = {                                   \
+                       .stream_name = "TX-CIF-Playback",       \
+                       .channels_min = 1,                      \
+                       .channels_max = 16,                     \
+                       .rates = SNDRV_PCM_RATE_8000_192000,    \
+                       .formats = SNDRV_PCM_FMTBIT_S8 |        \
+                                  SNDRV_PCM_FMTBIT_S16_LE |    \
+                                  SNDRV_PCM_FMTBIT_S32_LE,     \
+               },                                              \
+               .capture = {                                    \
+                       .stream_name = "TX-CIF-Capture",        \
+                       .channels_min = 1,                      \
+                       .channels_max = 16,                     \
+                       .rates = SNDRV_PCM_RATE_8000_192000,    \
+                       .formats = SNDRV_PCM_FMTBIT_S8 |        \
+                                  SNDRV_PCM_FMTBIT_S16_LE |    \
+                                  SNDRV_PCM_FMTBIT_S32_LE,     \
+               },                                              \
+               .ops = &tegra210_amx_out_dai_ops,               \
+       }
+
+static struct snd_soc_dai_driver tegra210_amx_dais[] = {
+       IN_DAI(1),
+       IN_DAI(2),
+       IN_DAI(3),
+       IN_DAI(4),
+       OUT_DAI,
+};
+
+static const struct snd_soc_dapm_widget tegra210_amx_widgets[] = {
+       SND_SOC_DAPM_AIF_IN("RX1", NULL, 0, TEGRA210_AMX_CTRL, 0, 0),
+       SND_SOC_DAPM_AIF_IN("RX2", NULL, 0, TEGRA210_AMX_CTRL, 1, 0),
+       SND_SOC_DAPM_AIF_IN("RX3", NULL, 0, TEGRA210_AMX_CTRL, 2, 0),
+       SND_SOC_DAPM_AIF_IN("RX4", NULL, 0, TEGRA210_AMX_CTRL, 3, 0),
+       SND_SOC_DAPM_AIF_OUT("TX", NULL, 0, TEGRA210_AMX_ENABLE,
+                            TEGRA210_AMX_ENABLE_SHIFT, 0),
+};
+
+#define STREAM_ROUTES(id, sname)                                         \
+       { "RX" #id " XBAR-" sname,      NULL,   "RX" #id " XBAR-TX" },    \
+       { "RX" #id "-CIF-" sname,       NULL,   "RX" #id " XBAR-" sname },\
+       { "RX" #id,                     NULL,   "RX" #id "-CIF-" sname }, \
+       { "TX",                         NULL,   "RX" #id },               \
+       { "TX-CIF-" sname,              NULL,   "TX" },                   \
+       { "XBAR-" sname,                NULL,   "TX-CIF-" sname },        \
+       { "XBAR-RX",                    NULL,   "XBAR-" sname }
+
+#define AMX_ROUTES(id)                 \
+       STREAM_ROUTES(id, "Playback"),  \
+       STREAM_ROUTES(id, "Capture")
+
+static const struct snd_soc_dapm_route tegra210_amx_routes[] = {
+       AMX_ROUTES(1),
+       AMX_ROUTES(2),
+       AMX_ROUTES(3),
+       AMX_ROUTES(4),
+};
+
+#define TEGRA210_AMX_BYTE_MAP_CTRL(reg)                                        \
+       SOC_SINGLE_EXT("Byte Map " #reg, reg, 0, 256, 0,                \
+                      tegra210_amx_get_byte_map,                       \
+                      tegra210_amx_put_byte_map)
+
+static struct snd_kcontrol_new tegra210_amx_controls[] = {
+       TEGRA210_AMX_BYTE_MAP_CTRL(0),
+       TEGRA210_AMX_BYTE_MAP_CTRL(1),
+       TEGRA210_AMX_BYTE_MAP_CTRL(2),
+       TEGRA210_AMX_BYTE_MAP_CTRL(3),
+       TEGRA210_AMX_BYTE_MAP_CTRL(4),
+       TEGRA210_AMX_BYTE_MAP_CTRL(5),
+       TEGRA210_AMX_BYTE_MAP_CTRL(6),
+       TEGRA210_AMX_BYTE_MAP_CTRL(7),
+       TEGRA210_AMX_BYTE_MAP_CTRL(8),
+       TEGRA210_AMX_BYTE_MAP_CTRL(9),
+       TEGRA210_AMX_BYTE_MAP_CTRL(10),
+       TEGRA210_AMX_BYTE_MAP_CTRL(11),
+       TEGRA210_AMX_BYTE_MAP_CTRL(12),
+       TEGRA210_AMX_BYTE_MAP_CTRL(13),
+       TEGRA210_AMX_BYTE_MAP_CTRL(14),
+       TEGRA210_AMX_BYTE_MAP_CTRL(15),
+       TEGRA210_AMX_BYTE_MAP_CTRL(16),
+       TEGRA210_AMX_BYTE_MAP_CTRL(17),
+       TEGRA210_AMX_BYTE_MAP_CTRL(18),
+       TEGRA210_AMX_BYTE_MAP_CTRL(19),
+       TEGRA210_AMX_BYTE_MAP_CTRL(20),
+       TEGRA210_AMX_BYTE_MAP_CTRL(21),
+       TEGRA210_AMX_BYTE_MAP_CTRL(22),
+       TEGRA210_AMX_BYTE_MAP_CTRL(23),
+       TEGRA210_AMX_BYTE_MAP_CTRL(24),
+       TEGRA210_AMX_BYTE_MAP_CTRL(25),
+       TEGRA210_AMX_BYTE_MAP_CTRL(26),
+       TEGRA210_AMX_BYTE_MAP_CTRL(27),
+       TEGRA210_AMX_BYTE_MAP_CTRL(28),
+       TEGRA210_AMX_BYTE_MAP_CTRL(29),
+       TEGRA210_AMX_BYTE_MAP_CTRL(30),
+       TEGRA210_AMX_BYTE_MAP_CTRL(31),
+       TEGRA210_AMX_BYTE_MAP_CTRL(32),
+       TEGRA210_AMX_BYTE_MAP_CTRL(33),
+       TEGRA210_AMX_BYTE_MAP_CTRL(34),
+       TEGRA210_AMX_BYTE_MAP_CTRL(35),
+       TEGRA210_AMX_BYTE_MAP_CTRL(36),
+       TEGRA210_AMX_BYTE_MAP_CTRL(37),
+       TEGRA210_AMX_BYTE_MAP_CTRL(38),
+       TEGRA210_AMX_BYTE_MAP_CTRL(39),
+       TEGRA210_AMX_BYTE_MAP_CTRL(40),
+       TEGRA210_AMX_BYTE_MAP_CTRL(41),
+       TEGRA210_AMX_BYTE_MAP_CTRL(42),
+       TEGRA210_AMX_BYTE_MAP_CTRL(43),
+       TEGRA210_AMX_BYTE_MAP_CTRL(44),
+       TEGRA210_AMX_BYTE_MAP_CTRL(45),
+       TEGRA210_AMX_BYTE_MAP_CTRL(46),
+       TEGRA210_AMX_BYTE_MAP_CTRL(47),
+       TEGRA210_AMX_BYTE_MAP_CTRL(48),
+       TEGRA210_AMX_BYTE_MAP_CTRL(49),
+       TEGRA210_AMX_BYTE_MAP_CTRL(50),
+       TEGRA210_AMX_BYTE_MAP_CTRL(51),
+       TEGRA210_AMX_BYTE_MAP_CTRL(52),
+       TEGRA210_AMX_BYTE_MAP_CTRL(53),
+       TEGRA210_AMX_BYTE_MAP_CTRL(54),
+       TEGRA210_AMX_BYTE_MAP_CTRL(55),
+       TEGRA210_AMX_BYTE_MAP_CTRL(56),
+       TEGRA210_AMX_BYTE_MAP_CTRL(57),
+       TEGRA210_AMX_BYTE_MAP_CTRL(58),
+       TEGRA210_AMX_BYTE_MAP_CTRL(59),
+       TEGRA210_AMX_BYTE_MAP_CTRL(60),
+       TEGRA210_AMX_BYTE_MAP_CTRL(61),
+       TEGRA210_AMX_BYTE_MAP_CTRL(62),
+       TEGRA210_AMX_BYTE_MAP_CTRL(63),
+};
+
+static const struct snd_soc_component_driver tegra210_amx_cmpnt = {
+       .dapm_widgets           = tegra210_amx_widgets,
+       .num_dapm_widgets       = ARRAY_SIZE(tegra210_amx_widgets),
+       .dapm_routes            = tegra210_amx_routes,
+       .num_dapm_routes        = ARRAY_SIZE(tegra210_amx_routes),
+       .controls               = tegra210_amx_controls,
+       .num_controls           = ARRAY_SIZE(tegra210_amx_controls),
+};
+
+static bool tegra210_amx_wr_reg(struct device *dev, unsigned int reg)
+{
+       switch (reg) {
+       case TEGRA210_AMX_RX_INT_MASK ... TEGRA210_AMX_RX4_CIF_CTRL:
+       case TEGRA210_AMX_TX_INT_MASK ... TEGRA210_AMX_CG:
+       case TEGRA210_AMX_CTRL ... TEGRA210_AMX_CYA:
+       case TEGRA210_AMX_CFG_RAM_CTRL ... TEGRA210_AMX_CFG_RAM_DATA:
+               return true;
+       default:
+               return false;
+       }
+}
+
+static bool tegra194_amx_wr_reg(struct device *dev, unsigned int reg)
+{
+       switch (reg) {
+       case TEGRA194_AMX_RX1_FRAME_PERIOD ... TEGRA194_AMX_RX4_FRAME_PERIOD:
+               return true;
+       default:
+               return tegra210_amx_wr_reg(dev, reg);
+       }
+}
+
+static bool tegra210_amx_rd_reg(struct device *dev, unsigned int reg)
+{
+       switch (reg) {
+       case TEGRA210_AMX_RX_STATUS ... TEGRA210_AMX_CFG_RAM_DATA:
+               return true;
+       default:
+               return false;
+       }
+}
+
+static bool tegra194_amx_rd_reg(struct device *dev, unsigned int reg)
+{
+       switch (reg) {
+       case TEGRA194_AMX_RX1_FRAME_PERIOD ... TEGRA194_AMX_RX4_FRAME_PERIOD:
+               return true;
+       default:
+               return tegra210_amx_rd_reg(dev, reg);
+       }
+}
+
+static bool tegra210_amx_volatile_reg(struct device *dev, unsigned int reg)
+{
+       switch (reg) {
+       case TEGRA210_AMX_RX_STATUS:
+       case TEGRA210_AMX_RX_INT_STATUS:
+       case TEGRA210_AMX_RX_INT_SET:
+       case TEGRA210_AMX_TX_STATUS:
+       case TEGRA210_AMX_TX_INT_STATUS:
+       case TEGRA210_AMX_TX_INT_SET:
+       case TEGRA210_AMX_SOFT_RESET:
+       case TEGRA210_AMX_STATUS:
+       case TEGRA210_AMX_INT_STATUS:
+       case TEGRA210_AMX_CFG_RAM_CTRL:
+       case TEGRA210_AMX_CFG_RAM_DATA:
+               return true;
+       default:
+               break;
+       }
+
+       return false;
+}
+
+static const struct regmap_config tegra210_amx_regmap_config = {
+       .reg_bits               = 32,
+       .reg_stride             = 4,
+       .val_bits               = 32,
+       .max_register           = TEGRA210_AMX_CFG_RAM_DATA,
+       .writeable_reg          = tegra210_amx_wr_reg,
+       .readable_reg           = tegra210_amx_rd_reg,
+       .volatile_reg           = tegra210_amx_volatile_reg,
+       .reg_defaults           = tegra210_amx_reg_defaults,
+       .num_reg_defaults       = ARRAY_SIZE(tegra210_amx_reg_defaults),
+       .cache_type             = REGCACHE_FLAT,
+};
+
+static const struct regmap_config tegra194_amx_regmap_config = {
+       .reg_bits               = 32,
+       .reg_stride             = 4,
+       .val_bits               = 32,
+       .max_register           = TEGRA194_AMX_RX4_LAST_FRAME_PERIOD,
+       .writeable_reg          = tegra194_amx_wr_reg,
+       .readable_reg           = tegra194_amx_rd_reg,
+       .volatile_reg           = tegra210_amx_volatile_reg,
+       .reg_defaults           = tegra210_amx_reg_defaults,
+       .num_reg_defaults       = ARRAY_SIZE(tegra210_amx_reg_defaults),
+       .cache_type             = REGCACHE_FLAT,
+};
+
+static const struct tegra210_amx_soc_data soc_data_tegra210 = {
+       .regmap_conf    = &tegra210_amx_regmap_config,
+};
+
+static const struct tegra210_amx_soc_data soc_data_tegra194 = {
+       .regmap_conf    = &tegra194_amx_regmap_config,
+       .auto_disable   = true,
+};
+
+static const struct of_device_id tegra210_amx_of_match[] = {
+       { .compatible = "nvidia,tegra210-amx", .data = &soc_data_tegra210 },
+       { .compatible = "nvidia,tegra194-amx", .data = &soc_data_tegra194 },
+       {},
+};
+MODULE_DEVICE_TABLE(of, tegra210_amx_of_match);
+
+static int tegra210_amx_platform_probe(struct platform_device *pdev)
+{
+       struct device *dev = &pdev->dev;
+       struct tegra210_amx *amx;
+       void __iomem *regs;
+       int err;
+       const struct of_device_id *match;
+       struct tegra210_amx_soc_data *soc_data;
+
+       match = of_match_device(tegra210_amx_of_match, dev);
+
+       soc_data = (struct tegra210_amx_soc_data *)match->data;
+
+       amx = devm_kzalloc(dev, sizeof(*amx), GFP_KERNEL);
+       if (!amx)
+               return -ENOMEM;
+
+       amx->soc_data = soc_data;
+
+       dev_set_drvdata(dev, amx);
+
+       regs = devm_platform_ioremap_resource(pdev, 0);
+       if (IS_ERR(regs))
+               return PTR_ERR(regs);
+
+       amx->regmap = devm_regmap_init_mmio(dev, regs,
+                                           soc_data->regmap_conf);
+       if (IS_ERR(amx->regmap)) {
+               dev_err(dev, "regmap init failed\n");
+               return PTR_ERR(amx->regmap);
+       }
+
+       regcache_cache_only(amx->regmap, true);
+
+       err = devm_snd_soc_register_component(dev, &tegra210_amx_cmpnt,
+                                             tegra210_amx_dais,
+                                             ARRAY_SIZE(tegra210_amx_dais));
+       if (err) {
+               dev_err(dev, "can't register AMX component, err: %d\n", err);
+               return err;
+       }
+
+       pm_runtime_enable(dev);
+
+       return 0;
+}
+
+static int tegra210_amx_platform_remove(struct platform_device *pdev)
+{
+       pm_runtime_disable(&pdev->dev);
+
+       return 0;
+}
+
+static const struct dev_pm_ops tegra210_amx_pm_ops = {
+       SET_RUNTIME_PM_OPS(tegra210_amx_runtime_suspend,
+                          tegra210_amx_runtime_resume, NULL)
+       SET_LATE_SYSTEM_SLEEP_PM_OPS(pm_runtime_force_suspend,
+                                    pm_runtime_force_resume)
+};
+
+static struct platform_driver tegra210_amx_driver = {
+       .driver = {
+               .name = "tegra210-amx",
+               .of_match_table = tegra210_amx_of_match,
+               .pm = &tegra210_amx_pm_ops,
+       },
+       .probe = tegra210_amx_platform_probe,
+       .remove = tegra210_amx_platform_remove,
+};
+module_platform_driver(tegra210_amx_driver);
+
+MODULE_AUTHOR("Songhee Baek <sbaek@nvidia.com>");
+MODULE_DESCRIPTION("Tegra210 AMX ASoC driver");
+MODULE_LICENSE("GPL v2");
diff --git a/sound/soc/tegra/tegra210_amx.h b/sound/soc/tegra/tegra210_amx.h
new file mode 100644 (file)
index 0000000..e277741
--- /dev/null
@@ -0,0 +1,93 @@
+/* SPDX-License-Identifier: GPL-2.0-only */
+/*
+ * tegra210_amx.h - Definitions for Tegra210 AMX driver
+ *
+ * Copyright (c) 2021, NVIDIA CORPORATION.  All rights reserved.
+ *
+ */
+
+#ifndef __TEGRA210_AMX_H__
+#define __TEGRA210_AMX_H__
+
+/* Register offsets from TEGRA210_AMX*_BASE */
+#define TEGRA210_AMX_RX_STATUS                 0x0c
+#define TEGRA210_AMX_RX_INT_STATUS             0x10
+#define TEGRA210_AMX_RX_INT_MASK               0x14
+#define TEGRA210_AMX_RX_INT_SET                        0x18
+#define TEGRA210_AMX_RX_INT_CLEAR              0x1c
+#define TEGRA210_AMX_RX1_CIF_CTRL              0x20
+#define TEGRA210_AMX_RX2_CIF_CTRL              0x24
+#define TEGRA210_AMX_RX3_CIF_CTRL              0x28
+#define TEGRA210_AMX_RX4_CIF_CTRL              0x2c
+#define TEGRA210_AMX_TX_STATUS                 0x4c
+#define TEGRA210_AMX_TX_INT_STATUS             0x50
+#define TEGRA210_AMX_TX_INT_MASK               0x54
+#define TEGRA210_AMX_TX_INT_SET                        0x58
+#define TEGRA210_AMX_TX_INT_CLEAR              0x5c
+#define TEGRA210_AMX_TX_CIF_CTRL               0x60
+#define TEGRA210_AMX_ENABLE                    0x80
+#define TEGRA210_AMX_SOFT_RESET                        0x84
+#define TEGRA210_AMX_CG                                0x88
+#define TEGRA210_AMX_STATUS                    0x8c
+#define TEGRA210_AMX_INT_STATUS                        0x90
+#define TEGRA210_AMX_CTRL                      0xa4
+#define TEGRA210_AMX_OUT_BYTE_EN0              0xa8
+#define TEGRA210_AMX_OUT_BYTE_EN1              0xac
+#define TEGRA210_AMX_CYA                       0xb0
+#define TEGRA210_AMX_CFG_RAM_CTRL              0xb8
+#define TEGRA210_AMX_CFG_RAM_DATA              0xbc
+
+#define TEGRA194_AMX_RX1_FRAME_PERIOD          0xc0
+#define TEGRA194_AMX_RX4_FRAME_PERIOD          0xcc
+#define TEGRA194_AMX_RX4_LAST_FRAME_PERIOD     0xdc
+
+/* Fields in TEGRA210_AMX_ENABLE */
+#define TEGRA210_AMX_ENABLE_SHIFT                      0
+
+/* Fields in TEGRA210_AMX_CTRL */
+#define TEGRA210_AMX_CTRL_MSTR_RX_NUM_SHIFT            14
+#define TEGRA210_AMX_CTRL_MSTR_RX_NUM_MASK             (3 << TEGRA210_AMX_CTRL_MSTR_RX_NUM_SHIFT)
+
+#define TEGRA210_AMX_CTRL_RX_DEP_SHIFT                 12
+#define TEGRA210_AMX_CTRL_RX_DEP_MASK                  (3 << TEGRA210_AMX_CTRL_RX_DEP_SHIFT)
+
+/* Fields in TEGRA210_AMX_CFG_RAM_CTRL */
+#define TEGRA210_AMX_CFG_RAM_CTRL_RW_SHIFT             14
+#define TEGRA210_AMX_CFG_RAM_CTRL_RW_WRITE             (1 << TEGRA210_AMX_CFG_RAM_CTRL_RW_SHIFT)
+
+#define TEGRA210_AMX_CFG_RAM_CTRL_ADDR_INIT_EN_SHIFT   13
+#define TEGRA210_AMX_CFG_RAM_CTRL_ADDR_INIT_EN         (1 << TEGRA210_AMX_CFG_RAM_CTRL_ADDR_INIT_EN_SHIFT)
+
+#define TEGRA210_AMX_CFG_RAM_CTRL_SEQ_ACCESS_EN_SHIFT  12
+#define TEGRA210_AMX_CFG_RAM_CTRL_SEQ_ACCESS_EN                (1 << TEGRA210_AMX_CFG_RAM_CTRL_SEQ_ACCESS_EN_SHIFT)
+
+#define TEGRA210_AMX_CFG_CTRL_RAM_ADDR_SHIFT           0
+
+/* Fields in TEGRA210_AMX_SOFT_RESET */
+#define TEGRA210_AMX_SOFT_RESET_SOFT_EN                        1
+#define TEGRA210_AMX_SOFT_RESET_SOFT_RESET_MASK                TEGRA210_AMX_SOFT_RESET_SOFT_EN
+
+#define TEGRA210_AMX_AUDIOCIF_CH_STRIDE                4
+#define TEGRA210_AMX_RAM_DEPTH                 16
+#define TEGRA210_AMX_MAP_STREAM_NUM_SHIFT      6
+#define TEGRA210_AMX_MAP_WORD_NUM_SHIFT                2
+#define TEGRA210_AMX_MAP_BYTE_NUM_SHIFT                0
+
+enum {
+       TEGRA210_AMX_WAIT_ON_ALL,
+       TEGRA210_AMX_WAIT_ON_ANY,
+};
+
+struct tegra210_amx_soc_data {
+       const struct regmap_config *regmap_conf;
+       bool auto_disable;
+};
+
+struct tegra210_amx {
+       const struct tegra210_amx_soc_data *soc_data;
+       unsigned int map[TEGRA210_AMX_RAM_DEPTH];
+       struct regmap *regmap;
+       unsigned int byte_mask[2];
+};
+
+#endif