Path: blob/master/sound/soc/mediatek/mt8188/mt8188-dai-pcm.c
26488 views
// SPDX-License-Identifier: GPL-2.01/*2* MediaTek ALSA SoC Audio DAI PCM I/F Control3*4* Copyright (c) 2022 MediaTek Inc.5* Author: Bicycle Tsai <[email protected]>6* Trevor Wu <[email protected]>7* Chun-Chia Chiu <[email protected]>8*/910#include <linux/bitfield.h>11#include <linux/regmap.h>12#include <sound/pcm_params.h>13#include "mt8188-afe-clk.h"14#include "mt8188-afe-common.h"15#include "mt8188-reg.h"1617enum {18MTK_DAI_PCM_FMT_I2S,19MTK_DAI_PCM_FMT_EIAJ,20MTK_DAI_PCM_FMT_MODEA,21MTK_DAI_PCM_FMT_MODEB,22};2324enum {25MTK_DAI_PCM_CLK_A1SYS,26MTK_DAI_PCM_CLK_A2SYS,27MTK_DAI_PCM_CLK_26M_48K,28MTK_DAI_PCM_CLK_26M_441K,29};3031struct mtk_dai_pcm_rate {32unsigned int rate;33unsigned int reg_value;34};3536struct mtk_dai_pcmif_priv {37unsigned int slave_mode;38unsigned int lrck_inv;39unsigned int bck_inv;40unsigned int format;41};4243static const struct mtk_dai_pcm_rate mtk_dai_pcm_rates[] = {44{ .rate = 8000, .reg_value = 0, },45{ .rate = 16000, .reg_value = 1, },46{ .rate = 32000, .reg_value = 2, },47{ .rate = 48000, .reg_value = 3, },48{ .rate = 11025, .reg_value = 1, },49{ .rate = 22050, .reg_value = 2, },50{ .rate = 44100, .reg_value = 3, },51};5253static int mtk_dai_pcm_mode(unsigned int rate)54{55int i;5657for (i = 0; i < ARRAY_SIZE(mtk_dai_pcm_rates); i++)58if (mtk_dai_pcm_rates[i].rate == rate)59return mtk_dai_pcm_rates[i].reg_value;6061return -EINVAL;62}6364static const struct snd_kcontrol_new mtk_dai_pcm_o000_mix[] = {65SOC_DAPM_SINGLE_AUTODISABLE("I000 Switch", AFE_CONN0, 0, 1, 0),66SOC_DAPM_SINGLE_AUTODISABLE("I070 Switch", AFE_CONN0_2, 6, 1, 0),67};6869static const struct snd_kcontrol_new mtk_dai_pcm_o001_mix[] = {70SOC_DAPM_SINGLE_AUTODISABLE("I001 Switch", AFE_CONN1, 1, 1, 0),71SOC_DAPM_SINGLE_AUTODISABLE("I071 Switch", AFE_CONN1_2, 7, 1, 0),72};7374static const struct snd_soc_dapm_widget mtk_dai_pcm_widgets[] = {75SND_SOC_DAPM_MIXER("I002", SND_SOC_NOPM, 0, 0, NULL, 0),76SND_SOC_DAPM_MIXER("I003", SND_SOC_NOPM, 0, 0, NULL, 0),77SND_SOC_DAPM_MIXER("O000", SND_SOC_NOPM, 0, 0,78mtk_dai_pcm_o000_mix,79ARRAY_SIZE(mtk_dai_pcm_o000_mix)),80SND_SOC_DAPM_MIXER("O001", SND_SOC_NOPM, 0, 0,81mtk_dai_pcm_o001_mix,82ARRAY_SIZE(mtk_dai_pcm_o001_mix)),8384SND_SOC_DAPM_SUPPLY("PCM_1_EN", PCM_INTF_CON1, 0, 0, NULL, 0),8586SND_SOC_DAPM_INPUT("PCM1_INPUT"),87SND_SOC_DAPM_OUTPUT("PCM1_OUTPUT"),8889SND_SOC_DAPM_CLOCK_SUPPLY("aud_asrc11"),90SND_SOC_DAPM_CLOCK_SUPPLY("aud_asrc12"),91SND_SOC_DAPM_CLOCK_SUPPLY("aud_pcmif"),92};9394static const struct snd_soc_dapm_route mtk_dai_pcm_routes[] = {95{"I002", NULL, "PCM1 Capture"},96{"I003", NULL, "PCM1 Capture"},9798{"O000", "I000 Switch", "I000"},99{"O001", "I001 Switch", "I001"},100101{"O000", "I070 Switch", "I070"},102{"O001", "I071 Switch", "I071"},103104{"PCM1 Playback", NULL, "O000"},105{"PCM1 Playback", NULL, "O001"},106107{"PCM1 Playback", NULL, "PCM_1_EN"},108{"PCM1 Playback", NULL, "aud_asrc12"},109{"PCM1 Playback", NULL, "aud_pcmif"},110111{"PCM1 Capture", NULL, "PCM_1_EN"},112{"PCM1 Capture", NULL, "aud_asrc11"},113{"PCM1 Capture", NULL, "aud_pcmif"},114115{"PCM1_OUTPUT", NULL, "PCM1 Playback"},116{"PCM1 Capture", NULL, "PCM1_INPUT"},117};118119static int mtk_dai_pcm_configure(struct snd_pcm_substream *substream,120struct snd_soc_dai *dai)121{122struct snd_pcm_runtime * const runtime = substream->runtime;123struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);124struct mt8188_afe_private *afe_priv = afe->platform_priv;125struct mtk_dai_pcmif_priv *pcmif_priv = NULL;126unsigned int slave_mode;127unsigned int lrck_inv;128unsigned int bck_inv;129unsigned int fmt;130unsigned int bit_width = dai->symmetric_sample_bits;131unsigned int val = 0;132unsigned int mask = 0;133int fs = 0;134int mode = 0;135136if (dai->id < 0)137return -EINVAL;138139pcmif_priv = afe_priv->dai_priv[dai->id];140slave_mode = pcmif_priv->slave_mode;141lrck_inv = pcmif_priv->lrck_inv;142bck_inv = pcmif_priv->bck_inv;143fmt = pcmif_priv->format;144145/* sync freq mode */146fs = mt8188_afe_fs_timing(runtime->rate);147if (fs < 0)148return -EINVAL;149150val |= FIELD_PREP(PCM_INTF_CON2_SYNC_FREQ_MODE_MASK, fs);151mask |= PCM_INTF_CON2_SYNC_FREQ_MODE_MASK;152153/* clk domain sel */154if (runtime->rate % 8000)155val |= FIELD_PREP(PCM_INTF_CON2_CLK_DOMAIN_SEL_MASK,156MTK_DAI_PCM_CLK_26M_441K);157else158val |= FIELD_PREP(PCM_INTF_CON2_CLK_DOMAIN_SEL_MASK,159MTK_DAI_PCM_CLK_26M_48K);160mask |= PCM_INTF_CON2_CLK_DOMAIN_SEL_MASK;161162regmap_update_bits(afe->regmap, PCM_INTF_CON2, mask, val);163164val = 0;165mask = 0;166167/* pcm mode */168mode = mtk_dai_pcm_mode(runtime->rate);169if (mode < 0)170return -EINVAL;171172val |= FIELD_PREP(PCM_INTF_CON1_PCM_MODE_MASK, mode);173mask |= PCM_INTF_CON1_PCM_MODE_MASK;174175/* pcm format */176val |= FIELD_PREP(PCM_INTF_CON1_PCM_FMT_MASK, fmt);177mask |= PCM_INTF_CON1_PCM_FMT_MASK;178179/* pcm sync length */180if (fmt == MTK_DAI_PCM_FMT_MODEA ||181fmt == MTK_DAI_PCM_FMT_MODEB)182val |= FIELD_PREP(PCM_INTF_CON1_SYNC_LENGTH_MASK, 1);183else184val |= FIELD_PREP(PCM_INTF_CON1_SYNC_LENGTH_MASK, bit_width);185mask |= PCM_INTF_CON1_SYNC_LENGTH_MASK;186187/* pcm bits, word length */188if (bit_width > 16) {189val |= PCM_INTF_CON1_PCM_24BIT;190val |= PCM_INTF_CON1_PCM_WLEN_64BCK;191} else {192val |= PCM_INTF_CON1_PCM_16BIT;193val |= PCM_INTF_CON1_PCM_WLEN_32BCK;194}195mask |= PCM_INTF_CON1_PCM_BIT_MASK;196mask |= PCM_INTF_CON1_PCM_WLEN_MASK;197198/* master/slave */199if (!slave_mode) {200val |= PCM_INTF_CON1_PCM_MASTER;201202if (lrck_inv)203val |= PCM_INTF_CON1_SYNC_OUT_INV;204if (bck_inv)205val |= PCM_INTF_CON1_BCLK_OUT_INV;206mask |= PCM_INTF_CON1_CLK_OUT_INV_MASK;207} else {208val |= PCM_INTF_CON1_PCM_SLAVE;209210if (lrck_inv)211val |= PCM_INTF_CON1_SYNC_IN_INV;212if (bck_inv)213val |= PCM_INTF_CON1_BCLK_IN_INV;214mask |= PCM_INTF_CON1_CLK_IN_INV_MASK;215216// TODO: add asrc setting for slave mode217}218mask |= PCM_INTF_CON1_PCM_M_S_MASK;219220regmap_update_bits(afe->regmap, PCM_INTF_CON1, mask, val);221222return 0;223}224225/* dai ops */226static int mtk_dai_pcm_prepare(struct snd_pcm_substream *substream,227struct snd_soc_dai *dai)228{229if (snd_soc_dai_get_widget_playback(dai)->active ||230snd_soc_dai_get_widget_capture(dai)->active)231return 0;232233return mtk_dai_pcm_configure(substream, dai);234}235236static int mtk_dai_pcm_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)237{238struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);239struct mt8188_afe_private *afe_priv = afe->platform_priv;240struct mtk_dai_pcmif_priv *pcmif_priv = NULL;241242dev_dbg(dai->dev, "%s fmt 0x%x\n", __func__, fmt);243244if (dai->id < 0)245return -EINVAL;246247pcmif_priv = afe_priv->dai_priv[dai->id];248249switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {250case SND_SOC_DAIFMT_I2S:251pcmif_priv->format = MTK_DAI_PCM_FMT_I2S;252break;253case SND_SOC_DAIFMT_DSP_A:254pcmif_priv->format = MTK_DAI_PCM_FMT_MODEA;255break;256case SND_SOC_DAIFMT_DSP_B:257pcmif_priv->format = MTK_DAI_PCM_FMT_MODEB;258break;259default:260return -EINVAL;261}262263switch (fmt & SND_SOC_DAIFMT_INV_MASK) {264case SND_SOC_DAIFMT_NB_NF:265pcmif_priv->bck_inv = 0;266pcmif_priv->lrck_inv = 0;267break;268case SND_SOC_DAIFMT_NB_IF:269pcmif_priv->bck_inv = 0;270pcmif_priv->lrck_inv = 1;271break;272case SND_SOC_DAIFMT_IB_NF:273pcmif_priv->bck_inv = 1;274pcmif_priv->lrck_inv = 0;275break;276case SND_SOC_DAIFMT_IB_IF:277pcmif_priv->bck_inv = 1;278pcmif_priv->lrck_inv = 1;279break;280default:281return -EINVAL;282}283284switch (fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) {285case SND_SOC_DAIFMT_BC_FC:286pcmif_priv->slave_mode = 1;287break;288case SND_SOC_DAIFMT_BP_FP:289pcmif_priv->slave_mode = 0;290break;291default:292return -EINVAL;293}294295return 0;296}297298static const struct snd_soc_dai_ops mtk_dai_pcm_ops = {299.prepare = mtk_dai_pcm_prepare,300.set_fmt = mtk_dai_pcm_set_fmt,301};302303/* dai driver */304#define MTK_PCM_RATES (SNDRV_PCM_RATE_8000_48000)305306#define MTK_PCM_FORMATS (SNDRV_PCM_FMTBIT_S16_LE |\307SNDRV_PCM_FMTBIT_S24_LE |\308SNDRV_PCM_FMTBIT_S32_LE)309310static struct snd_soc_dai_driver mtk_dai_pcm_driver[] = {311{312.name = "PCM1",313.id = MT8188_AFE_IO_PCM,314.playback = {315.stream_name = "PCM1 Playback",316.channels_min = 1,317.channels_max = 2,318.rates = MTK_PCM_RATES,319.formats = MTK_PCM_FORMATS,320},321.capture = {322.stream_name = "PCM1 Capture",323.channels_min = 1,324.channels_max = 2,325.rates = MTK_PCM_RATES,326.formats = MTK_PCM_FORMATS,327},328.ops = &mtk_dai_pcm_ops,329.symmetric_rate = 1,330.symmetric_sample_bits = 1,331},332};333334static int init_pcmif_priv_data(struct mtk_base_afe *afe)335{336struct mt8188_afe_private *afe_priv = afe->platform_priv;337struct mtk_dai_pcmif_priv *pcmif_priv;338339pcmif_priv = devm_kzalloc(afe->dev, sizeof(struct mtk_dai_pcmif_priv),340GFP_KERNEL);341if (!pcmif_priv)342return -ENOMEM;343344afe_priv->dai_priv[MT8188_AFE_IO_PCM] = pcmif_priv;345return 0;346}347348int mt8188_dai_pcm_register(struct mtk_base_afe *afe)349{350struct mtk_base_afe_dai *dai;351352dai = devm_kzalloc(afe->dev, sizeof(*dai), GFP_KERNEL);353if (!dai)354return -ENOMEM;355356list_add(&dai->list, &afe->sub_dais);357358dai->dai_drivers = mtk_dai_pcm_driver;359dai->num_dai_drivers = ARRAY_SIZE(mtk_dai_pcm_driver);360361dai->dapm_widgets = mtk_dai_pcm_widgets;362dai->num_dapm_widgets = ARRAY_SIZE(mtk_dai_pcm_widgets);363dai->dapm_routes = mtk_dai_pcm_routes;364dai->num_dapm_routes = ARRAY_SIZE(mtk_dai_pcm_routes);365366return init_pcmif_priv_data(afe);367}368369370