Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Trevor Wu | 1533 | 99.61% | 1 | 50.00% |
Kuninori Morimoto | 6 | 0.39% | 1 | 50.00% |
Total | 1539 | 2 |
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
// SPDX-License-Identifier: GPL-2.0 /* * MediaTek ALSA SoC Audio DAI PCM I/F Control * * Copyright (c) 2022 MediaTek Inc. * Author: Bicycle Tsai <bicycle.tsai@mediatek.com> * Trevor Wu <trevor.wu@mediatek.com> * Chun-Chia Chiu <chun-chia.chiu@mediatek.com> */ #include <linux/bitfield.h> #include <linux/regmap.h> #include <sound/pcm_params.h> #include "mt8188-afe-clk.h" #include "mt8188-afe-common.h" #include "mt8188-reg.h" enum { MTK_DAI_PCM_FMT_I2S, MTK_DAI_PCM_FMT_EIAJ, MTK_DAI_PCM_FMT_MODEA, MTK_DAI_PCM_FMT_MODEB, }; enum { MTK_DAI_PCM_CLK_A1SYS, MTK_DAI_PCM_CLK_A2SYS, MTK_DAI_PCM_CLK_26M_48K, MTK_DAI_PCM_CLK_26M_441K, }; struct mtk_dai_pcm_rate { unsigned int rate; unsigned int reg_value; }; struct mtk_dai_pcmif_priv { unsigned int slave_mode; unsigned int lrck_inv; unsigned int bck_inv; unsigned int format; }; static const struct mtk_dai_pcm_rate mtk_dai_pcm_rates[] = { { .rate = 8000, .reg_value = 0, }, { .rate = 16000, .reg_value = 1, }, { .rate = 32000, .reg_value = 2, }, { .rate = 48000, .reg_value = 3, }, { .rate = 11025, .reg_value = 1, }, { .rate = 22050, .reg_value = 2, }, { .rate = 44100, .reg_value = 3, }, }; static int mtk_dai_pcm_mode(unsigned int rate) { int i; for (i = 0; i < ARRAY_SIZE(mtk_dai_pcm_rates); i++) if (mtk_dai_pcm_rates[i].rate == rate) return mtk_dai_pcm_rates[i].reg_value; return -EINVAL; } static const struct snd_kcontrol_new mtk_dai_pcm_o000_mix[] = { SOC_DAPM_SINGLE_AUTODISABLE("I000 Switch", AFE_CONN0, 0, 1, 0), SOC_DAPM_SINGLE_AUTODISABLE("I070 Switch", AFE_CONN0_2, 6, 1, 0), }; static const struct snd_kcontrol_new mtk_dai_pcm_o001_mix[] = { SOC_DAPM_SINGLE_AUTODISABLE("I001 Switch", AFE_CONN1, 1, 1, 0), SOC_DAPM_SINGLE_AUTODISABLE("I071 Switch", AFE_CONN1_2, 7, 1, 0), }; static const struct snd_soc_dapm_widget mtk_dai_pcm_widgets[] = { SND_SOC_DAPM_MIXER("I002", SND_SOC_NOPM, 0, 0, NULL, 0), SND_SOC_DAPM_MIXER("I003", SND_SOC_NOPM, 0, 0, NULL, 0), SND_SOC_DAPM_MIXER("O000", SND_SOC_NOPM, 0, 0, mtk_dai_pcm_o000_mix, ARRAY_SIZE(mtk_dai_pcm_o000_mix)), SND_SOC_DAPM_MIXER("O001", SND_SOC_NOPM, 0, 0, mtk_dai_pcm_o001_mix, ARRAY_SIZE(mtk_dai_pcm_o001_mix)), SND_SOC_DAPM_SUPPLY("PCM_1_EN", PCM_INTF_CON1, 0, 0, NULL, 0), SND_SOC_DAPM_INPUT("PCM1_INPUT"), SND_SOC_DAPM_OUTPUT("PCM1_OUTPUT"), SND_SOC_DAPM_CLOCK_SUPPLY("aud_asrc11"), SND_SOC_DAPM_CLOCK_SUPPLY("aud_asrc12"), SND_SOC_DAPM_CLOCK_SUPPLY("aud_pcmif"), }; static const struct snd_soc_dapm_route mtk_dai_pcm_routes[] = { {"I002", NULL, "PCM1 Capture"}, {"I003", NULL, "PCM1 Capture"}, {"O000", "I000 Switch", "I000"}, {"O001", "I001 Switch", "I001"}, {"O000", "I070 Switch", "I070"}, {"O001", "I071 Switch", "I071"}, {"PCM1 Playback", NULL, "O000"}, {"PCM1 Playback", NULL, "O001"}, {"PCM1 Playback", NULL, "PCM_1_EN"}, {"PCM1 Playback", NULL, "aud_asrc12"}, {"PCM1 Playback", NULL, "aud_pcmif"}, {"PCM1 Capture", NULL, "PCM_1_EN"}, {"PCM1 Capture", NULL, "aud_asrc11"}, {"PCM1 Capture", NULL, "aud_pcmif"}, {"PCM1_OUTPUT", NULL, "PCM1 Playback"}, {"PCM1 Capture", NULL, "PCM1_INPUT"}, }; static int mtk_dai_pcm_configure(struct snd_pcm_substream *substream, struct snd_soc_dai *dai) { struct snd_pcm_runtime * const runtime = substream->runtime; struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai); struct mt8188_afe_private *afe_priv = afe->platform_priv; struct mtk_dai_pcmif_priv *pcmif_priv = NULL; unsigned int slave_mode; unsigned int lrck_inv; unsigned int bck_inv; unsigned int fmt; unsigned int bit_width = dai->sample_bits; unsigned int val = 0; unsigned int mask = 0; int fs = 0; int mode = 0; if (dai->id < 0) return -EINVAL; pcmif_priv = afe_priv->dai_priv[dai->id]; slave_mode = pcmif_priv->slave_mode; lrck_inv = pcmif_priv->lrck_inv; bck_inv = pcmif_priv->bck_inv; fmt = pcmif_priv->format; /* sync freq mode */ fs = mt8188_afe_fs_timing(runtime->rate); if (fs < 0) return -EINVAL; val |= FIELD_PREP(PCM_INTF_CON2_SYNC_FREQ_MODE_MASK, fs); mask |= PCM_INTF_CON2_SYNC_FREQ_MODE_MASK; /* clk domain sel */ if (runtime->rate % 8000) val |= FIELD_PREP(PCM_INTF_CON2_CLK_DOMAIN_SEL_MASK, MTK_DAI_PCM_CLK_26M_441K); else val |= FIELD_PREP(PCM_INTF_CON2_CLK_DOMAIN_SEL_MASK, MTK_DAI_PCM_CLK_26M_48K); mask |= PCM_INTF_CON2_CLK_DOMAIN_SEL_MASK; regmap_update_bits(afe->regmap, PCM_INTF_CON2, mask, val); val = 0; mask = 0; /* pcm mode */ mode = mtk_dai_pcm_mode(runtime->rate); if (mode < 0) return -EINVAL; val |= FIELD_PREP(PCM_INTF_CON1_PCM_MODE_MASK, mode); mask |= PCM_INTF_CON1_PCM_MODE_MASK; /* pcm format */ val |= FIELD_PREP(PCM_INTF_CON1_PCM_FMT_MASK, fmt); mask |= PCM_INTF_CON1_PCM_FMT_MASK; /* pcm sync length */ if (fmt == MTK_DAI_PCM_FMT_MODEA || fmt == MTK_DAI_PCM_FMT_MODEB) val |= FIELD_PREP(PCM_INTF_CON1_SYNC_LENGTH_MASK, 1); else val |= FIELD_PREP(PCM_INTF_CON1_SYNC_LENGTH_MASK, bit_width); mask |= PCM_INTF_CON1_SYNC_LENGTH_MASK; /* pcm bits, word length */ if (bit_width > 16) { val |= PCM_INTF_CON1_PCM_24BIT; val |= PCM_INTF_CON1_PCM_WLEN_64BCK; } else { val |= PCM_INTF_CON1_PCM_16BIT; val |= PCM_INTF_CON1_PCM_WLEN_32BCK; } mask |= PCM_INTF_CON1_PCM_BIT_MASK; mask |= PCM_INTF_CON1_PCM_WLEN_MASK; /* master/slave */ if (!slave_mode) { val |= PCM_INTF_CON1_PCM_MASTER; if (lrck_inv) val |= PCM_INTF_CON1_SYNC_OUT_INV; if (bck_inv) val |= PCM_INTF_CON1_BCLK_OUT_INV; mask |= PCM_INTF_CON1_CLK_OUT_INV_MASK; } else { val |= PCM_INTF_CON1_PCM_SLAVE; if (lrck_inv) val |= PCM_INTF_CON1_SYNC_IN_INV; if (bck_inv) val |= PCM_INTF_CON1_BCLK_IN_INV; mask |= PCM_INTF_CON1_CLK_IN_INV_MASK; // TODO: add asrc setting for slave mode } mask |= PCM_INTF_CON1_PCM_M_S_MASK; regmap_update_bits(afe->regmap, PCM_INTF_CON1, mask, val); return 0; } /* dai ops */ static int mtk_dai_pcm_prepare(struct snd_pcm_substream *substream, struct snd_soc_dai *dai) { if (snd_soc_dai_get_widget_playback(dai)->active || snd_soc_dai_get_widget_capture(dai)->active) return 0; return mtk_dai_pcm_configure(substream, dai); } static int mtk_dai_pcm_set_fmt(struct snd_soc_dai *dai, unsigned int fmt) { struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai); struct mt8188_afe_private *afe_priv = afe->platform_priv; struct mtk_dai_pcmif_priv *pcmif_priv = NULL; dev_dbg(dai->dev, "%s fmt 0x%x\n", __func__, fmt); if (dai->id < 0) return -EINVAL; pcmif_priv = afe_priv->dai_priv[dai->id]; switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) { case SND_SOC_DAIFMT_I2S: pcmif_priv->format = MTK_DAI_PCM_FMT_I2S; break; case SND_SOC_DAIFMT_DSP_A: pcmif_priv->format = MTK_DAI_PCM_FMT_MODEA; break; case SND_SOC_DAIFMT_DSP_B: pcmif_priv->format = MTK_DAI_PCM_FMT_MODEB; break; default: return -EINVAL; } switch (fmt & SND_SOC_DAIFMT_INV_MASK) { case SND_SOC_DAIFMT_NB_NF: pcmif_priv->bck_inv = 0; pcmif_priv->lrck_inv = 0; break; case SND_SOC_DAIFMT_NB_IF: pcmif_priv->bck_inv = 0; pcmif_priv->lrck_inv = 1; break; case SND_SOC_DAIFMT_IB_NF: pcmif_priv->bck_inv = 1; pcmif_priv->lrck_inv = 0; break; case SND_SOC_DAIFMT_IB_IF: pcmif_priv->bck_inv = 1; pcmif_priv->lrck_inv = 1; break; default: return -EINVAL; } switch (fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) { case SND_SOC_DAIFMT_BC_FC: pcmif_priv->slave_mode = 1; break; case SND_SOC_DAIFMT_BP_FP: pcmif_priv->slave_mode = 0; break; default: return -EINVAL; } return 0; } static const struct snd_soc_dai_ops mtk_dai_pcm_ops = { .prepare = mtk_dai_pcm_prepare, .set_fmt = mtk_dai_pcm_set_fmt, }; /* dai driver */ #define MTK_PCM_RATES (SNDRV_PCM_RATE_8000_48000) #define MTK_PCM_FORMATS (SNDRV_PCM_FMTBIT_S16_LE |\ SNDRV_PCM_FMTBIT_S24_LE |\ SNDRV_PCM_FMTBIT_S32_LE) static struct snd_soc_dai_driver mtk_dai_pcm_driver[] = { { .name = "PCM1", .id = MT8188_AFE_IO_PCM, .playback = { .stream_name = "PCM1 Playback", .channels_min = 1, .channels_max = 2, .rates = MTK_PCM_RATES, .formats = MTK_PCM_FORMATS, }, .capture = { .stream_name = "PCM1 Capture", .channels_min = 1, .channels_max = 2, .rates = MTK_PCM_RATES, .formats = MTK_PCM_FORMATS, }, .ops = &mtk_dai_pcm_ops, .symmetric_rate = 1, .symmetric_sample_bits = 1, }, }; static int init_pcmif_priv_data(struct mtk_base_afe *afe) { struct mt8188_afe_private *afe_priv = afe->platform_priv; struct mtk_dai_pcmif_priv *pcmif_priv; pcmif_priv = devm_kzalloc(afe->dev, sizeof(struct mtk_dai_pcmif_priv), GFP_KERNEL); if (!pcmif_priv) return -ENOMEM; afe_priv->dai_priv[MT8188_AFE_IO_PCM] = pcmif_priv; return 0; } int mt8188_dai_pcm_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 = mtk_dai_pcm_driver; dai->num_dai_drivers = ARRAY_SIZE(mtk_dai_pcm_driver); dai->dapm_widgets = mtk_dai_pcm_widgets; dai->num_dapm_widgets = ARRAY_SIZE(mtk_dai_pcm_widgets); dai->dapm_routes = mtk_dai_pcm_routes; dai->num_dapm_routes = ARRAY_SIZE(mtk_dai_pcm_routes); return init_pcmif_priv_data(afe); }
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1