root/sound/soc/mediatek/mt8189/mt8189-afe-pcm.c
// SPDX-License-Identifier: GPL-2.0
/*
 *  Mediatek ALSA SoC AFE platform driver for 8189
 *
 *  Copyright (c) 2025 MediaTek Inc.
 *  Author: Darren Ye <darren.ye@mediatek.com>
 */

#include <linux/delay.h>
#include <linux/dma-mapping.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/of_device.h>
#include <linux/of_reserved_mem.h>
#include <linux/pm_runtime.h>
#include <linux/regmap.h>

#include <sound/pcm.h>
#include <sound/soc.h>

#include "mt8189-afe-clk.h"
#include "mt8189-afe-common.h"
#include "mt8189-interconnection.h"

#include "../common/mtk-afe-fe-dai.h"
#include "../common/mtk-afe-platform-driver.h"

static const struct snd_pcm_hardware mt8189_afe_hardware = {
        .info = (SNDRV_PCM_INFO_MMAP |
                 SNDRV_PCM_INFO_NO_PERIOD_WAKEUP |
                 SNDRV_PCM_INFO_INTERLEAVED |
                 SNDRV_PCM_INFO_MMAP_VALID),
        .formats = (SNDRV_PCM_FMTBIT_S16_LE |
                    SNDRV_PCM_FMTBIT_S24_LE |
                    SNDRV_PCM_FMTBIT_S32_LE),
        .period_bytes_min = 96,
        .period_bytes_max = 4 * 48 * 1024,
        .periods_min = 2,
        .periods_max = 256,
        .buffer_bytes_max = 256 * 1024,
        .fifo_size = 0,
};

static unsigned int mt8189_rate_transform(struct device *dev, unsigned int rate)
{
        switch (rate) {
        case 8000:
                return MTK_AFE_IPM2P0_RATE_8K;
        case 11025:
                return MTK_AFE_IPM2P0_RATE_11K;
        case 12000:
                return MTK_AFE_IPM2P0_RATE_12K;
        case 16000:
                return MTK_AFE_IPM2P0_RATE_16K;
        case 22050:
                return MTK_AFE_IPM2P0_RATE_22K;
        case 24000:
                return MTK_AFE_IPM2P0_RATE_24K;
        case 32000:
                return MTK_AFE_IPM2P0_RATE_32K;
        case 44100:
                return MTK_AFE_IPM2P0_RATE_44K;
        case 48000:
                return MTK_AFE_IPM2P0_RATE_48K;
        case 88200:
                return MTK_AFE_IPM2P0_RATE_88K;
        case 96000:
                return MTK_AFE_IPM2P0_RATE_96K;
        case 176400:
                return MTK_AFE_IPM2P0_RATE_176K;
        case 192000:
                return MTK_AFE_IPM2P0_RATE_192K;
        /* not support 260K */
        case 352800:
                return MTK_AFE_IPM2P0_RATE_352K;
        case 384000:
                return MTK_AFE_IPM2P0_RATE_384K;
        default:
                dev_warn(dev, "rate %u invalid, use %d!!!\n",
                         rate, MTK_AFE_IPM2P0_RATE_48K);
                return MTK_AFE_IPM2P0_RATE_48K;
        }
}

static inline unsigned int calculate_cm_update(unsigned int rate,
                                               unsigned int ch)
{
        return (((26000000 / rate) - 10) / (ch / 2)) - 1;
}

static int mt8189_set_cm(struct mtk_base_afe *afe, int id,
                         bool update, bool swap, unsigned int ch)
{
        struct mt8189_afe_private *afe_priv = afe->platform_priv;
        unsigned int rate = afe_priv->cm_rate[id];
        unsigned int rate_val = mt8189_rate_transform(afe->dev, rate);
        unsigned int update_val = update ? calculate_cm_update(rate, ch) : 0x64;
        int reg = AFE_CM0_CON0 + 0x10 * id;

        dev_dbg(afe->dev, "%s()-0, CM%d, rate %d, update %d, swap %d, ch %d\n",
                __func__, id, rate, update, swap, ch);

        /* update cnt */
        regmap_update_bits(afe->regmap, reg,
                           AFE_CM_UPDATE_CNT_MASK << AFE_CM_UPDATE_CNT_SFT,
                           update_val << AFE_CM_UPDATE_CNT_SFT);

        /* rate */
        regmap_update_bits(afe->regmap, reg,
                           AFE_CM_1X_EN_SEL_FS_MASK << AFE_CM_1X_EN_SEL_FS_SFT,
                           rate_val << AFE_CM_1X_EN_SEL_FS_SFT);

        /* ch num */
        regmap_update_bits(afe->regmap, reg,
                           AFE_CM_CH_NUM_MASK << AFE_CM_CH_NUM_SFT,
                           (ch - 1) << AFE_CM_CH_NUM_SFT);

        /* swap */
        regmap_update_bits(afe->regmap, reg,
                           AFE_CM_BYTE_SWAP_MASK << AFE_CM_BYTE_SWAP_SFT,
                           swap << AFE_CM_BYTE_SWAP_SFT);

        return 0;
}

static int mt8189_enable_cm_bypass(struct mtk_base_afe *afe, int id, bool en)
{
        return regmap_update_bits(afe->regmap, AFE_CM0_CON0 + 0x10 * id,
                                  AFE_CM_BYPASS_MODE_MASK <<
                                  AFE_CM_BYPASS_MODE_SFT,
                                  en << AFE_CM_BYPASS_MODE_SFT);
}

static int mt8189_fe_startup(struct snd_pcm_substream *substream,
                             struct snd_soc_dai *dai)
{
        struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);
        struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
        struct snd_pcm_runtime *runtime = substream->runtime;
        struct snd_soc_dai *cpu_dai = snd_soc_rtd_to_cpu(rtd, 0);
        int memif_num = cpu_dai->id;
        struct mtk_base_afe_memif *memif = &afe->memif[memif_num];
        const struct snd_pcm_hardware *mtk_afe_hardware = afe->mtk_afe_hardware;
        int ret;

        dev_dbg(afe->dev, "%s(), memif_num: %d.\n", __func__, memif_num);

        memif->substream = substream;

        snd_pcm_hw_constraint_step(substream->runtime, 0,
                                   SNDRV_PCM_HW_PARAM_BUFFER_BYTES, 16);

        snd_soc_set_runtime_hwparams(substream, mtk_afe_hardware);

        ret = snd_pcm_hw_constraint_integer(runtime,
                                            SNDRV_PCM_HW_PARAM_PERIODS);
        if (ret < 0)
                dev_warn(afe->dev, "snd_pcm_hw_constraint_integer failed\n");

        /* dynamic allocate irq to memif */
        if (memif->irq_usage < 0) {
                int irq_id = mtk_dynamic_irq_acquire(afe);

                if (irq_id != afe->irqs_size) {
                        /* link */
                        memif->irq_usage = irq_id;
                } else {
                        dev_err(afe->dev, "%s() error: no more asys irq\n",
                                __func__);
                        ret = -EBUSY;
                }
        }

        return ret;
}

static void mt8189_fe_shutdown(struct snd_pcm_substream *substream,
                               struct snd_soc_dai *dai)
{
        struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);
        struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
        struct snd_soc_dai *cpu_dai = snd_soc_rtd_to_cpu(rtd, 0);
        int memif_num = cpu_dai->id;
        struct mtk_base_afe_memif *memif = &afe->memif[memif_num];
        int irq_id = memif->irq_usage;

        dev_dbg(afe->dev, "%s(), memif_num: %d.\n", __func__, memif_num);

        memif->substream = NULL;

        if (!memif->const_irq) {
                mtk_dynamic_irq_release(afe, irq_id);
                memif->irq_usage = -1;
                memif->substream = NULL;
        }
}

static int mt8189_fe_hw_params(struct snd_pcm_substream *substream,
                               struct snd_pcm_hw_params *params,
                               struct snd_soc_dai *dai)
{
        struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
        struct mt8189_afe_private *afe_priv = afe->platform_priv;
        int id = dai->id;
        int cm;

        switch (id) {
        case MT8189_MEMIF_VUL8:
        case MT8189_MEMIF_VUL_CM0:
                cm = CM0;
                break;
        case MT8189_MEMIF_VUL9:
        case MT8189_MEMIF_VUL_CM1:
                cm = CM1;
                break;
        default:
                cm = CM0;
                break;
        }

        afe_priv->cm_rate[cm] = params_rate(params);
        afe_priv->cm_channels = params_channels(params);

        return mtk_afe_fe_hw_params(substream, params, dai);
}

static int mt8189_fe_trigger(struct snd_pcm_substream *substream, int cmd,
                             struct snd_soc_dai *dai)
{
        struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);
        struct snd_pcm_runtime *const runtime = substream->runtime;
        struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
        struct snd_soc_dai *cpu_dai = snd_soc_rtd_to_cpu(rtd, 0);
        int id = cpu_dai->id;
        struct mtk_base_afe_memif *memif = &afe->memif[id];
        int irq_id = memif->irq_usage;
        struct mtk_base_afe_irq *irqs = &afe->irqs[irq_id];
        const struct mtk_base_irq_data *irq_data = irqs->irq_data;
        unsigned int counter = runtime->period_size;
        unsigned int rate = runtime->rate;
        unsigned int tmp_reg;
        int fs;
        int ret;

        dev_dbg(afe->dev, "%s(), %s cmd %d, irq_id %d, dai_id %d\n", __func__,
                memif->data->name, cmd, irq_id, id);

        switch (cmd) {
        case SNDRV_PCM_TRIGGER_START:
        case SNDRV_PCM_TRIGGER_RESUME:
                ret = mtk_memif_set_enable(afe, id);
                if (ret) {
                        dev_err(afe->dev, "id %d, memif enable fail.\n", id);
                        return ret;
                }

                /*
                 * for small latency record
                 * ul memif need read some data before irq enable
                 * the context of this triger ops is atmoic, so it cannot sleep
                 */
                if (substream->stream == SNDRV_PCM_STREAM_CAPTURE)
                        if ((runtime->period_size * 1000) / rate <= 10)
                                udelay(300);

                regmap_update_bits(afe->regmap,
                                   irq_data->irq_cnt_reg,
                                   irq_data->irq_cnt_maskbit <<
                                   irq_data->irq_cnt_shift,
                                   counter << irq_data->irq_cnt_shift);

                /* set irq fs */
                fs = afe->irq_fs(substream, rate);
                if (fs < 0)
                        return -EINVAL;

                if (irq_data->irq_fs_reg >= 0)
                        regmap_update_bits(afe->regmap,
                                           irq_data->irq_fs_reg,
                                           irq_data->irq_fs_maskbit <<
                                           irq_data->irq_fs_shift,
                                           fs << irq_data->irq_fs_shift);

                /* enable interrupt */
                regmap_update_bits(afe->regmap,
                                   irq_data->irq_en_reg,
                                   1 << irq_data->irq_en_shift,
                                   1 << irq_data->irq_en_shift);

                return 0;
        case SNDRV_PCM_TRIGGER_STOP:
        case SNDRV_PCM_TRIGGER_SUSPEND:
                ret = mtk_memif_set_disable(afe, id);
                if (ret)
                        dev_warn(afe->dev, "id %d, memif disable fail\n", id);

                /* disable interrupt */
                regmap_update_bits(afe->regmap,
                                   irq_data->irq_en_reg,
                                   1 << irq_data->irq_en_shift,
                                   0 << irq_data->irq_en_shift);

                /*
                 * clear pending IRQ, if the register read as one, there is no
                 * need to write one to clear operation.
                 */
                regmap_read(afe->regmap, irq_data->irq_clr_reg, &tmp_reg);
                regmap_update_bits(afe->regmap, irq_data->irq_clr_reg,
                                   AFE_IRQ_CLR_CFG_MASK_SFT |
                                   AFE_IRQ_MISS_FLAG_CLR_CFG_MASK_SFT,
                                   tmp_reg ^ (AFE_IRQ_CLR_CFG_MASK_SFT |
                                              AFE_IRQ_MISS_FLAG_CLR_CFG_MASK_SFT));

                return ret;
        default:
                return -EINVAL;
        }
}

static int mt8189_memif_fs(struct snd_pcm_substream *substream,
                           unsigned int rate)
{
        struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);
        struct snd_soc_component *component =
                snd_soc_rtdcom_lookup(rtd, AFE_PCM_NAME);
        struct mtk_base_afe *afe = NULL;

        if (!component)
                return -EINVAL;

        afe = snd_soc_component_get_drvdata(component);
        if (!afe)
                return -EINVAL;

        return mt8189_rate_transform(afe->dev, rate);
}

static int mt8189_get_dai_fs(struct mtk_base_afe *afe,
                             int dai_id, unsigned int rate)
{
        return mt8189_rate_transform(afe->dev, rate);
}

static int mt8189_irq_fs(struct snd_pcm_substream *substream, unsigned int rate)
{
        struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);
        struct snd_soc_component *component =
                snd_soc_rtdcom_lookup(rtd, AFE_PCM_NAME);
        struct mtk_base_afe *afe = NULL;

        if (!component)
                return -EINVAL;
        afe = snd_soc_component_get_drvdata(component);

        return mt8189_rate_transform(afe->dev, rate);
}

static int mt8189_get_memif_pbuf_size(struct snd_pcm_substream *substream)
{
        struct snd_pcm_runtime *runtime = substream->runtime;

        if ((runtime->period_size * 1000) / runtime->rate > 10)
                return MT8189_MEMIF_PBUF_SIZE_256_BYTES;

        return MT8189_MEMIF_PBUF_SIZE_32_BYTES;
}

/* FE DAIs */
static const struct snd_soc_dai_ops mt8189_memif_dai_ops = {
        .startup        = mt8189_fe_startup,
        .shutdown       = mt8189_fe_shutdown,
        .hw_params      = mt8189_fe_hw_params,
        .hw_free        = mtk_afe_fe_hw_free,
        .prepare        = mtk_afe_fe_prepare,
        .trigger        = mt8189_fe_trigger,
};

#define MTK_PCM_RATES (SNDRV_PCM_RATE_8000_48000 | \
                       SNDRV_PCM_RATE_88200 | \
                       SNDRV_PCM_RATE_96000 | \
                       SNDRV_PCM_RATE_176400 | \
                       SNDRV_PCM_RATE_192000)

#define MTK_PCM_DAI_RATES (SNDRV_PCM_RATE_8000 | \
                           SNDRV_PCM_RATE_16000 | \
                           SNDRV_PCM_RATE_32000 | \
                           SNDRV_PCM_RATE_48000)

#define MTK_PCM_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | \
                         SNDRV_PCM_FMTBIT_S24_LE | \
                         SNDRV_PCM_FMTBIT_S32_LE)

#define MT8189_FE_DAI_PLAYBACK(_name, _id, max_ch) \
{ \
        .name = #_name, \
        .id = _id, \
        .playback = { \
                .stream_name = #_name, \
                .channels_min = 1, \
                .channels_max = max_ch, \
                .rates = MTK_PCM_RATES, \
                .formats = MTK_PCM_FORMATS, \
        }, \
        .ops = &mt8189_memif_dai_ops, \
}

#define MT8189_FE_DAI_CAPTURE(_name, _id, max_ch) \
{ \
        .name = #_name, \
        .id = _id, \
        .capture = { \
                .stream_name = #_name, \
                .channels_min = 1, \
                .channels_max = max_ch, \
                .rates = MTK_PCM_RATES, \
                .formats = MTK_PCM_FORMATS, \
        }, \
        .ops = &mt8189_memif_dai_ops, \
}

static struct snd_soc_dai_driver mt8189_memif_dai_driver[] = {
        /* FE DAIs: memory interfaces to CPU */
        /* Playback */
        MT8189_FE_DAI_PLAYBACK(DL0, MT8189_MEMIF_DL0, 2),
        MT8189_FE_DAI_PLAYBACK(DL1, MT8189_MEMIF_DL1, 2),
        MT8189_FE_DAI_PLAYBACK(DL2, MT8189_MEMIF_DL2, 2),
        MT8189_FE_DAI_PLAYBACK(DL3, MT8189_MEMIF_DL3, 2),
        MT8189_FE_DAI_PLAYBACK(DL4, MT8189_MEMIF_DL4, 2),
        MT8189_FE_DAI_PLAYBACK(DL5, MT8189_MEMIF_DL5, 2),
        MT8189_FE_DAI_PLAYBACK(DL6, MT8189_MEMIF_DL6, 2),
        MT8189_FE_DAI_PLAYBACK(DL7, MT8189_MEMIF_DL7, 2),
        MT8189_FE_DAI_PLAYBACK(DL8, MT8189_MEMIF_DL8, 2),
        MT8189_FE_DAI_PLAYBACK(DL23, MT8189_MEMIF_DL23, 2),
        MT8189_FE_DAI_PLAYBACK(DL24, MT8189_MEMIF_DL24, 2),
        MT8189_FE_DAI_PLAYBACK(DL25, MT8189_MEMIF_DL25, 2),
        MT8189_FE_DAI_PLAYBACK(DL_24CH, MT8189_MEMIF_DL_24CH, 8),
        MT8189_FE_DAI_PLAYBACK(HDMI, MT8189_MEMIF_HDMI, 8),
        /* Capture */
        MT8189_FE_DAI_CAPTURE(UL0, MT8189_MEMIF_VUL0, 2),
        MT8189_FE_DAI_CAPTURE(UL1, MT8189_MEMIF_VUL1, 2),
        MT8189_FE_DAI_CAPTURE(UL2, MT8189_MEMIF_VUL2, 2),
        MT8189_FE_DAI_CAPTURE(UL3, MT8189_MEMIF_VUL3, 2),
        MT8189_FE_DAI_CAPTURE(UL4, MT8189_MEMIF_VUL4, 2),
        MT8189_FE_DAI_CAPTURE(UL5, MT8189_MEMIF_VUL5, 2),
        MT8189_FE_DAI_CAPTURE(UL6, MT8189_MEMIF_VUL6, 2),
        MT8189_FE_DAI_CAPTURE(UL7, MT8189_MEMIF_VUL7, 2),
        MT8189_FE_DAI_CAPTURE(UL8, MT8189_MEMIF_VUL8, 2),
        MT8189_FE_DAI_CAPTURE(UL9, MT8189_MEMIF_VUL9, 16),
        MT8189_FE_DAI_CAPTURE(UL10, MT8189_MEMIF_VUL10, 2),
        MT8189_FE_DAI_CAPTURE(UL24, MT8189_MEMIF_VUL24, 2),
        MT8189_FE_DAI_CAPTURE(UL25, MT8189_MEMIF_VUL25, 2),
        MT8189_FE_DAI_CAPTURE(UL_CM0, MT8189_MEMIF_VUL_CM0, 8),
        MT8189_FE_DAI_CAPTURE(UL_CM1, MT8189_MEMIF_VUL_CM1, 16),
        MT8189_FE_DAI_CAPTURE(UL_ETDM_IN0, MT8189_MEMIF_ETDM_IN0, 2),
        MT8189_FE_DAI_CAPTURE(UL_ETDM_IN1, MT8189_MEMIF_ETDM_IN1, 2),
};

static int ul_cm0_event(struct snd_soc_dapm_widget *w,
                        struct snd_kcontrol *kcontrol,
                        int event)
{
        struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
        struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt);
        struct mt8189_afe_private *afe_priv = afe->platform_priv;
        unsigned int channels = afe_priv->cm_channels;

        dev_dbg(afe->dev, "%s(), event 0x%x, name %s, channels %d\n",
                __func__, event, w->name, channels);

        switch (event) {
        case SND_SOC_DAPM_PRE_PMU:
                mt8189_enable_cm_bypass(afe, CM0, false);
                mt8189_set_cm(afe, CM0, true, false, channels);
                regmap_update_bits(afe->regmap, AUDIO_TOP_CON0,
                                   PDN_CM0_MASK_SFT, 0 << PDN_CM0_SFT);

                break;
        case SND_SOC_DAPM_PRE_PMD:
                mt8189_enable_cm_bypass(afe, CM0, true);
                regmap_update_bits(afe->regmap, AUDIO_TOP_CON0,
                                   PDN_CM0_MASK_SFT, 1 << PDN_CM0_SFT);
                break;
        default:
                break;
        }

        return 0;
}

static int ul_cm1_event(struct snd_soc_dapm_widget *w,
                        struct snd_kcontrol *kcontrol,
                        int event)
{
        struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
        struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt);
        struct mt8189_afe_private *afe_priv = afe->platform_priv;
        unsigned int channels = afe_priv->cm_channels;

        dev_dbg(afe->dev, "%s(), event 0x%x, name %s, channels %d\n",
                __func__, event, w->name, channels);

        switch (event) {
        case SND_SOC_DAPM_PRE_PMU:
                mt8189_enable_cm_bypass(afe, CM1, false);
                mt8189_set_cm(afe, CM1, true, false, channels);
                regmap_update_bits(afe->regmap, AUDIO_TOP_CON0,
                                   PDN_CM1_MASK_SFT, 0 << PDN_CM1_SFT);
                break;
        case SND_SOC_DAPM_POST_PMD:
                mt8189_enable_cm_bypass(afe, CM1, true);
                regmap_update_bits(afe->regmap, AUDIO_TOP_CON0,
                                   PDN_CM1_MASK_SFT, 1 << PDN_CM1_SFT);
                break;
        default:
                break;
        }

        return 0;
}

/*
 * dma widget & routes
 * The mixer controls and routes are by no means fully implemented,
 * only the ones that are intended to be used are, as other wise a fully
 * interconnected switch bar mixer would introduce way too many unused
 * controls.
 */
static const struct snd_kcontrol_new memif_ul0_ch1_mix[] = {
        /* Normal record */
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN018_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN018_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN018_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN018_0,
                                    I_ADDA_UL_CH4, 1, 0),
        /* AP DMIC */
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH1", AFE_CONN018_0,
                                    I_DMIC0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH2", AFE_CONN018_0,
                                    I_DMIC0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH1", AFE_CONN018_1,
                                    I_DL0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL1_CH1", AFE_CONN018_1,
                                    I_DL1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH1", AFE_CONN018_1,
                                    I_DL2_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL3_CH1", AFE_CONN018_1,
                                    I_DL3_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL4_CH1", AFE_CONN018_1,
                                    I_DL4_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL6_CH1", AFE_CONN018_1,
                                    I_DL6_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL7_CH1", AFE_CONN018_1,
                                    I_DL7_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL23_CH1", AFE_CONN018_2,
                                    I_DL23_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL_24CH_CH1", AFE_CONN018_1,
                                    I_DL_24CH_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH1", AFE_CONN018_4,
                                    I_PCM_0_CAP_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN0_CH1", AFE_CONN018_4,
                                    I_I2SIN0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN1_CH1", AFE_CONN018_4,
                                    I_I2SIN1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_0_OUT_CH1", AFE_CONN018_6,
                                    I_SRC_0_OUT_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_2_OUT_CH1", AFE_CONN018_6,
                                    I_SRC_2_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul0_ch2_mix[] = {
        /* Normal record */
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN019_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN019_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN019_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN019_0,
                                    I_ADDA_UL_CH4, 1, 0),
        /* AP DMIC */
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH2", AFE_CONN019_0,
                                    I_DMIC0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH2", AFE_CONN019_1,
                                    I_DL0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL1_CH2", AFE_CONN019_1,
                                    I_DL1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH2", AFE_CONN019_1,
                                    I_DL2_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL3_CH2", AFE_CONN019_1,
                                    I_DL3_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL4_CH2", AFE_CONN019_1,
                                    I_DL4_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL6_CH2", AFE_CONN019_1,
                                    I_DL6_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL7_CH2", AFE_CONN019_1,
                                    I_DL7_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL23_CH2", AFE_CONN018_2,
                                    I_DL23_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL_24CH_CH2", AFE_CONN019_1,
                                    I_DL_24CH_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH1", AFE_CONN019_4,
                                    I_PCM_0_CAP_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH2", AFE_CONN019_4,
                                    I_PCM_0_CAP_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN0_CH2", AFE_CONN019_4,
                                    I_I2SIN0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN1_CH2", AFE_CONN019_4,
                                    I_I2SIN1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_0_OUT_CH2", AFE_CONN019_6,
                                    I_SRC_0_OUT_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_2_OUT_CH2", AFE_CONN019_6,
                                    I_SRC_2_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul1_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN020_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH1", AFE_CONN020_1,
                                    I_DL0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL1_CH1", AFE_CONN020_1,
                                    I_DL1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH1", AFE_CONN020_1,
                                    I_DL2_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL3_CH1", AFE_CONN020_1,
                                    I_DL3_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL4_CH1", AFE_CONN020_1,
                                    I_DL4_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL6_CH1", AFE_CONN020_1,
                                    I_DL6_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL7_CH1", AFE_CONN020_1,
                                    I_DL7_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL23_CH1", AFE_CONN020_2,
                                    I_DL23_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL_24CH_CH1", AFE_CONN020_1,
                                    I_DL_24CH_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH1", AFE_CONN020_4,
                                    I_PCM_0_CAP_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN0_CH1", AFE_CONN020_4,
                                    I_I2SIN0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN1_CH1", AFE_CONN020_4,
                                    I_I2SIN1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_0_OUT_CH1", AFE_CONN020_6,
                                    I_SRC_0_OUT_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_2_OUT_CH1", AFE_CONN020_6,
                                    I_SRC_2_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul1_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN021_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH2", AFE_CONN021_1,
                                    I_DL0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL1_CH2", AFE_CONN021_1,
                                    I_DL1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH2", AFE_CONN021_1,
                                    I_DL2_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL3_CH2", AFE_CONN021_1,
                                    I_DL3_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL4_CH2", AFE_CONN021_1,
                                    I_DL4_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL6_CH2", AFE_CONN021_1,
                                    I_DL6_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL7_CH2", AFE_CONN021_1,
                                    I_DL7_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL23_CH2", AFE_CONN021_2,
                                    I_DL23_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL_24CH_CH2", AFE_CONN021_1,
                                    I_DL_24CH_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH1", AFE_CONN021_4,
                                    I_PCM_0_CAP_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH2", AFE_CONN021_4,
                                    I_PCM_0_CAP_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN0_CH2", AFE_CONN021_4,
                                    I_I2SIN0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN1_CH2", AFE_CONN021_4,
                                    I_I2SIN1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_0_OUT_CH2", AFE_CONN021_6,
                                    I_SRC_0_OUT_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_2_OUT_CH2", AFE_CONN021_6,
                                    I_SRC_2_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul2_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN022_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN022_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN022_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN022_0,
                                    I_ADDA_UL_CH4, 1, 0),
        /* AP DMIC */
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH3", AFE_CONN022_0,
                                    I_DMIC1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_GAIN1_OUT_CH1", AFE_CONN022_0,
                                    I_GAIN1_OUT_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_1_OUT_CH1", AFE_CONN022_6,
                                    I_SRC_1_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul2_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN023_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN023_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN023_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN023_0,
                                    I_ADDA_UL_CH4, 1, 0),
        /* AP DMIC */
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH4", AFE_CONN023_0,
                                    I_DMIC1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_GAIN1_OUT_CH2", AFE_CONN023_0,
                                    I_GAIN1_OUT_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_1_OUT_CH2", AFE_CONN023_6,
                                    I_SRC_1_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul3_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN024_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN1_CH1", AFE_CONN024_4,
                                    I_I2SIN1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_3_OUT_CH1", AFE_CONN024_6,
                                    I_SRC_3_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul3_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN025_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN1_CH2", AFE_CONN025_4,
                                    I_I2SIN1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_3_OUT_CH2", AFE_CONN025_6,
                                    I_SRC_3_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul4_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN026_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH1", AFE_CONN026_1,
                                    I_DL0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL1_CH1", AFE_CONN026_1,
                                    I_DL1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL6_CH1", AFE_CONN026_1,
                                    I_DL6_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH1", AFE_CONN026_1,
                                    I_DL2_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL3_CH1", AFE_CONN026_1,
                                    I_DL3_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL_24CH_CH1", AFE_CONN026_1,
                                    I_DL_24CH_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH1", AFE_CONN026_4,
                                    I_PCM_0_CAP_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_GAIN0_OUT_CH1", AFE_CONN026_0,
                                    I_GAIN0_OUT_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_3_OUT_CH1", AFE_CONN026_6,
                                    I_SRC_3_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul4_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN027_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH2", AFE_CONN027_1,
                                    I_DL0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL1_CH2", AFE_CONN027_1,
                                    I_DL1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL6_CH2", AFE_CONN027_1,
                                    I_DL6_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH2", AFE_CONN027_1,
                                    I_DL2_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL3_CH2", AFE_CONN027_1,
                                    I_DL3_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL_24CH_CH2", AFE_CONN027_1,
                                    I_DL_24CH_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH1", AFE_CONN027_4,
                                    I_PCM_0_CAP_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH2", AFE_CONN027_4,
                                    I_PCM_0_CAP_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_GAIN0_OUT_CH2", AFE_CONN027_0,
                                    I_GAIN0_OUT_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_3_OUT_CH2", AFE_CONN027_6,
                                    I_SRC_3_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul5_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN028_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH1", AFE_CONN028_1,
                                    I_DL0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL1_CH1", AFE_CONN028_1,
                                    I_DL1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL6_CH1", AFE_CONN028_1,
                                    I_DL6_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH1", AFE_CONN028_1,
                                    I_DL2_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL3_CH1", AFE_CONN028_1,
                                    I_DL3_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL_24CH_CH1", AFE_CONN028_1,
                                    I_DL_24CH_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("GAIN0_OUT_CH1", AFE_CONN028_0,
                                    I_GAIN0_OUT_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_3_OUT_CH1", AFE_CONN028_6,
                                    I_SRC_3_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul5_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN029_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH2", AFE_CONN029_1,
                                    I_DL0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL1_CH2", AFE_CONN029_1,
                                    I_DL1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL6_CH2", AFE_CONN029_1,
                                    I_DL6_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH2", AFE_CONN029_1,
                                    I_DL2_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL3_CH2", AFE_CONN029_1,
                                    I_DL3_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL_24CH_CH2", AFE_CONN029_1,
                                    I_DL_24CH_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH1", AFE_CONN029_4,
                                    I_PCM_0_CAP_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH2", AFE_CONN029_4,
                                    I_PCM_0_CAP_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("GAIN0_OUT_CH2", AFE_CONN029_0,
                                    I_GAIN0_OUT_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_3_OUT_CH2", AFE_CONN029_6,
                                    I_SRC_3_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul6_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN030_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH1", AFE_CONN030_0,
                                    I_DMIC0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH1", AFE_CONN030_1,
                                    I_DL1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH1", AFE_CONN030_1,
                                    I_DL2_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN0_CH1", AFE_CONN030_4,
                                    I_I2SIN0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_4_OUT_CH1", AFE_CONN030_6,
                                    I_SRC_4_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul6_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN031_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH2", AFE_CONN031_0,
                                    I_DMIC0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH2", AFE_CONN031_1,
                                    I_DL1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH2", AFE_CONN031_1,
                                    I_DL2_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN0_CH2", AFE_CONN031_4,
                                    I_I2SIN0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_4_OUT_CH2", AFE_CONN031_6,
                                    I_SRC_4_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul7_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN032_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN032_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH3", AFE_CONN032_0,
                                    I_DMIC1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH1", AFE_CONN032_1,
                                    I_DL1_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH1", AFE_CONN032_1,
                                    I_DL2_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN0_CH1", AFE_CONN032_4,
                                    I_I2SIN0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_4_OUT_CH1", AFE_CONN032_6,
                                    I_SRC_4_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul7_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN033_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN033_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH4", AFE_CONN033_0,
                                    I_DMIC1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL0_CH2", AFE_CONN033_1,
                                    I_DL1_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("DL2_CH2", AFE_CONN033_1,
                                    I_DL2_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("I2SIN0_CH2", AFE_CONN033_4,
                                    I_I2SIN0_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_4_OUT_CH2", AFE_CONN033_6,
                                    I_SRC_4_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul8_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN034_0,
                                    I_ADDA_UL_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul8_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN035_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH1", AFE_CONN035_4,
                                    I_PCM_0_CAP_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("PCM_0_CAP_CH2", AFE_CONN035_4,
                                    I_PCM_0_CAP_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul9_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN036_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN036_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN036_0,
                                    I_ADDA_UL_CH3, 1, 0),
};

static const struct snd_kcontrol_new memif_ul9_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN037_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN037_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN037_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN037_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul24_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN066_0,
                                    I_ADDA_UL_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul24_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN067_0,
                                    I_ADDA_UL_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm0_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN040_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN040_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN040_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN040_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH1", AFE_CONN040_0,
                                    I_DMIC0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_GAIN1_OUT_CH1", AFE_CONN040_0,
                                    I_GAIN1_OUT_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_0_OUT_CH1", AFE_CONN040_6,
                                    I_SRC_0_OUT_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_1_OUT_CH1", AFE_CONN040_6,
                                    I_SRC_1_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm0_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN041_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN041_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN041_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN041_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH2", AFE_CONN041_0,
                                    I_DMIC0_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_GAIN1_OUT_CH2", AFE_CONN041_0,
                                    I_GAIN1_OUT_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_0_OUT_CH2", AFE_CONN041_6,
                                    I_SRC_0_OUT_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_1_OUT_CH2", AFE_CONN041_6,
                                    I_SRC_1_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm0_ch3_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN042_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN042_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN042_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN042_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH3", AFE_CONN042_0,
                                    I_DMIC1_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm0_ch4_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN043_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN043_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN043_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN043_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("AP_DMIC_UL_CH4", AFE_CONN043_0,
                                    I_DMIC1_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm0_ch5_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN044_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN044_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN044_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN044_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm0_ch6_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN045_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN045_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN045_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN045_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm0_ch7_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN046_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN046_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN046_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN046_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm0_ch8_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN047_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN047_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN047_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN047_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch1_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN048_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN048_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN048_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN048_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH5", AFE_CONN048_0,
                                    I_ADDA_UL_CH5, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH6", AFE_CONN048_0,
                                    I_ADDA_UL_CH6, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_0_OUT_CH1", AFE_CONN048_6,
                                    I_SRC_0_OUT_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_3_OUT_CH1", AFE_CONN048_6,
                                    I_SRC_3_OUT_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_4_OUT_CH1", AFE_CONN048_6,
                                    I_SRC_4_OUT_CH1, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch2_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN049_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN049_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN049_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN049_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH5", AFE_CONN049_0,
                                    I_ADDA_UL_CH5, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH6", AFE_CONN049_0,
                                    I_ADDA_UL_CH6, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_0_OUT_CH2", AFE_CONN049_6,
                                    I_SRC_0_OUT_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_3_OUT_CH2", AFE_CONN049_6,
                                    I_SRC_3_OUT_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("HW_SRC_4_OUT_CH2", AFE_CONN049_6,
                                    I_SRC_4_OUT_CH2, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch3_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN050_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN050_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN050_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN050_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH5", AFE_CONN050_0,
                                    I_ADDA_UL_CH5, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH6", AFE_CONN050_0,
                                    I_ADDA_UL_CH6, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch4_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN051_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN051_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN051_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN051_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH5", AFE_CONN051_0,
                                    I_ADDA_UL_CH5, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH6", AFE_CONN051_0,
                                    I_ADDA_UL_CH6, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch5_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN052_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN052_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN052_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN052_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH5", AFE_CONN052_0,
                                    I_ADDA_UL_CH5, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH6", AFE_CONN052_0,
                                    I_ADDA_UL_CH6, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch6_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN053_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN053_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN053_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN053_0,
                                    I_ADDA_UL_CH4, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH5", AFE_CONN053_0,
                                    I_ADDA_UL_CH5, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH6", AFE_CONN053_0,
                                    I_ADDA_UL_CH6, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch7_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN054_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN054_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN054_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN054_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch8_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN055_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN055_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN055_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN055_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch9_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN056_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN056_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN056_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN056_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch10_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN057_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN057_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN057_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN057_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch11_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN058_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN058_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN058_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN058_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch12_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN059_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN059_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN059_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN059_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch13_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN060_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN060_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN060_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN060_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch14_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN061_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN061_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN061_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN061_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch15_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN062_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN062_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN062_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN062_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const struct snd_kcontrol_new memif_ul_cm1_ch16_mix[] = {
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH1", AFE_CONN063_0,
                                    I_ADDA_UL_CH1, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH2", AFE_CONN063_0,
                                    I_ADDA_UL_CH2, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH3", AFE_CONN063_0,
                                    I_ADDA_UL_CH3, 1, 0),
        SOC_DAPM_SINGLE_AUTODISABLE("ADDA_UL_CH4", AFE_CONN063_0,
                                    I_ADDA_UL_CH4, 1, 0),
};

static const char * const cm0_mux_texts[] = {
        "CM0_2CH_PATH",
        "CM0_8CH_PATH",
};

static const char * const cm1_mux_map_texts[] = {
        "CM1_2CH_PATH",
        "CM1_16CH_PATH",
};

static SOC_ENUM_SINGLE_DECL(ul_cm0_mux_map_enum,
                            AFE_CM0_CON0,
                            AFE_CM0_OUTPUT_MUX_SFT,
                            cm0_mux_texts);
static SOC_ENUM_SINGLE_DECL(ul_cm1_mux_map_enum,
                            AFE_CM1_CON0,
                            AFE_CM1_OUTPUT_MUX_SFT,
                            cm1_mux_map_texts);

static const struct snd_kcontrol_new ul_cm0_mux_control =
        SOC_DAPM_ENUM("CM0_UL_MUX Route", ul_cm0_mux_map_enum);
static const struct snd_kcontrol_new ul_cm1_mux_control =
        SOC_DAPM_ENUM("CM1_UL_MUX Route", ul_cm1_mux_map_enum);

static const struct snd_soc_dapm_widget mt8189_memif_widgets[] = {
        /* inter-connections */
        SND_SOC_DAPM_MIXER("UL0_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul0_ch1_mix, ARRAY_SIZE(memif_ul0_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL0_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul0_ch2_mix, ARRAY_SIZE(memif_ul0_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL1_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul1_ch1_mix, ARRAY_SIZE(memif_ul1_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL1_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul1_ch2_mix, ARRAY_SIZE(memif_ul1_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL2_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul2_ch1_mix, ARRAY_SIZE(memif_ul2_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL2_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul2_ch2_mix, ARRAY_SIZE(memif_ul2_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL3_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul3_ch1_mix, ARRAY_SIZE(memif_ul3_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL3_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul3_ch2_mix, ARRAY_SIZE(memif_ul3_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL4_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul4_ch1_mix, ARRAY_SIZE(memif_ul4_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL4_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul4_ch2_mix, ARRAY_SIZE(memif_ul4_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL5_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul5_ch1_mix, ARRAY_SIZE(memif_ul5_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL5_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul5_ch2_mix, ARRAY_SIZE(memif_ul5_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL6_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul6_ch1_mix, ARRAY_SIZE(memif_ul6_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL6_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul6_ch2_mix, ARRAY_SIZE(memif_ul6_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL7_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul7_ch1_mix, ARRAY_SIZE(memif_ul7_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL7_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul7_ch2_mix, ARRAY_SIZE(memif_ul7_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL8_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul8_ch1_mix, ARRAY_SIZE(memif_ul8_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL8_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul8_ch2_mix, ARRAY_SIZE(memif_ul8_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL9_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul9_ch1_mix, ARRAY_SIZE(memif_ul9_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL9_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul9_ch2_mix, ARRAY_SIZE(memif_ul9_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL24_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul24_ch1_mix, ARRAY_SIZE(memif_ul24_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL24_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul24_ch2_mix, ARRAY_SIZE(memif_ul24_ch2_mix)),

        SND_SOC_DAPM_MIXER("UL_CM0_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm0_ch1_mix, ARRAY_SIZE(memif_ul_cm0_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL_CM0_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm0_ch2_mix, ARRAY_SIZE(memif_ul_cm0_ch2_mix)),
        SND_SOC_DAPM_MIXER("UL_CM0_CH3", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm0_ch3_mix, ARRAY_SIZE(memif_ul_cm0_ch3_mix)),
        SND_SOC_DAPM_MIXER("UL_CM0_CH4", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm0_ch4_mix, ARRAY_SIZE(memif_ul_cm0_ch4_mix)),
        SND_SOC_DAPM_MIXER("UL_CM0_CH5", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm0_ch5_mix, ARRAY_SIZE(memif_ul_cm0_ch5_mix)),
        SND_SOC_DAPM_MIXER("UL_CM0_CH6", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm0_ch6_mix, ARRAY_SIZE(memif_ul_cm0_ch6_mix)),
        SND_SOC_DAPM_MIXER("UL_CM0_CH7", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm0_ch7_mix, ARRAY_SIZE(memif_ul_cm0_ch7_mix)),
        SND_SOC_DAPM_MIXER("UL_CM0_CH8", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm0_ch8_mix, ARRAY_SIZE(memif_ul_cm0_ch8_mix)),
        SND_SOC_DAPM_MUX_E("CM0_UL_MUX", SND_SOC_NOPM, 0, 0,
                           &ul_cm0_mux_control,
                           ul_cm0_event,
                           SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_PRE_PMD),

        SND_SOC_DAPM_MIXER("UL_CM1_CH1", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch1_mix, ARRAY_SIZE(memif_ul_cm1_ch1_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH2", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch2_mix, ARRAY_SIZE(memif_ul_cm1_ch2_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH3", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch3_mix, ARRAY_SIZE(memif_ul_cm1_ch3_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH4", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch4_mix, ARRAY_SIZE(memif_ul_cm1_ch4_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH5", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch5_mix, ARRAY_SIZE(memif_ul_cm1_ch5_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH6", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch6_mix, ARRAY_SIZE(memif_ul_cm1_ch6_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH7", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch7_mix, ARRAY_SIZE(memif_ul_cm1_ch7_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH8", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch8_mix, ARRAY_SIZE(memif_ul_cm1_ch8_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH9", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch9_mix, ARRAY_SIZE(memif_ul_cm1_ch9_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH10", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch10_mix, ARRAY_SIZE(memif_ul_cm1_ch10_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH11", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch11_mix, ARRAY_SIZE(memif_ul_cm1_ch11_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH12", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch12_mix, ARRAY_SIZE(memif_ul_cm1_ch12_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH13", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch13_mix, ARRAY_SIZE(memif_ul_cm1_ch13_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH14", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch14_mix, ARRAY_SIZE(memif_ul_cm1_ch14_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH15", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch15_mix, ARRAY_SIZE(memif_ul_cm1_ch15_mix)),
        SND_SOC_DAPM_MIXER("UL_CM1_CH16", SND_SOC_NOPM, 0, 0,
                           memif_ul_cm1_ch16_mix, ARRAY_SIZE(memif_ul_cm1_ch16_mix)),
        SND_SOC_DAPM_MUX("CM1_UL_MUX", SND_SOC_NOPM, 0, 0,
                         &ul_cm1_mux_control),
        SND_SOC_DAPM_SUPPLY("CM0_Enable",
                            AFE_CM0_CON0, AFE_CM0_ON_SFT, 0,
                            ul_cm0_event,
                            SND_SOC_DAPM_PRE_PMU |
                            SND_SOC_DAPM_PRE_PMD),

        SND_SOC_DAPM_SUPPLY("CM1_Enable",
                            AFE_CM1_CON0, AFE_CM0_ON_SFT, 0,
                            ul_cm1_event,
                            SND_SOC_DAPM_PRE_PMU |
                            SND_SOC_DAPM_PRE_PMD),

        /* dynamic pinctrl */
        SND_SOC_DAPM_PINCTRL("I2S0_PIN", "aud-gpio-i2s0-on", "aud-gpio-i2s0-off"),
        SND_SOC_DAPM_PINCTRL("I2S1_PIN", "aud-gpio-i2s1-on", "aud-gpio-i2s1-off"),
        SND_SOC_DAPM_PINCTRL("PCM0_PIN", "aud-gpio-pcm-on", "aud-gpio-pcm-off"),
        SND_SOC_DAPM_PINCTRL("AP_DMIC0_PIN", "aud-gpio-ap-dmic-on", "aud-gpio-ap-dmic-off"),
        SND_SOC_DAPM_PINCTRL("AP_DMIC1_PIN", "aud-gpio-ap-dmic1-on", "aud-gpio-ap-dmic1-off"),
};

static const struct snd_soc_dapm_route mt8189_memif_routes[] = {
        {"UL0", NULL, "UL0_CH1"},
        {"UL0", NULL, "UL0_CH2"},
        /* Normal record */
        {"UL0_CH1", "ADDA_UL_CH1", "ADDA_UL_Mux"},
        {"UL0_CH1", "ADDA_UL_CH2", "ADDA_UL_Mux"},
        {"UL0_CH1", "ADDA_UL_CH3", "ADDA_CH34_UL_Mux"},
        {"UL0_CH1", "ADDA_UL_CH4", "ADDA_CH34_UL_Mux"},
        {"UL0_CH2", "ADDA_UL_CH1", "ADDA_UL_Mux"},
        {"UL0_CH2", "ADDA_UL_CH2", "ADDA_UL_Mux"},
        {"UL0_CH2", "ADDA_UL_CH3", "ADDA_CH34_UL_Mux"},
        {"UL0_CH2", "ADDA_UL_CH4", "ADDA_CH34_UL_Mux"},

        {"UL0_CH1", "AP_DMIC_UL_CH1", "AP DMIC Capture"},
        {"UL0_CH1", "AP_DMIC_UL_CH2", "AP DMIC Capture"},
        {"UL0_CH2", "AP_DMIC_UL_CH2", "AP DMIC Capture"},

        {"UL0_CH1", "I2SIN0_CH1", "I2SIN0"},
        {"UL0_CH2", "I2SIN0_CH2", "I2SIN0"},
        {"UL0_CH1", "I2SIN1_CH1", "I2SIN1"},
        {"UL0_CH2", "I2SIN1_CH2", "I2SIN1"},

        {"UL0_CH1", "PCM_0_CAP_CH1", "PCM 0 Capture"},
        {"UL0_CH2", "PCM_0_CAP_CH1", "PCM 0 Capture"},

        {"UL1", NULL, "UL1_CH1"},
        {"UL1", NULL, "UL1_CH2"},

        {"UL1_CH1", "ADDA_UL_CH1", "ADDA_UL_Mux"},
        {"UL1_CH2", "ADDA_UL_CH2", "ADDA_UL_Mux"},

        {"UL1_CH1", "I2SIN0_CH1", "I2SIN0"},
        {"UL1_CH2", "I2SIN0_CH2", "I2SIN0"},
        {"UL1_CH1", "I2SIN1_CH1", "I2SIN1"},
        {"UL1_CH2", "I2SIN1_CH2", "I2SIN1"},

        {"UL1_CH1", "PCM_0_CAP_CH1", "PCM 0 Capture"},
        {"UL1_CH2", "PCM_0_CAP_CH1", "PCM 0 Capture"},

        {"UL2", NULL, "UL2_CH1"},
        {"UL2", NULL, "UL2_CH2"},

        {"UL2_CH1", "ADDA_UL_CH1", "ADDA_UL_Mux"},
        {"UL2_CH1", "ADDA_UL_CH2", "ADDA_UL_Mux"},
        {"UL2_CH1", "ADDA_UL_CH3", "ADDA_CH34_UL_Mux"},
        {"UL2_CH1", "ADDA_UL_CH4", "ADDA_CH34_UL_Mux"},
        {"UL2_CH2", "ADDA_UL_CH1", "ADDA_UL_Mux"},
        {"UL2_CH2", "ADDA_UL_CH2", "ADDA_UL_Mux"},
        {"UL2_CH2", "ADDA_UL_CH3", "ADDA_CH34_UL_Mux"},
        {"UL2_CH2", "ADDA_UL_CH4", "ADDA_CH34_UL_Mux"},

        {"UL2_CH1", "AP_DMIC_UL_CH3", "AP DMIC CH34 Capture"},
        {"UL2_CH2", "AP_DMIC_UL_CH4", "AP DMIC CH34 Capture"},

        {"UL3", NULL, "UL3_CH1"},
        {"UL3", NULL, "UL3_CH2"},

        {"UL3_CH1", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL3_CH2", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL3_CH1", "I2SIN1_CH1", "I2SIN1"},
        {"UL3_CH2", "I2SIN1_CH2", "I2SIN1"},

        {"UL4", NULL, "UL4_CH1"},
        {"UL4", NULL, "UL4_CH2"},
        {"UL4_CH1", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL4_CH2", "ADDA_UL_CH2", "ADDA Capture"},

        {"UL4_CH1", "PCM_0_CAP_CH1", "PCM 0 Capture"},
        {"UL4_CH2", "PCM_0_CAP_CH1", "PCM 0 Capture"},

        {"UL5", NULL, "UL5_CH1"},
        {"UL5", NULL, "UL5_CH2"},

        {"UL5_CH1", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL5_CH2", "ADDA_UL_CH2", "ADDA Capture"},

        {"UL6", NULL, "UL6_CH1"},
        {"UL6", NULL, "UL6_CH2"},
        {"UL6_CH1", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL6_CH2", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL6_CH1", "I2SIN0_CH1", "I2SIN0"},
        {"UL6_CH2", "I2SIN0_CH2", "I2SIN0"},
        {"UL6_CH1", "AP_DMIC_UL_CH1", "AP DMIC Capture"},
        {"UL6_CH2", "AP_DMIC_UL_CH2", "AP DMIC Capture"},

        {"UL7", NULL, "UL7_CH1"},
        {"UL7", NULL, "UL7_CH2"},
        {"UL7_CH1", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL7_CH1", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL7_CH2", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL7_CH2", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL7_CH1", "I2SIN0_CH1", "I2SIN0"},
        {"UL7_CH2", "I2SIN0_CH2", "I2SIN0"},
        {"UL7_CH1", "AP_DMIC_UL_CH3", "AP DMIC CH34 Capture"},
        {"UL7_CH2", "AP_DMIC_UL_CH4", "AP DMIC CH34 Capture"},

        {"UL8", NULL, "CM0_UL_MUX"},
        {"CM0_UL_MUX", "CM0_2CH_PATH", "UL8_CH1"},
        {"CM0_UL_MUX", "CM0_2CH_PATH", "UL8_CH2"},
        {"CM0_UL_MUX", "CM0_8CH_PATH", "UL_CM0_CH1"},
        {"CM0_UL_MUX", "CM0_8CH_PATH", "UL_CM0_CH2"},
        {"CM0_UL_MUX", "CM0_8CH_PATH", "UL_CM0_CH3"},
        {"CM0_UL_MUX", "CM0_8CH_PATH", "UL_CM0_CH4"},
        {"CM0_UL_MUX", "CM0_8CH_PATH", "UL_CM0_CH5"},
        {"CM0_UL_MUX", "CM0_8CH_PATH", "UL_CM0_CH6"},
        {"CM0_UL_MUX", "CM0_8CH_PATH", "UL_CM0_CH7"},
        {"CM0_UL_MUX", "CM0_8CH_PATH", "UL_CM0_CH8"},

        {"UL_CM0", NULL, "CM0_Enable"},

        {"UL9", NULL, "CM1_UL_MUX"},
        {"CM1_UL_MUX", "CM1_2CH_PATH", "UL9_CH1"},
        {"CM1_UL_MUX", "CM1_2CH_PATH", "UL9_CH2"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH1"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH2"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH3"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH4"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH5"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH6"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH7"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH8"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH9"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH10"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH11"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH12"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH13"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH14"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH15"},
        {"CM1_UL_MUX", "CM1_16CH_PATH", "UL_CM1_CH16"},

        {"UL_CM1", NULL, "CM1_Enable"},

        /* UL9 o36o37 <- ADDA */
        {"UL9_CH1", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL9_CH1", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL9_CH2", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL9_CH2", "ADDA_UL_CH2", "ADDA Capture"},

        {"UL24", NULL, "UL24_CH1"},
        {"UL24", NULL, "UL24_CH2"},
        {"UL24_CH1", "ADDA_UL_CH1", "ADDA Capture"},

        {"UL_CM0", NULL, "UL_CM0_CH1"},
        {"UL_CM0", NULL, "UL_CM0_CH2"},
        {"UL_CM0", NULL, "UL_CM0_CH3"},
        {"UL_CM0", NULL, "UL_CM0_CH4"},
        {"UL_CM0", NULL, "UL_CM0_CH5"},
        {"UL_CM0", NULL, "UL_CM0_CH6"},
        {"UL_CM0", NULL, "UL_CM0_CH7"},
        {"UL_CM0", NULL, "UL_CM0_CH8"},
        {"UL_CM0_CH1", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM0_CH1", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL_CM0_CH2", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM0_CH2", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL_CM0_CH3", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM0_CH3", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL_CM0_CH4", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM0_CH4", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL_CM0_CH1", "AP_DMIC_UL_CH1", "AP DMIC Capture"},
        {"UL_CM0_CH2", "AP_DMIC_UL_CH2", "AP DMIC Capture"},
        {"UL_CM0_CH3", "AP_DMIC_UL_CH3", "AP DMIC CH34 Capture"},
        {"UL_CM0_CH4", "AP_DMIC_UL_CH4", "AP DMIC CH34 Capture"},

        {"UL_CM1", NULL, "UL_CM1_CH1"},
        {"UL_CM1", NULL, "UL_CM1_CH2"},
        {"UL_CM1", NULL, "UL_CM1_CH3"},
        {"UL_CM1", NULL, "UL_CM1_CH4"},
        {"UL_CM1", NULL, "UL_CM1_CH5"},
        {"UL_CM1", NULL, "UL_CM1_CH6"},
        {"UL_CM1", NULL, "UL_CM1_CH7"},
        {"UL_CM1", NULL, "UL_CM1_CH8"},
        {"UL_CM1", NULL, "UL_CM1_CH9"},
        {"UL_CM1", NULL, "UL_CM1_CH10"},
        {"UL_CM1", NULL, "UL_CM1_CH11"},
        {"UL_CM1", NULL, "UL_CM1_CH12"},
        {"UL_CM1", NULL, "UL_CM1_CH13"},
        {"UL_CM1", NULL, "UL_CM1_CH14"},
        {"UL_CM1", NULL, "UL_CM1_CH15"},
        {"UL_CM1", NULL, "UL_CM1_CH16"},
        {"UL_CM1_CH1", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM1_CH1", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL_CM1_CH2", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM1_CH2", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL_CM1_CH3", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM1_CH3", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL_CM1_CH4", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM1_CH4", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL_CM1_CH5", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM1_CH5", "ADDA_UL_CH2", "ADDA Capture"},
        {"UL_CM1_CH6", "ADDA_UL_CH1", "ADDA Capture"},
        {"UL_CM1_CH6", "ADDA_UL_CH2", "ADDA Capture"},

        /* Audio Pin */
        {"I2SOUT0", NULL, "I2S0_PIN"},
        {"I2SIN0", NULL, "I2S0_PIN"},
        {"I2SOUT1", NULL, "I2S1_PIN"},
        {"I2SIN1", NULL, "I2S1_PIN"},
        {"PCM 0 Playback", NULL, "PCM0_PIN"},
        {"AP DMIC Capture", NULL, "AP_DMIC0_PIN"},
        {"AP DMIC CH34 Capture", NULL, "AP_DMIC1_PIN"},
};

#define MT8189_DL_MEMIF(_id) \
        [MT8189_MEMIF_##_id] = { \
                .name = #_id, \
                .id = MT8189_MEMIF_##_id, \
                .reg_ofs_base = AFE_##_id##_BASE, \
                .reg_ofs_cur = AFE_##_id##_CUR, \
                .reg_ofs_end = AFE_##_id##_END, \
                .reg_ofs_base_msb = AFE_##_id##_BASE_MSB, \
                .reg_ofs_cur_msb = AFE_##_id##_CUR_MSB, \
                .reg_ofs_end_msb = AFE_##_id##_END_MSB, \
                .fs_reg = AFE_##_id##_CON0, \
                .fs_shift = _id##_SEL_FS_SFT, \
                .fs_maskbit = _id##_SEL_FS_MASK, \
                .mono_reg = AFE_##_id##_CON0, \
                .mono_shift = _id##_MONO_SFT, \
                .enable_reg = AFE_##_id##_CON0, \
                .enable_shift = _id##_ON_SFT, \
                .hd_reg = AFE_##_id##_CON0, \
                .hd_shift = _id##_HD_MODE_SFT, \
                .hd_align_reg = AFE_##_id##_CON0, \
                .hd_align_mshift = _id##_HALIGN_SFT, \
                .agent_disable_reg = -1, \
                .agent_disable_shift = -1, \
                .msb_reg = -1, \
                .msb_shift = -1, \
                .pbuf_reg = AFE_##_id##_CON0, \
                .pbuf_mask = _id##_PBUF_SIZE_MASK, \
                .pbuf_shift = _id##_PBUF_SIZE_SFT, \
                .minlen_reg = AFE_##_id##_CON0, \
                .minlen_mask = _id##_MINLEN_MASK, \
                .minlen_shift = _id##_MINLEN_SFT, \
}

#define MT8189_MULTI_DL_MEMIF(_id) \
        [MT8189_MEMIF_##_id] = { \
                .name = #_id, \
                .id = MT8189_MEMIF_##_id, \
                .reg_ofs_base = AFE_##_id##_BASE, \
                .reg_ofs_cur = AFE_##_id##_CUR, \
                .reg_ofs_end = AFE_##_id##_END, \
                .reg_ofs_base_msb = AFE_##_id##_BASE_MSB, \
                .reg_ofs_cur_msb = AFE_##_id##_CUR_MSB, \
                .reg_ofs_end_msb = AFE_##_id##_END_MSB, \
                .fs_reg = AFE_##_id##_CON0, \
                .fs_shift = _id##_SEL_FS_SFT, \
                .fs_maskbit = _id##_SEL_FS_MASK, \
                .mono_reg = -1, \
                .mono_shift = -1, \
                .enable_reg = AFE_##_id##_CON0, \
                .enable_shift = _id##_ON_SFT, \
                .hd_reg = AFE_##_id##_CON0, \
                .hd_shift = _id##_HD_MODE_SFT, \
                .hd_align_reg = AFE_##_id##_CON0, \
                .hd_align_mshift = _id##_HALIGN_SFT, \
                .agent_disable_reg = -1, \
                .agent_disable_shift = -1, \
                .msb_reg = -1, \
                .msb_shift = -1, \
                .pbuf_reg = AFE_##_id##_CON0, \
                .pbuf_mask = _id##_PBUF_SIZE_MASK, \
                .pbuf_shift = _id##_PBUF_SIZE_SFT, \
                .minlen_reg = AFE_##_id##_CON0, \
                .minlen_mask = _id##_MINLEN_MASK, \
                .minlen_shift = _id##_MINLEN_SFT, \
                .ch_num_reg = AFE_##_id##_CON0, \
                .ch_num_maskbit = _id##_NUM_MASK, \
                .ch_num_shift = _id##_NUM_SFT, \
}

#define MT8189_UL_MEMIF(_id, _fs_shift, _fs_maskbit, _mono_shift) \
        [MT8189_MEMIF_##_id] = { \
                .name = #_id, \
                .id = MT8189_MEMIF_##_id, \
                .reg_ofs_base = AFE_##_id##_BASE, \
                .reg_ofs_cur = AFE_##_id##_CUR, \
                .reg_ofs_end = AFE_##_id##_END, \
                .reg_ofs_base_msb = AFE_##_id##_BASE_MSB, \
                .reg_ofs_cur_msb = AFE_##_id##_CUR_MSB, \
                .reg_ofs_end_msb = AFE_##_id##_END_MSB, \
                .fs_reg = AFE_##_id##_CON0, \
                .fs_shift = _fs_shift, \
                .fs_maskbit = _fs_maskbit, \
                .mono_reg = AFE_##_id##_CON0, \
                .mono_shift = _mono_shift, \
                .enable_reg = AFE_##_id##_CON0, \
                .enable_shift = _id##_ON_SFT, \
                .hd_reg = AFE_##_id##_CON0, \
                .hd_shift = _id##_HD_MODE_SFT, \
                .hd_align_reg = AFE_##_id##_CON0, \
                .hd_align_mshift = _id##_HALIGN_SFT, \
                .agent_disable_reg = -1, \
                .agent_disable_shift = -1, \
                .msb_reg = -1, \
                .msb_shift = -1, \
        }

/* For convenience with macros: missing register fields */
#define HDMI_SEL_FS_SFT                 -1
#define HDMI_SEL_FS_MASK                -1

/* For convenience with macros: register name differences */
#define AFE_HDMI_BASE                   AFE_HDMI_OUT_BASE
#define AFE_HDMI_CUR                    AFE_HDMI_OUT_CUR
#define AFE_HDMI_END                    AFE_HDMI_OUT_END
#define AFE_HDMI_BASE_MSB               AFE_HDMI_OUT_BASE_MSB
#define AFE_HDMI_CUR_MSB                AFE_HDMI_OUT_CUR_MSB
#define AFE_HDMI_END_MSB                AFE_HDMI_OUT_END_MSB
#define AFE_HDMI_CON0                   AFE_HDMI_OUT_CON0
#define HDMI_ON_SFT                     HDMI_OUT_ON_SFT
#define HDMI_HD_MODE_SFT                HDMI_OUT_HD_MODE_SFT
#define HDMI_HALIGN_SFT                 HDMI_OUT_HALIGN_SFT
#define HDMI_PBUF_SIZE_MASK             HDMI_OUT_PBUF_SIZE_MASK
#define HDMI_PBUF_SIZE_SFT              HDMI_OUT_PBUF_SIZE_SFT
#define HDMI_MINLEN_MASK                HDMI_OUT_MINLEN_MASK
#define HDMI_MINLEN_SFT                 HDMI_OUT_MINLEN_SFT
#define HDMI_NUM_MASK                   HDMI_CH_NUM_MASK
#define HDMI_NUM_SFT                    HDMI_CH_NUM_SFT

static const struct mtk_base_memif_data memif_data[MT8189_MEMIF_NUM] = {
        MT8189_DL_MEMIF(DL0),
        MT8189_DL_MEMIF(DL1),
        MT8189_DL_MEMIF(DL2),
        MT8189_DL_MEMIF(DL3),
        MT8189_DL_MEMIF(DL4),
        MT8189_DL_MEMIF(DL5),
        MT8189_DL_MEMIF(DL6),
        MT8189_DL_MEMIF(DL7),
        MT8189_DL_MEMIF(DL8),
        MT8189_DL_MEMIF(DL23),
        MT8189_DL_MEMIF(DL24),
        MT8189_DL_MEMIF(DL25),
        MT8189_MULTI_DL_MEMIF(DL_24CH),
        MT8189_MULTI_DL_MEMIF(HDMI),
        MT8189_UL_MEMIF(VUL0, VUL0_SEL_FS_SFT, VUL0_SEL_FS_MASK, VUL0_MONO_SFT),
        MT8189_UL_MEMIF(VUL1, VUL1_SEL_FS_SFT, VUL1_SEL_FS_MASK, VUL1_MONO_SFT),
        MT8189_UL_MEMIF(VUL2, VUL2_SEL_FS_SFT, VUL2_SEL_FS_MASK, VUL2_MONO_SFT),
        MT8189_UL_MEMIF(VUL3, VUL3_SEL_FS_SFT, VUL3_SEL_FS_MASK, VUL3_MONO_SFT),
        MT8189_UL_MEMIF(VUL4, VUL4_SEL_FS_SFT, VUL4_SEL_FS_MASK, VUL4_MONO_SFT),
        MT8189_UL_MEMIF(VUL5, VUL5_SEL_FS_SFT, VUL5_SEL_FS_MASK, VUL5_MONO_SFT),
        MT8189_UL_MEMIF(VUL6, VUL6_SEL_FS_SFT, VUL6_SEL_FS_MASK, VUL6_MONO_SFT),
        MT8189_UL_MEMIF(VUL7, VUL7_SEL_FS_SFT, VUL7_SEL_FS_MASK, VUL7_MONO_SFT),
        MT8189_UL_MEMIF(VUL8, VUL8_SEL_FS_SFT, VUL8_SEL_FS_MASK, VUL8_MONO_SFT),
        MT8189_UL_MEMIF(VUL9, VUL9_SEL_FS_SFT, VUL9_SEL_FS_MASK, VUL9_MONO_SFT),
        MT8189_UL_MEMIF(VUL10, VUL10_SEL_FS_SFT, VUL10_SEL_FS_MASK, VUL10_MONO_SFT),
        MT8189_UL_MEMIF(VUL24, VUL24_SEL_FS_SFT, VUL24_SEL_FS_MASK, VUL24_MONO_SFT),
        MT8189_UL_MEMIF(VUL25, VUL25_SEL_FS_SFT, VUL25_SEL_FS_MASK, VUL25_MONO_SFT),
        MT8189_UL_MEMIF(VUL_CM0, -1, -1, -1),
        MT8189_UL_MEMIF(VUL_CM1, -1, -1, -1),
        MT8189_UL_MEMIF(ETDM_IN0, REG_FS_TIMING_SEL_SFT, REG_FS_TIMING_SEL_MASK, -1),
        MT8189_UL_MEMIF(ETDM_IN1, REG_FS_TIMING_SEL_SFT, REG_FS_TIMING_SEL_MASK, -1),
};

#define MT8189_AFE_IRQ(_id)     \
        [MT8189_IRQ_##_id] = { \
                .id = MT8189_IRQ_##_id, \
                .irq_cnt_reg = AFE_IRQ##_id##_MCU_CFG1, \
                .irq_cnt_shift = AFE_IRQ_CNT_SHIFT, \
                .irq_cnt_maskbit = AFE_IRQ_CNT_MASK, \
                .irq_fs_reg = AFE_IRQ##_id##_MCU_CFG0, \
                .irq_fs_shift = AFE_IRQ##_id##_MCU_FS_SFT, \
                .irq_fs_maskbit = AFE_IRQ##_id##_MCU_FS_MASK, \
                .irq_en_reg = AFE_IRQ##_id##_MCU_CFG0, \
                .irq_en_shift = AFE_IRQ##_id##_MCU_ON_SFT, \
                .irq_clr_reg = AFE_IRQ##_id##_MCU_CFG1, \
                .irq_clr_shift = AFE_IRQ##_id##_CLR_CFG_SFT, \
        }

#define MT8189_AFE_TDM_IRQ(_id) \
        [MT8189_IRQ_##_id] = { \
                .id = MT8189_CUS_IRQ_TDM, \
                .irq_cnt_reg = AFE_CUSTOM_IRQ0_MCU_CFG1, \
                .irq_cnt_shift = AFE_CUSTOM_IRQ0_MCU_CNT_SFT, \
                .irq_cnt_maskbit = AFE_CUSTOM_IRQ0_MCU_CNT_MASK, \
                .irq_fs_reg = -1, \
                .irq_fs_shift = -1, \
                .irq_fs_maskbit = -1, \
                .irq_en_reg = AFE_CUSTOM_IRQ0_MCU_CFG0, \
                .irq_en_shift = AFE_CUSTOM_IRQ0_MCU_ON_SFT, \
                .irq_clr_reg = AFE_CUSTOM_IRQ0_MCU_CFG1, \
                .irq_clr_shift = AFE_CUSTOM_IRQ0_CLR_CFG_SFT, \
        }

static const struct mtk_base_irq_data irq_data[MT8189_IRQ_NUM] = {
        MT8189_AFE_IRQ(0),
        MT8189_AFE_IRQ(1),
        MT8189_AFE_IRQ(2),
        MT8189_AFE_IRQ(3),
        MT8189_AFE_IRQ(4),
        MT8189_AFE_IRQ(5),
        MT8189_AFE_IRQ(6),
        MT8189_AFE_IRQ(7),
        MT8189_AFE_IRQ(8),
        MT8189_AFE_IRQ(9),
        MT8189_AFE_IRQ(10),
        MT8189_AFE_IRQ(11),
        MT8189_AFE_IRQ(12),
        MT8189_AFE_IRQ(13),
        MT8189_AFE_IRQ(14),
        MT8189_AFE_IRQ(15),
        MT8189_AFE_IRQ(16),
        MT8189_AFE_IRQ(17),
        MT8189_AFE_IRQ(18),
        MT8189_AFE_IRQ(19),
        MT8189_AFE_IRQ(20),
        MT8189_AFE_IRQ(21),
        MT8189_AFE_IRQ(22),
        MT8189_AFE_IRQ(23),
        MT8189_AFE_IRQ(24),
        MT8189_AFE_IRQ(25),
        MT8189_AFE_IRQ(26),
        MT8189_AFE_TDM_IRQ(31),
};

static const int memif_irq_usage[MT8189_MEMIF_NUM] = {
        /* TODO: verify each memif & irq */
        [MT8189_MEMIF_DL0] = MT8189_IRQ_0,
        [MT8189_MEMIF_DL1] = MT8189_IRQ_1,
        [MT8189_MEMIF_DL2] = MT8189_IRQ_2,
        [MT8189_MEMIF_DL3] = MT8189_IRQ_3,
        [MT8189_MEMIF_DL4] = MT8189_IRQ_4,
        [MT8189_MEMIF_DL5] = MT8189_IRQ_5,
        [MT8189_MEMIF_DL6] = MT8189_IRQ_6,
        [MT8189_MEMIF_DL7] = MT8189_IRQ_7,
        [MT8189_MEMIF_DL8] = MT8189_IRQ_8,
        [MT8189_MEMIF_DL23] = MT8189_IRQ_9,
        [MT8189_MEMIF_DL24] = MT8189_IRQ_10,
        [MT8189_MEMIF_DL25] = MT8189_IRQ_11,
        [MT8189_MEMIF_DL_24CH] = MT8189_IRQ_12,
        [MT8189_MEMIF_VUL0] = MT8189_IRQ_13,
        [MT8189_MEMIF_VUL1] = MT8189_IRQ_14,
        [MT8189_MEMIF_VUL2] = MT8189_IRQ_15,
        [MT8189_MEMIF_VUL3] = MT8189_IRQ_16,
        [MT8189_MEMIF_VUL4] = MT8189_IRQ_17,
        [MT8189_MEMIF_VUL5] = MT8189_IRQ_18,
        [MT8189_MEMIF_VUL6] = MT8189_IRQ_19,
        [MT8189_MEMIF_VUL7] = MT8189_IRQ_20,
        [MT8189_MEMIF_VUL8] = MT8189_IRQ_21,
        [MT8189_MEMIF_VUL9] = MT8189_IRQ_22,
        [MT8189_MEMIF_VUL10] = MT8189_IRQ_23,
        [MT8189_MEMIF_VUL24] = MT8189_IRQ_24,
        [MT8189_MEMIF_VUL25] = MT8189_IRQ_25,
        [MT8189_MEMIF_VUL_CM0] = MT8189_IRQ_26,
        [MT8189_MEMIF_VUL_CM1] = MT8189_IRQ_0,
        [MT8189_MEMIF_ETDM_IN0] = MT8189_IRQ_0,
        [MT8189_MEMIF_ETDM_IN1] = MT8189_IRQ_0,
        [MT8189_MEMIF_HDMI] = MT8189_IRQ_31
};

static bool mt8189_is_volatile_reg(struct device *dev, unsigned int reg)
{
        /* these auto-gen reg has read-only bit, so put it as volatile */
        /* volatile reg cannot be cached, so cannot be set when power off */
        switch (reg) {
        case AUDIO_TOP_CON0:
        case AUDIO_TOP_CON1:
        case AUDIO_TOP_CON2:
        case AUDIO_TOP_CON3:
        case AUDIO_TOP_CON4:
        case AFE_APLL1_TUNER_MON0:
        case AFE_APLL2_TUNER_MON0:
        case AFE_SPM_CONTROL_ACK:
        case AUDIO_TOP_IP_VERSION:
        case AUDIO_ENGEN_CON0_MON:
        case AFE_CONNSYS_I2S_IPM_VER_MON:
        case AFE_CONNSYS_I2S_MON:
        case AFE_PCM_INTF_MON:
        case AFE_PCM_TOP_IP_VERSION:
        case AFE_IRQ_MCU_STATUS:
        case AFE_CUSTOM_IRQ_MCU_STATUS:
        case AFE_IRQ_MCU_MON0:
        case AFE_IRQ_MCU_MON1:
        case AFE_IRQ_MCU_MON2:
        case AFE_IRQ0_CNT_MON:
        case AFE_IRQ1_CNT_MON:
        case AFE_IRQ2_CNT_MON:
        case AFE_IRQ3_CNT_MON:
        case AFE_IRQ4_CNT_MON:
        case AFE_IRQ5_CNT_MON:
        case AFE_IRQ6_CNT_MON:
        case AFE_IRQ7_CNT_MON:
        case AFE_IRQ8_CNT_MON:
        case AFE_IRQ9_CNT_MON:
        case AFE_IRQ10_CNT_MON:
        case AFE_IRQ11_CNT_MON:
        case AFE_IRQ12_CNT_MON:
        case AFE_IRQ13_CNT_MON:
        case AFE_IRQ14_CNT_MON:
        case AFE_IRQ15_CNT_MON:
        case AFE_IRQ16_CNT_MON:
        case AFE_IRQ17_CNT_MON:
        case AFE_IRQ18_CNT_MON:
        case AFE_IRQ19_CNT_MON:
        case AFE_IRQ20_CNT_MON:
        case AFE_IRQ21_CNT_MON:
        case AFE_IRQ22_CNT_MON:
        case AFE_IRQ23_CNT_MON:
        case AFE_IRQ24_CNT_MON:
        case AFE_IRQ25_CNT_MON:
        case AFE_IRQ26_CNT_MON:
        case AFE_CM0_MON:
        case AFE_CM0_IP_VERSION:
        case AFE_CM1_MON:
        case AFE_CM1_IP_VERSION:
        case AFE_ADDA_UL0_SRC_DEBUG_MON0:
        case AFE_ADDA_UL0_SRC_MON0:
        case AFE_ADDA_UL0_SRC_MON1:
        case AFE_ADDA_UL0_IP_VERSION:
        case AFE_ADDA_DMIC0_SRC_DEBUG_MON0:
        case AFE_ADDA_DMIC0_SRC_MON0:
        case AFE_ADDA_DMIC0_SRC_MON1:
        case AFE_ADDA_DMIC0_IP_VERSION:
        case AFE_ADDA_DMIC1_SRC_DEBUG_MON0:
        case AFE_ADDA_DMIC1_SRC_MON0:
        case AFE_ADDA_DMIC1_SRC_MON1:
        case AFE_ADDA_DMIC1_IP_VERSION:
        case AFE_MTKAIF_IPM_VER_MON:
        case AFE_MTKAIF_MON:
        case AFE_AUD_PAD_TOP_MON:
        case AFE_ADDA_MTKAIFV4_MON0:
        case AFE_ADDA_MTKAIFV4_MON1:
        case AFE_ADDA6_MTKAIFV4_MON0:
        case ETDM_IN0_MON:
        case ETDM_IN1_MON:
        case ETDM_OUT0_MON:
        case ETDM_OUT1_MON:
        case ETDM_OUT4_MON:
        case AFE_CONN_MON0:
        case AFE_CONN_MON1:
        case AFE_CONN_MON2:
        case AFE_CONN_MON3:
        case AFE_CONN_MON4:
        case AFE_CONN_MON5:
        case AFE_CBIP_SLV_DECODER_MON0:
        case AFE_CBIP_SLV_DECODER_MON1:
        case AFE_CBIP_SLV_MUX_MON0:
        case AFE_CBIP_SLV_MUX_MON1:
        case AFE_DL0_CUR_MSB:
        case AFE_DL0_CUR:
        case AFE_DL0_RCH_MON:
        case AFE_DL0_LCH_MON:
        case AFE_DL1_CUR_MSB:
        case AFE_DL1_CUR:
        case AFE_DL1_RCH_MON:
        case AFE_DL1_LCH_MON:
        case AFE_DL2_CUR_MSB:
        case AFE_DL2_CUR:
        case AFE_DL2_RCH_MON:
        case AFE_DL2_LCH_MON:
        case AFE_DL3_CUR_MSB:
        case AFE_DL3_CUR:
        case AFE_DL3_RCH_MON:
        case AFE_DL3_LCH_MON:
        case AFE_DL4_CUR_MSB:
        case AFE_DL4_CUR:
        case AFE_DL4_RCH_MON:
        case AFE_DL4_LCH_MON:
        case AFE_DL5_CUR_MSB:
        case AFE_DL5_CUR:
        case AFE_DL5_RCH_MON:
        case AFE_DL5_LCH_MON:
        case AFE_DL6_CUR_MSB:
        case AFE_DL6_CUR:
        case AFE_DL6_RCH_MON:
        case AFE_DL6_LCH_MON:
        case AFE_DL7_CUR_MSB:
        case AFE_DL7_CUR:
        case AFE_DL7_RCH_MON:
        case AFE_DL7_LCH_MON:
        case AFE_DL8_CUR_MSB:
        case AFE_DL8_CUR:
        case AFE_DL8_RCH_MON:
        case AFE_DL8_LCH_MON:
        case AFE_DL_24CH_CUR_MSB:
        case AFE_DL_24CH_CUR:
        case AFE_DL23_CUR_MSB:
        case AFE_DL23_CUR:
        case AFE_DL23_RCH_MON:
        case AFE_DL23_LCH_MON:
        case AFE_DL24_CUR_MSB:
        case AFE_DL24_CUR:
        case AFE_DL24_RCH_MON:
        case AFE_DL24_LCH_MON:
        case AFE_DL25_CUR_MSB:
        case AFE_DL25_CUR:
        case AFE_DL25_RCH_MON:
        case AFE_DL25_LCH_MON:
        case AFE_VUL0_CUR_MSB:
        case AFE_VUL0_CUR:
        case AFE_VUL1_CUR_MSB:
        case AFE_VUL1_CUR:
        case AFE_VUL2_CUR_MSB:
        case AFE_VUL2_CUR:
        case AFE_VUL3_CUR_MSB:
        case AFE_VUL3_CUR:
        case AFE_VUL4_CUR_MSB:
        case AFE_VUL4_CUR:
        case AFE_VUL5_CUR_MSB:
        case AFE_VUL5_CUR:
        case AFE_VUL6_CUR_MSB:
        case AFE_VUL6_CUR:
        case AFE_VUL7_CUR_MSB:
        case AFE_VUL7_CUR:
        case AFE_VUL8_CUR_MSB:
        case AFE_VUL8_CUR:
        case AFE_VUL9_CUR_MSB:
        case AFE_VUL9_CUR:
        case AFE_VUL10_CUR_MSB:
        case AFE_VUL10_CUR:
        case AFE_VUL24_CUR_MSB:
        case AFE_VUL24_CUR:
        case AFE_VUL25_CUR_MSB:
        case AFE_VUL25_CUR:
        case AFE_VUL_CM0_CUR_MSB:
        case AFE_VUL_CM0_CUR:
        case AFE_VUL_CM1_CUR_MSB:
        case AFE_VUL_CM1_CUR:
        case AFE_ETDM_IN0_CUR_MSB:
        case AFE_ETDM_IN0_CUR:
        case AFE_ETDM_IN1_CUR_MSB:
        case AFE_ETDM_IN1_CUR:
        case AFE_HDMI_OUT_CUR_MSB:
        case AFE_HDMI_OUT_CUR:
        case AFE_HDMI_OUT_END:
        case AFE_HDMI_OUT_MON0:
        case AFE_PROT_SIDEBAND0_MON:
        case AFE_PROT_SIDEBAND1_MON:
        case AFE_PROT_SIDEBAND2_MON:
        case AFE_PROT_SIDEBAND3_MON:
        case AFE_DOMAIN_SIDEBAND0_MON:
        case AFE_DOMAIN_SIDEBAND1_MON:
        case AFE_DOMAIN_SIDEBAND2_MON:
        case AFE_DOMAIN_SIDEBAND3_MON:
        case AFE_DOMAIN_SIDEBAND4_MON:
        case AFE_DOMAIN_SIDEBAND5_MON:
        case AFE_DOMAIN_SIDEBAND6_MON:
        case AFE_DOMAIN_SIDEBAND7_MON:
        case AFE_DOMAIN_SIDEBAND8_MON:
        case AFE_DOMAIN_SIDEBAND9_MON:
        case AFE_PCM0_INTF_CON1_MASK_MON:
        case AFE_CONNSYS_I2S_CON_MASK_MON:
        case AFE_MTKAIF0_CFG0_MASK_MON:
        case AFE_MTKAIF1_CFG0_MASK_MON:
        case AFE_ADDA_UL0_SRC_CON0_MASK_MON:
        case AFE_ASRC_NEW_CON0:
        case AFE_ASRC_NEW_CON6:
        case AFE_ASRC_NEW_CON8:
        case AFE_ASRC_NEW_CON9:
        case AFE_ASRC_NEW_CON12:
        case AFE_ASRC_NEW_IP_VERSION:
        case AFE_GASRC0_NEW_CON0:
        case AFE_GASRC0_NEW_CON6:
        case AFE_GASRC0_NEW_CON8:
        case AFE_GASRC0_NEW_CON9:
        case AFE_GASRC0_NEW_CON10:
        case AFE_GASRC0_NEW_CON11:
        case AFE_GASRC0_NEW_CON12:
        case AFE_GASRC0_NEW_IP_VERSION:
        case AFE_GASRC1_NEW_CON0:
        case AFE_GASRC1_NEW_CON6:
        case AFE_GASRC1_NEW_CON8:
        case AFE_GASRC1_NEW_CON9:
        case AFE_GASRC1_NEW_CON12:
        case AFE_GASRC1_NEW_IP_VERSION:
        case AFE_GASRC2_NEW_CON0:
        case AFE_GASRC2_NEW_CON6:
        case AFE_GASRC2_NEW_CON8:
        case AFE_GASRC2_NEW_CON9:
        case AFE_GASRC2_NEW_CON12:
        case AFE_GASRC2_NEW_IP_VERSION:
        case AFE_GAIN0_CUR_L:
        case AFE_GAIN0_CUR_R:
        case AFE_GAIN1_CUR_L:
        case AFE_GAIN1_CUR_R:
        case AFE_GAIN2_CUR_L:
        case AFE_GAIN2_CUR_R:
        case AFE_GAIN3_CUR_L:
        case AFE_GAIN3_CUR_R:
        case AFE_IRQ_MCU_EN:
        case AFE_CUSTOM_IRQ_MCU_EN:
        case AFE_IRQ_MCU_DSP_EN:
        case AFE_IRQ_MCU_DSP2_EN:
        case AFE_DL5_CON0:
        case AFE_DL6_CON0:
        case AFE_DL23_CON0:
        case AFE_DL_24CH_CON0:
        case AFE_VUL1_CON0:
        case AFE_VUL3_CON0:
        case AFE_VUL4_CON0:
        case AFE_VUL5_CON0:
        case AFE_VUL9_CON0:
        case AFE_VUL25_CON0:
        case AFE_IRQ0_MCU_CFG0:
        case AFE_IRQ1_MCU_CFG0:
        case AFE_IRQ2_MCU_CFG0:
        case AFE_IRQ3_MCU_CFG0:
        case AFE_IRQ4_MCU_CFG0:
        case AFE_IRQ5_MCU_CFG0:
        case AFE_IRQ6_MCU_CFG0:
        case AFE_IRQ7_MCU_CFG0:
        case AFE_IRQ8_MCU_CFG0:
        case AFE_IRQ9_MCU_CFG0:
        case AFE_IRQ10_MCU_CFG0:
        case AFE_IRQ11_MCU_CFG0:
        case AFE_IRQ12_MCU_CFG0:
        case AFE_IRQ13_MCU_CFG0:
        case AFE_IRQ14_MCU_CFG0:
        case AFE_IRQ15_MCU_CFG0:
        case AFE_IRQ16_MCU_CFG0:
        case AFE_IRQ17_MCU_CFG0:
        case AFE_IRQ18_MCU_CFG0:
        case AFE_IRQ19_MCU_CFG0:
        case AFE_IRQ20_MCU_CFG0:
        case AFE_IRQ21_MCU_CFG0:
        case AFE_IRQ22_MCU_CFG0:
        case AFE_IRQ23_MCU_CFG0:
        case AFE_IRQ24_MCU_CFG0:
        case AFE_IRQ25_MCU_CFG0:
        case AFE_IRQ26_MCU_CFG0:
        case AFE_CUSTOM_IRQ0_MCU_CFG0:
        case AFE_IRQ0_MCU_CFG1:
        case AFE_IRQ1_MCU_CFG1:
        case AFE_IRQ2_MCU_CFG1:
        case AFE_IRQ3_MCU_CFG1:
        case AFE_IRQ4_MCU_CFG1:
        case AFE_IRQ5_MCU_CFG1:
        case AFE_IRQ6_MCU_CFG1:
        case AFE_IRQ7_MCU_CFG1:
        case AFE_IRQ8_MCU_CFG1:
        case AFE_IRQ9_MCU_CFG1:
        case AFE_IRQ10_MCU_CFG1:
        case AFE_IRQ11_MCU_CFG1:
        case AFE_IRQ12_MCU_CFG1:
        case AFE_IRQ13_MCU_CFG1:
        case AFE_IRQ14_MCU_CFG1:
        case AFE_IRQ15_MCU_CFG1:
        case AFE_IRQ16_MCU_CFG1:
        case AFE_IRQ17_MCU_CFG1:
        case AFE_IRQ18_MCU_CFG1:
        case AFE_IRQ19_MCU_CFG1:
        case AFE_IRQ20_MCU_CFG1:
        case AFE_IRQ21_MCU_CFG1:
        case AFE_IRQ22_MCU_CFG1:
        case AFE_IRQ23_MCU_CFG1:
        case AFE_IRQ24_MCU_CFG1:
        case AFE_IRQ25_MCU_CFG1:
        case AFE_IRQ26_MCU_CFG1:
        case AFE_CUSTOM_IRQ0_MCU_CFG1:
        /* for vow using */
        case AFE_IRQ_MCU_SCP_EN:
        case AFE_VUL_CM0_BASE_MSB:
        case AFE_VUL_CM0_BASE:
        case AFE_VUL_CM0_END_MSB:
        case AFE_VUL_CM0_END:
        case AFE_VUL_CM0_CON0:
                return true;
        default:
                return false;
        };
}

static const struct regmap_config mt8189_afe_regmap_config = {
        .reg_bits = 32,
        .reg_stride = 4,
        .val_bits = 32,

        .volatile_reg = mt8189_is_volatile_reg,

        .max_register = AFE_MAX_REGISTER,
        .num_reg_defaults_raw = AFE_MAX_REGISTER,

        .cache_type = REGCACHE_FLAT,
};

static irqreturn_t mt8189_afe_irq_handler(int irq_id, void *dev)
{
        struct mtk_base_afe *afe = dev;
        struct mtk_base_afe_irq *irq;
        u32 status;
        u32 status_mcu;
        u32 mcu_en;
        u32 cus_status;
        u32 cus_status_mcu;
        u32 cus_mcu_en;
        u32 tmp_reg;
        int ret, cus_ret;
        int i;
        struct timespec64 ts64;
        u64 t1, t2;
        /* one interrupt period = 5ms */
        const u64 timeout_limit = 5000000;

        /* get irq that is sent to MCU */
        regmap_read(afe->regmap, AFE_IRQ_MCU_EN, &mcu_en);
        regmap_read(afe->regmap, AFE_CUSTOM_IRQ_MCU_EN, &cus_mcu_en);

        ret = regmap_read(afe->regmap, AFE_IRQ_MCU_STATUS, &status);
        cus_ret = regmap_read(afe->regmap, AFE_CUSTOM_IRQ_MCU_STATUS, &cus_status);
        /* only care IRQ which is sent to MCU */
        status_mcu = status & mcu_en & AFE_IRQ_STATUS_BITS;
        cus_status_mcu = cus_status & cus_mcu_en & AFE_IRQ_STATUS_BITS;
        if ((ret || status_mcu == 0) && (cus_ret || cus_status_mcu == 0)) {
                dev_err(afe->dev, "%s(), irq status err, ret %d, 0x%x:0x%x:0x%x:0x%x\n",
                        __func__, ret, status, mcu_en, cus_status_mcu, cus_mcu_en);
                return IRQ_NONE;
        }

        ktime_get_ts64(&ts64);
        t1 = ktime_get_ns();

        for (i = 0; i < MT8189_MEMIF_NUM; i++) {
                struct mtk_base_afe_memif *memif = &afe->memif[i];

                if (!memif->substream)
                        continue;

                if (memif->irq_usage < 0)
                        continue;
                irq = &afe->irqs[memif->irq_usage];

                if (i == MT8189_MEMIF_HDMI) {
                        if (cus_status_mcu & BIT(irq->irq_data->id))
                                snd_pcm_period_elapsed(memif->substream);
                } else if (status_mcu & BIT(irq->irq_data->id)) {
                        snd_pcm_period_elapsed(memif->substream);
                }
        }

        ktime_get_ts64(&ts64);
        t2 = ktime_get_ns();
        t2 = t2 - t1; /* in ns (10^9) */

        if (t2 > timeout_limit)
                dev_warn(afe->dev, "IRQ handler exceeded time limit by %llu ns\n",
                         t2 - timeout_limit);

        /* clear irq */
        for (i = 0; i < MT8189_IRQ_NUM; ++i) {
                if (((cus_status_mcu & BIT(irq_data[i].id)) && i == MT8189_IRQ_31) ||
                    ((status_mcu & BIT(irq_data[i].id)) && i != MT8189_IRQ_31)) {
                        regmap_read(afe->regmap, irq_data[i].irq_clr_reg, &tmp_reg);
                        regmap_update_bits(afe->regmap, irq_data[i].irq_clr_reg,
                                           AFE_IRQ_CLR_CFG_MASK_SFT |
                                           AFE_IRQ_MISS_FLAG_CLR_CFG_MASK_SFT,
                                           tmp_reg ^ (AFE_IRQ_CLR_CFG_MASK_SFT |
                                           AFE_IRQ_MISS_FLAG_CLR_CFG_MASK_SFT));
                }
        }

        return IRQ_HANDLED;
}

static int mt8189_afe_runtime_suspend(struct device *dev)
{
        struct mtk_base_afe *afe = dev_get_drvdata(dev);
        unsigned int value;
        unsigned int tmp_reg;
        int ret, i;

        if (!afe->regmap) {
                dev_warn(afe->dev, "%s() skip regmap\n", __func__);
                goto skip_regmap;
        }

        /* disable AFE */
        mt8189_afe_disable_main_clock(afe);

        ret = regmap_read_poll_timeout(afe->regmap,
                                       AUDIO_ENGEN_CON0_MON,
                                       value,
                                       (value & AUDIO_ENGEN_MON_SFT) == 0,
                                       20,
                                       1 * 1000 * 1000);
        dev_dbg(afe->dev, "%s() read_poll ret %d\n", __func__, ret);
        if (ret)
                dev_warn(afe->dev, "%s(), ret %d\n", __func__, ret);

        /* make sure all irq status are cleared */
        for (i = 0; i < MT8189_IRQ_NUM; i++) {
                regmap_read(afe->regmap, irq_data[i].irq_clr_reg, &tmp_reg);
                regmap_update_bits(afe->regmap, irq_data[i].irq_clr_reg,
                                   AFE_IRQ_CLR_CFG_MASK_SFT |
                                   AFE_IRQ_MISS_FLAG_CLR_CFG_MASK_SFT,
                                   tmp_reg ^ (AFE_IRQ_CLR_CFG_MASK_SFT |
                                              AFE_IRQ_MISS_FLAG_CLR_CFG_MASK_SFT));
        }

        /* reset audio 26M request */
        regmap_update_bits(afe->regmap,
                           AFE_SPM_CONTROL_REQ, 0x1, 0x0);

        /* cache only */
        regcache_cache_only(afe->regmap, true);
        regcache_mark_dirty(afe->regmap);

skip_regmap:
        mt8189_afe_disable_reg_rw_clk(afe);
        return 0;
}

static int mt8189_afe_runtime_resume(struct device *dev)
{
        struct mtk_base_afe *afe = dev_get_drvdata(dev);
        int ret;

        ret = mt8189_afe_enable_reg_rw_clk(afe);
        if (ret)
                return ret;

        if (!afe->regmap) {
                dev_warn(afe->dev, "skip regmap\n");
                return 0;
        }

        regcache_cache_only(afe->regmap, false);
        regcache_sync(afe->regmap);

        /* set audio 26M request */
        regmap_update_bits(afe->regmap, AFE_SPM_CONTROL_REQ, 0x1, 0x1);
        regmap_update_bits(afe->regmap, AFE_CBIP_CFG0, 0x1, 0x1);

        /* force cpu use 8_24 format when writing 32bit data */
        regmap_update_bits(afe->regmap, AFE_MEMIF_CON0,
                           CPU_HD_ALIGN_MASK_SFT, 0 << CPU_HD_ALIGN_SFT);

        /* enable AFE */
        mt8189_afe_enable_main_clock(afe);

        return 0;
}

static int mt8189_afe_component_probe(struct snd_soc_component *component)
{
        struct mtk_base_afe *afe = snd_soc_component_get_drvdata(component);

        /* enable clock for regcache get default value from hw */
        pm_runtime_get_sync(afe->dev);
        mtk_afe_add_sub_dai_control(component);
        pm_runtime_put_sync(afe->dev);

        return 0;
}

static int mt8189_afe_pcm_open(struct snd_soc_component *component,
                               struct snd_pcm_substream *substream)
{
        /* set the wait_for_avail to 2 sec*/
        substream->wait_time = msecs_to_jiffies(2 * 1000);

        return 0;
}

static void mt8189_afe_pcm_free(struct snd_soc_component *component,
                                struct snd_pcm *pcm)
{
        snd_pcm_lib_preallocate_free_for_all(pcm);
}

static const struct snd_soc_component_driver mt8189_afe_component = {
        .name = AFE_PCM_NAME,
        .probe = mt8189_afe_component_probe,
        .pcm_construct = mtk_afe_pcm_new,
        .pcm_destruct = mt8189_afe_pcm_free,
        .open = mt8189_afe_pcm_open,
        .pointer = mtk_afe_pcm_pointer,
};

static int mt8189_dai_memif_register(struct mtk_base_afe *afe)
{
        struct mtk_base_afe_dai *dai;

        dai = devm_kzalloc(afe->dev, sizeof(*dai), GFP_KERNEL);
        if (!dai)
                return -ENOMEM;

        list_add(&dai->list, &afe->sub_dais);

        dai->dai_drivers = mt8189_memif_dai_driver;
        dai->num_dai_drivers = ARRAY_SIZE(mt8189_memif_dai_driver);
        dai->dapm_widgets = mt8189_memif_widgets;
        dai->num_dapm_widgets = ARRAY_SIZE(mt8189_memif_widgets);
        dai->dapm_routes = mt8189_memif_routes;
        dai->num_dapm_routes = ARRAY_SIZE(mt8189_memif_routes);

        return 0;
}

typedef int (*dai_register_cb)(struct mtk_base_afe *);
static const dai_register_cb dai_register_cbs[] = {
        mt8189_dai_adda_register,
        mt8189_dai_i2s_register,
        mt8189_dai_pcm_register,
        mt8189_dai_tdm_register,
        mt8189_dai_memif_register,
};

static const struct reg_sequence mt8189_cg_patch[] = {
        { AUDIO_TOP_CON4, 0x361c },
};

static int mt8189_afe_pcm_dev_probe(struct platform_device *pdev)
{
        int ret, i;
        unsigned int tmp_reg;
        int irq_id;
        struct mtk_base_afe *afe;
        struct mt8189_afe_private *afe_priv;
        struct device *dev = &pdev->dev;

        ret = dma_set_mask_and_coherent(dev, DMA_BIT_MASK(34));
        if (ret)
                return ret;

        ret = of_reserved_mem_device_init(dev);
        if (ret)
                dev_warn(dev, "failed to assign memory region: %d\n", ret);

        afe = devm_kzalloc(dev, sizeof(*afe), GFP_KERNEL);
        if (!afe)
                return -ENOMEM;

        platform_set_drvdata(pdev, afe);

        afe->platform_priv = devm_kzalloc(dev, sizeof(*afe_priv),
                                          GFP_KERNEL);
        if (!afe->platform_priv)
                return -ENOMEM;

        afe_priv = afe->platform_priv;
        afe->dev = dev;

        afe->base_addr = devm_platform_ioremap_resource(pdev, 0);
        if (IS_ERR(afe->base_addr))
                return dev_err_probe(dev, PTR_ERR(afe->base_addr),
                                     "AFE base_addr not found\n");

        /* init audio related clock */
        ret = mt8189_init_clock(afe);
        if (ret)
                return dev_err_probe(dev, ret, "init clock error.\n");

        /* init memif */
        /* IPM2.0 no need banding */
        afe->memif_32bit_supported = 1;
        afe->memif_size = MT8189_MEMIF_NUM;
        afe->memif = devm_kcalloc(dev, afe->memif_size, sizeof(*afe->memif),
                                  GFP_KERNEL);

        if (!afe->memif)
                return -ENOMEM;

        for (i = 0; i < afe->memif_size; i++) {
                afe->memif[i].data = &memif_data[i];
                afe->memif[i].irq_usage = memif_irq_usage[i];
                afe->memif[i].const_irq = 1;
        }

        mutex_init(&afe->irq_alloc_lock);

        /* init irq */
        afe->irqs_size = MT8189_IRQ_NUM;
        afe->irqs = devm_kcalloc(dev, afe->irqs_size, sizeof(*afe->irqs),
                                 GFP_KERNEL);

        if (!afe->irqs)
                return -ENOMEM;

        for (i = 0; i < afe->irqs_size; i++)
                afe->irqs[i].irq_data = &irq_data[i];

        /* request irq */
        irq_id = platform_get_irq(pdev, 0);
        if (irq_id < 0)
                return dev_err_probe(dev, irq_id, "no irq found");

        ret = devm_request_irq(dev, irq_id, mt8189_afe_irq_handler,
                               IRQF_TRIGGER_NONE,
                               "Afe_ISR_Handle", afe);
        if (ret)
                return dev_err_probe(dev, ret, "could not request_irq for Afe_ISR_Handle\n");

        /* init sub_dais */
        INIT_LIST_HEAD(&afe->sub_dais);

        for (i = 0; i < ARRAY_SIZE(dai_register_cbs); i++) {
                ret = dai_register_cbs[i](afe);
                if (ret)
                        return dev_err_probe(dev, ret, "dai register i %d fail\n", i);
        }

        /* init dai_driver and component_driver */
        ret = mtk_afe_combine_sub_dai(afe);
        if (ret)
                return dev_err_probe(dev, ret, "mtk_afe_combine_sub_dai fail\n");

        /* others */
        afe->mtk_afe_hardware = &mt8189_afe_hardware;
        afe->memif_fs = mt8189_memif_fs;
        afe->irq_fs = mt8189_irq_fs;
        afe->get_dai_fs = mt8189_get_dai_fs;
        afe->get_memif_pbuf_size = mt8189_get_memif_pbuf_size;

        afe->runtime_resume = mt8189_afe_runtime_resume;
        afe->runtime_suspend = mt8189_afe_runtime_suspend;

        ret = devm_pm_runtime_enable(dev);
        if (ret)
                return ret;

        /*
         * Audio device is part of genpd. Registering it as a syscore device
         * ensure the proper power-on sequence of the AFE device.
         */
        dev_pm_syscore_device(dev, true);

        /* enable clock for regcache get default value from hw */
        pm_runtime_get_sync(dev);

        afe->regmap = devm_regmap_init_mmio(dev, afe->base_addr,
                                            &mt8189_afe_regmap_config);
        if (IS_ERR(afe->regmap))
                return PTR_ERR(afe->regmap);

        ret = regmap_register_patch(afe->regmap, mt8189_cg_patch,
                                    ARRAY_SIZE(mt8189_cg_patch));
        if (ret < 0) {
                dev_err(dev, "Failed to apply cg patch\n");
                goto err_pm_disable;
        }

        regmap_read(afe->regmap, AFE_IRQ_MCU_EN, &tmp_reg);
        regmap_write(afe->regmap, AFE_IRQ_MCU_EN, 0xffffffff);
        regmap_read(afe->regmap, AFE_IRQ_MCU_EN, &tmp_reg);

        pm_runtime_put_sync(dev);

        regcache_cache_only(afe->regmap, true);
        regcache_mark_dirty(afe->regmap);

        /* register component */
        ret = devm_snd_soc_register_component(dev,
                                              &mt8189_afe_component,
                                              afe->dai_drivers,
                                              afe->num_dai_drivers);
        if (ret) {
                dev_err(dev, "afe component err: %d\n", ret);
                goto err_pm_disable;
        }

        return 0;

err_pm_disable:
        pm_runtime_put_sync(dev);
        return ret;
}

static void mt8189_afe_pcm_dev_remove(struct platform_device *pdev)
{
        struct mtk_base_afe *afe = platform_get_drvdata(pdev);
        struct device *dev = &pdev->dev;

        pm_runtime_put_sync(dev);
        if (!pm_runtime_status_suspended(dev))
                mt8189_afe_runtime_suspend(dev);

        mt8189_afe_disable_main_clock(afe);
        /* disable afe clock */
        mt8189_afe_disable_reg_rw_clk(afe);
        of_reserved_mem_device_release(dev);
}

static const struct of_device_id mt8189_afe_pcm_dt_match[] = {
        { .compatible = "mediatek,mt8189-afe-pcm", },
        {},
};
MODULE_DEVICE_TABLE(of, mt8189_afe_pcm_dt_match);

static const struct dev_pm_ops mt8189_afe_pm_ops = {
        SET_RUNTIME_PM_OPS(mt8189_afe_runtime_suspend,
                           mt8189_afe_runtime_resume, NULL)
};

static struct platform_driver mt8189_afe_pcm_driver = {
        .driver = {
                .name = "mt8189-afe-pcm",
                .of_match_table = mt8189_afe_pcm_dt_match,
                .pm = &mt8189_afe_pm_ops,
        },
        .probe = mt8189_afe_pcm_dev_probe,
        .remove = mt8189_afe_pcm_dev_remove,
};
module_platform_driver(mt8189_afe_pcm_driver);

MODULE_DESCRIPTION("Mediatek ALSA SoC AFE platform driver for 8189");
MODULE_AUTHOR("Darren Ye <darren.ye@mediatek.com>");
MODULE_LICENSE("GPL");