1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * tegra30_i2s.c - Tegra30 I2S driver
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Author: Stephen Warren <swarren@nvidia.com>
6*4882a593Smuzhiyun * Copyright (c) 2010-2012, NVIDIA CORPORATION. All rights reserved.
7*4882a593Smuzhiyun *
8*4882a593Smuzhiyun * Based on code copyright/by:
9*4882a593Smuzhiyun *
10*4882a593Smuzhiyun * Copyright (c) 2009-2010, NVIDIA Corporation.
11*4882a593Smuzhiyun * Scott Peterson <speterson@nvidia.com>
12*4882a593Smuzhiyun *
13*4882a593Smuzhiyun * Copyright (C) 2010 Google, Inc.
14*4882a593Smuzhiyun * Iliyan Malchev <malchev@google.com>
15*4882a593Smuzhiyun */
16*4882a593Smuzhiyun
17*4882a593Smuzhiyun #include <linux/clk.h>
18*4882a593Smuzhiyun #include <linux/device.h>
19*4882a593Smuzhiyun #include <linux/io.h>
20*4882a593Smuzhiyun #include <linux/module.h>
21*4882a593Smuzhiyun #include <linux/of.h>
22*4882a593Smuzhiyun #include <linux/of_device.h>
23*4882a593Smuzhiyun #include <linux/platform_device.h>
24*4882a593Smuzhiyun #include <linux/pm_runtime.h>
25*4882a593Smuzhiyun #include <linux/regmap.h>
26*4882a593Smuzhiyun #include <linux/slab.h>
27*4882a593Smuzhiyun #include <sound/core.h>
28*4882a593Smuzhiyun #include <sound/pcm.h>
29*4882a593Smuzhiyun #include <sound/pcm_params.h>
30*4882a593Smuzhiyun #include <sound/soc.h>
31*4882a593Smuzhiyun #include <sound/dmaengine_pcm.h>
32*4882a593Smuzhiyun
33*4882a593Smuzhiyun #include "tegra30_ahub.h"
34*4882a593Smuzhiyun #include "tegra30_i2s.h"
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun #define DRV_NAME "tegra30-i2s"
37*4882a593Smuzhiyun
tegra30_i2s_runtime_suspend(struct device * dev)38*4882a593Smuzhiyun static int tegra30_i2s_runtime_suspend(struct device *dev)
39*4882a593Smuzhiyun {
40*4882a593Smuzhiyun struct tegra30_i2s *i2s = dev_get_drvdata(dev);
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun regcache_cache_only(i2s->regmap, true);
43*4882a593Smuzhiyun
44*4882a593Smuzhiyun clk_disable_unprepare(i2s->clk_i2s);
45*4882a593Smuzhiyun
46*4882a593Smuzhiyun return 0;
47*4882a593Smuzhiyun }
48*4882a593Smuzhiyun
tegra30_i2s_runtime_resume(struct device * dev)49*4882a593Smuzhiyun static int tegra30_i2s_runtime_resume(struct device *dev)
50*4882a593Smuzhiyun {
51*4882a593Smuzhiyun struct tegra30_i2s *i2s = dev_get_drvdata(dev);
52*4882a593Smuzhiyun int ret;
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun ret = clk_prepare_enable(i2s->clk_i2s);
55*4882a593Smuzhiyun if (ret) {
56*4882a593Smuzhiyun dev_err(dev, "clk_enable failed: %d\n", ret);
57*4882a593Smuzhiyun return ret;
58*4882a593Smuzhiyun }
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun regcache_cache_only(i2s->regmap, false);
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun return 0;
63*4882a593Smuzhiyun }
64*4882a593Smuzhiyun
tegra30_i2s_set_fmt(struct snd_soc_dai * dai,unsigned int fmt)65*4882a593Smuzhiyun static int tegra30_i2s_set_fmt(struct snd_soc_dai *dai,
66*4882a593Smuzhiyun unsigned int fmt)
67*4882a593Smuzhiyun {
68*4882a593Smuzhiyun struct tegra30_i2s *i2s = snd_soc_dai_get_drvdata(dai);
69*4882a593Smuzhiyun unsigned int mask = 0, val = 0;
70*4882a593Smuzhiyun
71*4882a593Smuzhiyun switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
72*4882a593Smuzhiyun case SND_SOC_DAIFMT_NB_NF:
73*4882a593Smuzhiyun break;
74*4882a593Smuzhiyun default:
75*4882a593Smuzhiyun return -EINVAL;
76*4882a593Smuzhiyun }
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun mask |= TEGRA30_I2S_CTRL_MASTER_ENABLE;
79*4882a593Smuzhiyun switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
80*4882a593Smuzhiyun case SND_SOC_DAIFMT_CBS_CFS:
81*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_MASTER_ENABLE;
82*4882a593Smuzhiyun break;
83*4882a593Smuzhiyun case SND_SOC_DAIFMT_CBM_CFM:
84*4882a593Smuzhiyun break;
85*4882a593Smuzhiyun default:
86*4882a593Smuzhiyun return -EINVAL;
87*4882a593Smuzhiyun }
88*4882a593Smuzhiyun
89*4882a593Smuzhiyun mask |= TEGRA30_I2S_CTRL_FRAME_FORMAT_MASK |
90*4882a593Smuzhiyun TEGRA30_I2S_CTRL_LRCK_MASK;
91*4882a593Smuzhiyun switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
92*4882a593Smuzhiyun case SND_SOC_DAIFMT_DSP_A:
93*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_FRAME_FORMAT_FSYNC;
94*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_LRCK_L_LOW;
95*4882a593Smuzhiyun break;
96*4882a593Smuzhiyun case SND_SOC_DAIFMT_DSP_B:
97*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_FRAME_FORMAT_FSYNC;
98*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_LRCK_R_LOW;
99*4882a593Smuzhiyun break;
100*4882a593Smuzhiyun case SND_SOC_DAIFMT_I2S:
101*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_FRAME_FORMAT_LRCK;
102*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_LRCK_L_LOW;
103*4882a593Smuzhiyun break;
104*4882a593Smuzhiyun case SND_SOC_DAIFMT_RIGHT_J:
105*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_FRAME_FORMAT_LRCK;
106*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_LRCK_L_LOW;
107*4882a593Smuzhiyun break;
108*4882a593Smuzhiyun case SND_SOC_DAIFMT_LEFT_J:
109*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_FRAME_FORMAT_LRCK;
110*4882a593Smuzhiyun val |= TEGRA30_I2S_CTRL_LRCK_L_LOW;
111*4882a593Smuzhiyun break;
112*4882a593Smuzhiyun default:
113*4882a593Smuzhiyun return -EINVAL;
114*4882a593Smuzhiyun }
115*4882a593Smuzhiyun
116*4882a593Smuzhiyun pm_runtime_get_sync(dai->dev);
117*4882a593Smuzhiyun regmap_update_bits(i2s->regmap, TEGRA30_I2S_CTRL, mask, val);
118*4882a593Smuzhiyun pm_runtime_put(dai->dev);
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun return 0;
121*4882a593Smuzhiyun }
122*4882a593Smuzhiyun
tegra30_i2s_hw_params(struct snd_pcm_substream * substream,struct snd_pcm_hw_params * params,struct snd_soc_dai * dai)123*4882a593Smuzhiyun static int tegra30_i2s_hw_params(struct snd_pcm_substream *substream,
124*4882a593Smuzhiyun struct snd_pcm_hw_params *params,
125*4882a593Smuzhiyun struct snd_soc_dai *dai)
126*4882a593Smuzhiyun {
127*4882a593Smuzhiyun struct device *dev = dai->dev;
128*4882a593Smuzhiyun struct tegra30_i2s *i2s = snd_soc_dai_get_drvdata(dai);
129*4882a593Smuzhiyun unsigned int mask, val, reg;
130*4882a593Smuzhiyun int ret, sample_size, srate, i2sclock, bitcnt;
131*4882a593Smuzhiyun struct tegra30_ahub_cif_conf cif_conf;
132*4882a593Smuzhiyun
133*4882a593Smuzhiyun if (params_channels(params) != 2)
134*4882a593Smuzhiyun return -EINVAL;
135*4882a593Smuzhiyun
136*4882a593Smuzhiyun mask = TEGRA30_I2S_CTRL_BIT_SIZE_MASK;
137*4882a593Smuzhiyun switch (params_format(params)) {
138*4882a593Smuzhiyun case SNDRV_PCM_FORMAT_S16_LE:
139*4882a593Smuzhiyun val = TEGRA30_I2S_CTRL_BIT_SIZE_16;
140*4882a593Smuzhiyun sample_size = 16;
141*4882a593Smuzhiyun break;
142*4882a593Smuzhiyun default:
143*4882a593Smuzhiyun return -EINVAL;
144*4882a593Smuzhiyun }
145*4882a593Smuzhiyun
146*4882a593Smuzhiyun regmap_update_bits(i2s->regmap, TEGRA30_I2S_CTRL, mask, val);
147*4882a593Smuzhiyun
148*4882a593Smuzhiyun srate = params_rate(params);
149*4882a593Smuzhiyun
150*4882a593Smuzhiyun /* Final "* 2" required by Tegra hardware */
151*4882a593Smuzhiyun i2sclock = srate * params_channels(params) * sample_size * 2;
152*4882a593Smuzhiyun
153*4882a593Smuzhiyun bitcnt = (i2sclock / (2 * srate)) - 1;
154*4882a593Smuzhiyun if (bitcnt < 0 || bitcnt > TEGRA30_I2S_TIMING_CHANNEL_BIT_COUNT_MASK_US)
155*4882a593Smuzhiyun return -EINVAL;
156*4882a593Smuzhiyun
157*4882a593Smuzhiyun ret = clk_set_rate(i2s->clk_i2s, i2sclock);
158*4882a593Smuzhiyun if (ret) {
159*4882a593Smuzhiyun dev_err(dev, "Can't set I2S clock rate: %d\n", ret);
160*4882a593Smuzhiyun return ret;
161*4882a593Smuzhiyun }
162*4882a593Smuzhiyun
163*4882a593Smuzhiyun val = bitcnt << TEGRA30_I2S_TIMING_CHANNEL_BIT_COUNT_SHIFT;
164*4882a593Smuzhiyun
165*4882a593Smuzhiyun if (i2sclock % (2 * srate))
166*4882a593Smuzhiyun val |= TEGRA30_I2S_TIMING_NON_SYM_ENABLE;
167*4882a593Smuzhiyun
168*4882a593Smuzhiyun regmap_write(i2s->regmap, TEGRA30_I2S_TIMING, val);
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun cif_conf.threshold = 0;
171*4882a593Smuzhiyun cif_conf.audio_channels = 2;
172*4882a593Smuzhiyun cif_conf.client_channels = 2;
173*4882a593Smuzhiyun cif_conf.audio_bits = TEGRA30_AUDIOCIF_BITS_16;
174*4882a593Smuzhiyun cif_conf.client_bits = TEGRA30_AUDIOCIF_BITS_16;
175*4882a593Smuzhiyun cif_conf.expand = 0;
176*4882a593Smuzhiyun cif_conf.stereo_conv = 0;
177*4882a593Smuzhiyun cif_conf.replicate = 0;
178*4882a593Smuzhiyun cif_conf.truncate = 0;
179*4882a593Smuzhiyun cif_conf.mono_conv = 0;
180*4882a593Smuzhiyun
181*4882a593Smuzhiyun if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
182*4882a593Smuzhiyun cif_conf.direction = TEGRA30_AUDIOCIF_DIRECTION_RX;
183*4882a593Smuzhiyun reg = TEGRA30_I2S_CIF_RX_CTRL;
184*4882a593Smuzhiyun } else {
185*4882a593Smuzhiyun cif_conf.direction = TEGRA30_AUDIOCIF_DIRECTION_TX;
186*4882a593Smuzhiyun reg = TEGRA30_I2S_CIF_TX_CTRL;
187*4882a593Smuzhiyun }
188*4882a593Smuzhiyun
189*4882a593Smuzhiyun i2s->soc_data->set_audio_cif(i2s->regmap, reg, &cif_conf);
190*4882a593Smuzhiyun
191*4882a593Smuzhiyun val = (1 << TEGRA30_I2S_OFFSET_RX_DATA_OFFSET_SHIFT) |
192*4882a593Smuzhiyun (1 << TEGRA30_I2S_OFFSET_TX_DATA_OFFSET_SHIFT);
193*4882a593Smuzhiyun regmap_write(i2s->regmap, TEGRA30_I2S_OFFSET, val);
194*4882a593Smuzhiyun
195*4882a593Smuzhiyun return 0;
196*4882a593Smuzhiyun }
197*4882a593Smuzhiyun
tegra30_i2s_start_playback(struct tegra30_i2s * i2s)198*4882a593Smuzhiyun static void tegra30_i2s_start_playback(struct tegra30_i2s *i2s)
199*4882a593Smuzhiyun {
200*4882a593Smuzhiyun tegra30_ahub_enable_tx_fifo(i2s->playback_fifo_cif);
201*4882a593Smuzhiyun regmap_update_bits(i2s->regmap, TEGRA30_I2S_CTRL,
202*4882a593Smuzhiyun TEGRA30_I2S_CTRL_XFER_EN_TX,
203*4882a593Smuzhiyun TEGRA30_I2S_CTRL_XFER_EN_TX);
204*4882a593Smuzhiyun }
205*4882a593Smuzhiyun
tegra30_i2s_stop_playback(struct tegra30_i2s * i2s)206*4882a593Smuzhiyun static void tegra30_i2s_stop_playback(struct tegra30_i2s *i2s)
207*4882a593Smuzhiyun {
208*4882a593Smuzhiyun tegra30_ahub_disable_tx_fifo(i2s->playback_fifo_cif);
209*4882a593Smuzhiyun regmap_update_bits(i2s->regmap, TEGRA30_I2S_CTRL,
210*4882a593Smuzhiyun TEGRA30_I2S_CTRL_XFER_EN_TX, 0);
211*4882a593Smuzhiyun }
212*4882a593Smuzhiyun
tegra30_i2s_start_capture(struct tegra30_i2s * i2s)213*4882a593Smuzhiyun static void tegra30_i2s_start_capture(struct tegra30_i2s *i2s)
214*4882a593Smuzhiyun {
215*4882a593Smuzhiyun tegra30_ahub_enable_rx_fifo(i2s->capture_fifo_cif);
216*4882a593Smuzhiyun regmap_update_bits(i2s->regmap, TEGRA30_I2S_CTRL,
217*4882a593Smuzhiyun TEGRA30_I2S_CTRL_XFER_EN_RX,
218*4882a593Smuzhiyun TEGRA30_I2S_CTRL_XFER_EN_RX);
219*4882a593Smuzhiyun }
220*4882a593Smuzhiyun
tegra30_i2s_stop_capture(struct tegra30_i2s * i2s)221*4882a593Smuzhiyun static void tegra30_i2s_stop_capture(struct tegra30_i2s *i2s)
222*4882a593Smuzhiyun {
223*4882a593Smuzhiyun regmap_update_bits(i2s->regmap, TEGRA30_I2S_CTRL,
224*4882a593Smuzhiyun TEGRA30_I2S_CTRL_XFER_EN_RX, 0);
225*4882a593Smuzhiyun tegra30_ahub_disable_rx_fifo(i2s->capture_fifo_cif);
226*4882a593Smuzhiyun }
227*4882a593Smuzhiyun
tegra30_i2s_trigger(struct snd_pcm_substream * substream,int cmd,struct snd_soc_dai * dai)228*4882a593Smuzhiyun static int tegra30_i2s_trigger(struct snd_pcm_substream *substream, int cmd,
229*4882a593Smuzhiyun struct snd_soc_dai *dai)
230*4882a593Smuzhiyun {
231*4882a593Smuzhiyun struct tegra30_i2s *i2s = snd_soc_dai_get_drvdata(dai);
232*4882a593Smuzhiyun
233*4882a593Smuzhiyun switch (cmd) {
234*4882a593Smuzhiyun case SNDRV_PCM_TRIGGER_START:
235*4882a593Smuzhiyun case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
236*4882a593Smuzhiyun case SNDRV_PCM_TRIGGER_RESUME:
237*4882a593Smuzhiyun if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
238*4882a593Smuzhiyun tegra30_i2s_start_playback(i2s);
239*4882a593Smuzhiyun else
240*4882a593Smuzhiyun tegra30_i2s_start_capture(i2s);
241*4882a593Smuzhiyun break;
242*4882a593Smuzhiyun case SNDRV_PCM_TRIGGER_STOP:
243*4882a593Smuzhiyun case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
244*4882a593Smuzhiyun case SNDRV_PCM_TRIGGER_SUSPEND:
245*4882a593Smuzhiyun if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
246*4882a593Smuzhiyun tegra30_i2s_stop_playback(i2s);
247*4882a593Smuzhiyun else
248*4882a593Smuzhiyun tegra30_i2s_stop_capture(i2s);
249*4882a593Smuzhiyun break;
250*4882a593Smuzhiyun default:
251*4882a593Smuzhiyun return -EINVAL;
252*4882a593Smuzhiyun }
253*4882a593Smuzhiyun
254*4882a593Smuzhiyun return 0;
255*4882a593Smuzhiyun }
256*4882a593Smuzhiyun
tegra30_i2s_set_tdm(struct snd_soc_dai * dai,unsigned int tx_mask,unsigned int rx_mask,int slots,int slot_width)257*4882a593Smuzhiyun static int tegra30_i2s_set_tdm(struct snd_soc_dai *dai,
258*4882a593Smuzhiyun unsigned int tx_mask, unsigned int rx_mask,
259*4882a593Smuzhiyun int slots, int slot_width)
260*4882a593Smuzhiyun {
261*4882a593Smuzhiyun struct tegra30_i2s *i2s = snd_soc_dai_get_drvdata(dai);
262*4882a593Smuzhiyun unsigned int mask, val;
263*4882a593Smuzhiyun
264*4882a593Smuzhiyun dev_dbg(dai->dev, "%s: txmask=0x%08x rxmask=0x%08x slots=%d width=%d\n",
265*4882a593Smuzhiyun __func__, tx_mask, rx_mask, slots, slot_width);
266*4882a593Smuzhiyun
267*4882a593Smuzhiyun mask = TEGRA30_I2S_SLOT_CTRL_TOTAL_SLOTS_MASK |
268*4882a593Smuzhiyun TEGRA30_I2S_SLOT_CTRL_RX_SLOT_ENABLES_MASK |
269*4882a593Smuzhiyun TEGRA30_I2S_SLOT_CTRL_TX_SLOT_ENABLES_MASK;
270*4882a593Smuzhiyun
271*4882a593Smuzhiyun val = (tx_mask << TEGRA30_I2S_SLOT_CTRL_TX_SLOT_ENABLES_SHIFT) |
272*4882a593Smuzhiyun (rx_mask << TEGRA30_I2S_SLOT_CTRL_RX_SLOT_ENABLES_SHIFT) |
273*4882a593Smuzhiyun ((slots - 1) << TEGRA30_I2S_SLOT_CTRL_TOTAL_SLOTS_SHIFT);
274*4882a593Smuzhiyun
275*4882a593Smuzhiyun pm_runtime_get_sync(dai->dev);
276*4882a593Smuzhiyun regmap_update_bits(i2s->regmap, TEGRA30_I2S_SLOT_CTRL, mask, val);
277*4882a593Smuzhiyun /* set the fsync width to minimum of 1 clock width */
278*4882a593Smuzhiyun regmap_update_bits(i2s->regmap, TEGRA30_I2S_CH_CTRL,
279*4882a593Smuzhiyun TEGRA30_I2S_CH_CTRL_FSYNC_WIDTH_MASK, 0x0);
280*4882a593Smuzhiyun pm_runtime_put(dai->dev);
281*4882a593Smuzhiyun
282*4882a593Smuzhiyun return 0;
283*4882a593Smuzhiyun }
284*4882a593Smuzhiyun
tegra30_i2s_probe(struct snd_soc_dai * dai)285*4882a593Smuzhiyun static int tegra30_i2s_probe(struct snd_soc_dai *dai)
286*4882a593Smuzhiyun {
287*4882a593Smuzhiyun struct tegra30_i2s *i2s = snd_soc_dai_get_drvdata(dai);
288*4882a593Smuzhiyun
289*4882a593Smuzhiyun dai->capture_dma_data = &i2s->capture_dma_data;
290*4882a593Smuzhiyun dai->playback_dma_data = &i2s->playback_dma_data;
291*4882a593Smuzhiyun
292*4882a593Smuzhiyun return 0;
293*4882a593Smuzhiyun }
294*4882a593Smuzhiyun
295*4882a593Smuzhiyun static const struct snd_soc_dai_ops tegra30_i2s_dai_ops = {
296*4882a593Smuzhiyun .set_fmt = tegra30_i2s_set_fmt,
297*4882a593Smuzhiyun .hw_params = tegra30_i2s_hw_params,
298*4882a593Smuzhiyun .trigger = tegra30_i2s_trigger,
299*4882a593Smuzhiyun .set_tdm_slot = tegra30_i2s_set_tdm,
300*4882a593Smuzhiyun };
301*4882a593Smuzhiyun
302*4882a593Smuzhiyun static const struct snd_soc_dai_driver tegra30_i2s_dai_template = {
303*4882a593Smuzhiyun .probe = tegra30_i2s_probe,
304*4882a593Smuzhiyun .playback = {
305*4882a593Smuzhiyun .stream_name = "Playback",
306*4882a593Smuzhiyun .channels_min = 2,
307*4882a593Smuzhiyun .channels_max = 2,
308*4882a593Smuzhiyun .rates = SNDRV_PCM_RATE_8000_96000,
309*4882a593Smuzhiyun .formats = SNDRV_PCM_FMTBIT_S16_LE,
310*4882a593Smuzhiyun },
311*4882a593Smuzhiyun .capture = {
312*4882a593Smuzhiyun .stream_name = "Capture",
313*4882a593Smuzhiyun .channels_min = 2,
314*4882a593Smuzhiyun .channels_max = 2,
315*4882a593Smuzhiyun .rates = SNDRV_PCM_RATE_8000_96000,
316*4882a593Smuzhiyun .formats = SNDRV_PCM_FMTBIT_S16_LE,
317*4882a593Smuzhiyun },
318*4882a593Smuzhiyun .ops = &tegra30_i2s_dai_ops,
319*4882a593Smuzhiyun .symmetric_rates = 1,
320*4882a593Smuzhiyun };
321*4882a593Smuzhiyun
322*4882a593Smuzhiyun static const struct snd_soc_component_driver tegra30_i2s_component = {
323*4882a593Smuzhiyun .name = DRV_NAME,
324*4882a593Smuzhiyun };
325*4882a593Smuzhiyun
tegra30_i2s_wr_rd_reg(struct device * dev,unsigned int reg)326*4882a593Smuzhiyun static bool tegra30_i2s_wr_rd_reg(struct device *dev, unsigned int reg)
327*4882a593Smuzhiyun {
328*4882a593Smuzhiyun switch (reg) {
329*4882a593Smuzhiyun case TEGRA30_I2S_CTRL:
330*4882a593Smuzhiyun case TEGRA30_I2S_TIMING:
331*4882a593Smuzhiyun case TEGRA30_I2S_OFFSET:
332*4882a593Smuzhiyun case TEGRA30_I2S_CH_CTRL:
333*4882a593Smuzhiyun case TEGRA30_I2S_SLOT_CTRL:
334*4882a593Smuzhiyun case TEGRA30_I2S_CIF_RX_CTRL:
335*4882a593Smuzhiyun case TEGRA30_I2S_CIF_TX_CTRL:
336*4882a593Smuzhiyun case TEGRA30_I2S_FLOWCTL:
337*4882a593Smuzhiyun case TEGRA30_I2S_TX_STEP:
338*4882a593Smuzhiyun case TEGRA30_I2S_FLOW_STATUS:
339*4882a593Smuzhiyun case TEGRA30_I2S_FLOW_TOTAL:
340*4882a593Smuzhiyun case TEGRA30_I2S_FLOW_OVER:
341*4882a593Smuzhiyun case TEGRA30_I2S_FLOW_UNDER:
342*4882a593Smuzhiyun case TEGRA30_I2S_LCOEF_1_4_0:
343*4882a593Smuzhiyun case TEGRA30_I2S_LCOEF_1_4_1:
344*4882a593Smuzhiyun case TEGRA30_I2S_LCOEF_1_4_2:
345*4882a593Smuzhiyun case TEGRA30_I2S_LCOEF_1_4_3:
346*4882a593Smuzhiyun case TEGRA30_I2S_LCOEF_1_4_4:
347*4882a593Smuzhiyun case TEGRA30_I2S_LCOEF_1_4_5:
348*4882a593Smuzhiyun case TEGRA30_I2S_LCOEF_2_4_0:
349*4882a593Smuzhiyun case TEGRA30_I2S_LCOEF_2_4_1:
350*4882a593Smuzhiyun case TEGRA30_I2S_LCOEF_2_4_2:
351*4882a593Smuzhiyun return true;
352*4882a593Smuzhiyun default:
353*4882a593Smuzhiyun return false;
354*4882a593Smuzhiyun }
355*4882a593Smuzhiyun }
356*4882a593Smuzhiyun
tegra30_i2s_volatile_reg(struct device * dev,unsigned int reg)357*4882a593Smuzhiyun static bool tegra30_i2s_volatile_reg(struct device *dev, unsigned int reg)
358*4882a593Smuzhiyun {
359*4882a593Smuzhiyun switch (reg) {
360*4882a593Smuzhiyun case TEGRA30_I2S_FLOW_STATUS:
361*4882a593Smuzhiyun case TEGRA30_I2S_FLOW_TOTAL:
362*4882a593Smuzhiyun case TEGRA30_I2S_FLOW_OVER:
363*4882a593Smuzhiyun case TEGRA30_I2S_FLOW_UNDER:
364*4882a593Smuzhiyun return true;
365*4882a593Smuzhiyun default:
366*4882a593Smuzhiyun return false;
367*4882a593Smuzhiyun }
368*4882a593Smuzhiyun }
369*4882a593Smuzhiyun
370*4882a593Smuzhiyun static const struct regmap_config tegra30_i2s_regmap_config = {
371*4882a593Smuzhiyun .reg_bits = 32,
372*4882a593Smuzhiyun .reg_stride = 4,
373*4882a593Smuzhiyun .val_bits = 32,
374*4882a593Smuzhiyun .max_register = TEGRA30_I2S_LCOEF_2_4_2,
375*4882a593Smuzhiyun .writeable_reg = tegra30_i2s_wr_rd_reg,
376*4882a593Smuzhiyun .readable_reg = tegra30_i2s_wr_rd_reg,
377*4882a593Smuzhiyun .volatile_reg = tegra30_i2s_volatile_reg,
378*4882a593Smuzhiyun .cache_type = REGCACHE_FLAT,
379*4882a593Smuzhiyun };
380*4882a593Smuzhiyun
381*4882a593Smuzhiyun static const struct tegra30_i2s_soc_data tegra30_i2s_config = {
382*4882a593Smuzhiyun .set_audio_cif = tegra30_ahub_set_cif,
383*4882a593Smuzhiyun };
384*4882a593Smuzhiyun
385*4882a593Smuzhiyun static const struct tegra30_i2s_soc_data tegra124_i2s_config = {
386*4882a593Smuzhiyun .set_audio_cif = tegra124_ahub_set_cif,
387*4882a593Smuzhiyun };
388*4882a593Smuzhiyun
389*4882a593Smuzhiyun static const struct of_device_id tegra30_i2s_of_match[] = {
390*4882a593Smuzhiyun { .compatible = "nvidia,tegra124-i2s", .data = &tegra124_i2s_config },
391*4882a593Smuzhiyun { .compatible = "nvidia,tegra30-i2s", .data = &tegra30_i2s_config },
392*4882a593Smuzhiyun {},
393*4882a593Smuzhiyun };
394*4882a593Smuzhiyun
tegra30_i2s_platform_probe(struct platform_device * pdev)395*4882a593Smuzhiyun static int tegra30_i2s_platform_probe(struct platform_device *pdev)
396*4882a593Smuzhiyun {
397*4882a593Smuzhiyun struct tegra30_i2s *i2s;
398*4882a593Smuzhiyun const struct of_device_id *match;
399*4882a593Smuzhiyun u32 cif_ids[2];
400*4882a593Smuzhiyun void __iomem *regs;
401*4882a593Smuzhiyun int ret;
402*4882a593Smuzhiyun
403*4882a593Smuzhiyun i2s = devm_kzalloc(&pdev->dev, sizeof(struct tegra30_i2s), GFP_KERNEL);
404*4882a593Smuzhiyun if (!i2s) {
405*4882a593Smuzhiyun ret = -ENOMEM;
406*4882a593Smuzhiyun goto err;
407*4882a593Smuzhiyun }
408*4882a593Smuzhiyun dev_set_drvdata(&pdev->dev, i2s);
409*4882a593Smuzhiyun
410*4882a593Smuzhiyun match = of_match_device(tegra30_i2s_of_match, &pdev->dev);
411*4882a593Smuzhiyun if (!match) {
412*4882a593Smuzhiyun dev_err(&pdev->dev, "Error: No device match found\n");
413*4882a593Smuzhiyun ret = -ENODEV;
414*4882a593Smuzhiyun goto err;
415*4882a593Smuzhiyun }
416*4882a593Smuzhiyun i2s->soc_data = (struct tegra30_i2s_soc_data *)match->data;
417*4882a593Smuzhiyun
418*4882a593Smuzhiyun i2s->dai = tegra30_i2s_dai_template;
419*4882a593Smuzhiyun i2s->dai.name = dev_name(&pdev->dev);
420*4882a593Smuzhiyun
421*4882a593Smuzhiyun ret = of_property_read_u32_array(pdev->dev.of_node,
422*4882a593Smuzhiyun "nvidia,ahub-cif-ids", cif_ids,
423*4882a593Smuzhiyun ARRAY_SIZE(cif_ids));
424*4882a593Smuzhiyun if (ret < 0)
425*4882a593Smuzhiyun goto err;
426*4882a593Smuzhiyun
427*4882a593Smuzhiyun i2s->playback_i2s_cif = cif_ids[0];
428*4882a593Smuzhiyun i2s->capture_i2s_cif = cif_ids[1];
429*4882a593Smuzhiyun
430*4882a593Smuzhiyun i2s->clk_i2s = clk_get(&pdev->dev, NULL);
431*4882a593Smuzhiyun if (IS_ERR(i2s->clk_i2s)) {
432*4882a593Smuzhiyun dev_err(&pdev->dev, "Can't retrieve i2s clock\n");
433*4882a593Smuzhiyun ret = PTR_ERR(i2s->clk_i2s);
434*4882a593Smuzhiyun goto err;
435*4882a593Smuzhiyun }
436*4882a593Smuzhiyun
437*4882a593Smuzhiyun regs = devm_platform_ioremap_resource(pdev, 0);
438*4882a593Smuzhiyun if (IS_ERR(regs)) {
439*4882a593Smuzhiyun ret = PTR_ERR(regs);
440*4882a593Smuzhiyun goto err_clk_put;
441*4882a593Smuzhiyun }
442*4882a593Smuzhiyun
443*4882a593Smuzhiyun i2s->regmap = devm_regmap_init_mmio(&pdev->dev, regs,
444*4882a593Smuzhiyun &tegra30_i2s_regmap_config);
445*4882a593Smuzhiyun if (IS_ERR(i2s->regmap)) {
446*4882a593Smuzhiyun dev_err(&pdev->dev, "regmap init failed\n");
447*4882a593Smuzhiyun ret = PTR_ERR(i2s->regmap);
448*4882a593Smuzhiyun goto err_clk_put;
449*4882a593Smuzhiyun }
450*4882a593Smuzhiyun regcache_cache_only(i2s->regmap, true);
451*4882a593Smuzhiyun
452*4882a593Smuzhiyun pm_runtime_enable(&pdev->dev);
453*4882a593Smuzhiyun if (!pm_runtime_enabled(&pdev->dev)) {
454*4882a593Smuzhiyun ret = tegra30_i2s_runtime_resume(&pdev->dev);
455*4882a593Smuzhiyun if (ret)
456*4882a593Smuzhiyun goto err_pm_disable;
457*4882a593Smuzhiyun }
458*4882a593Smuzhiyun
459*4882a593Smuzhiyun i2s->playback_dma_data.addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
460*4882a593Smuzhiyun i2s->playback_dma_data.maxburst = 4;
461*4882a593Smuzhiyun ret = tegra30_ahub_allocate_tx_fifo(&i2s->playback_fifo_cif,
462*4882a593Smuzhiyun i2s->playback_dma_chan,
463*4882a593Smuzhiyun sizeof(i2s->playback_dma_chan),
464*4882a593Smuzhiyun &i2s->playback_dma_data.addr);
465*4882a593Smuzhiyun if (ret) {
466*4882a593Smuzhiyun dev_err(&pdev->dev, "Could not alloc TX FIFO: %d\n", ret);
467*4882a593Smuzhiyun goto err_suspend;
468*4882a593Smuzhiyun }
469*4882a593Smuzhiyun ret = tegra30_ahub_set_rx_cif_source(i2s->playback_i2s_cif,
470*4882a593Smuzhiyun i2s->playback_fifo_cif);
471*4882a593Smuzhiyun if (ret) {
472*4882a593Smuzhiyun dev_err(&pdev->dev, "Could not route TX FIFO: %d\n", ret);
473*4882a593Smuzhiyun goto err_free_tx_fifo;
474*4882a593Smuzhiyun }
475*4882a593Smuzhiyun
476*4882a593Smuzhiyun i2s->capture_dma_data.addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
477*4882a593Smuzhiyun i2s->capture_dma_data.maxburst = 4;
478*4882a593Smuzhiyun ret = tegra30_ahub_allocate_rx_fifo(&i2s->capture_fifo_cif,
479*4882a593Smuzhiyun i2s->capture_dma_chan,
480*4882a593Smuzhiyun sizeof(i2s->capture_dma_chan),
481*4882a593Smuzhiyun &i2s->capture_dma_data.addr);
482*4882a593Smuzhiyun if (ret) {
483*4882a593Smuzhiyun dev_err(&pdev->dev, "Could not alloc RX FIFO: %d\n", ret);
484*4882a593Smuzhiyun goto err_unroute_tx_fifo;
485*4882a593Smuzhiyun }
486*4882a593Smuzhiyun ret = tegra30_ahub_set_rx_cif_source(i2s->capture_fifo_cif,
487*4882a593Smuzhiyun i2s->capture_i2s_cif);
488*4882a593Smuzhiyun if (ret) {
489*4882a593Smuzhiyun dev_err(&pdev->dev, "Could not route TX FIFO: %d\n", ret);
490*4882a593Smuzhiyun goto err_free_rx_fifo;
491*4882a593Smuzhiyun }
492*4882a593Smuzhiyun
493*4882a593Smuzhiyun ret = snd_soc_register_component(&pdev->dev, &tegra30_i2s_component,
494*4882a593Smuzhiyun &i2s->dai, 1);
495*4882a593Smuzhiyun if (ret) {
496*4882a593Smuzhiyun dev_err(&pdev->dev, "Could not register DAI: %d\n", ret);
497*4882a593Smuzhiyun ret = -ENOMEM;
498*4882a593Smuzhiyun goto err_unroute_rx_fifo;
499*4882a593Smuzhiyun }
500*4882a593Smuzhiyun
501*4882a593Smuzhiyun ret = tegra_pcm_platform_register_with_chan_names(&pdev->dev,
502*4882a593Smuzhiyun &i2s->dma_config, i2s->playback_dma_chan,
503*4882a593Smuzhiyun i2s->capture_dma_chan);
504*4882a593Smuzhiyun if (ret) {
505*4882a593Smuzhiyun dev_err(&pdev->dev, "Could not register PCM: %d\n", ret);
506*4882a593Smuzhiyun goto err_unregister_component;
507*4882a593Smuzhiyun }
508*4882a593Smuzhiyun
509*4882a593Smuzhiyun return 0;
510*4882a593Smuzhiyun
511*4882a593Smuzhiyun err_unregister_component:
512*4882a593Smuzhiyun snd_soc_unregister_component(&pdev->dev);
513*4882a593Smuzhiyun err_unroute_rx_fifo:
514*4882a593Smuzhiyun tegra30_ahub_unset_rx_cif_source(i2s->capture_fifo_cif);
515*4882a593Smuzhiyun err_free_rx_fifo:
516*4882a593Smuzhiyun tegra30_ahub_free_rx_fifo(i2s->capture_fifo_cif);
517*4882a593Smuzhiyun err_unroute_tx_fifo:
518*4882a593Smuzhiyun tegra30_ahub_unset_rx_cif_source(i2s->playback_i2s_cif);
519*4882a593Smuzhiyun err_free_tx_fifo:
520*4882a593Smuzhiyun tegra30_ahub_free_tx_fifo(i2s->playback_fifo_cif);
521*4882a593Smuzhiyun err_suspend:
522*4882a593Smuzhiyun if (!pm_runtime_status_suspended(&pdev->dev))
523*4882a593Smuzhiyun tegra30_i2s_runtime_suspend(&pdev->dev);
524*4882a593Smuzhiyun err_pm_disable:
525*4882a593Smuzhiyun pm_runtime_disable(&pdev->dev);
526*4882a593Smuzhiyun err_clk_put:
527*4882a593Smuzhiyun clk_put(i2s->clk_i2s);
528*4882a593Smuzhiyun err:
529*4882a593Smuzhiyun return ret;
530*4882a593Smuzhiyun }
531*4882a593Smuzhiyun
tegra30_i2s_platform_remove(struct platform_device * pdev)532*4882a593Smuzhiyun static int tegra30_i2s_platform_remove(struct platform_device *pdev)
533*4882a593Smuzhiyun {
534*4882a593Smuzhiyun struct tegra30_i2s *i2s = dev_get_drvdata(&pdev->dev);
535*4882a593Smuzhiyun
536*4882a593Smuzhiyun pm_runtime_disable(&pdev->dev);
537*4882a593Smuzhiyun if (!pm_runtime_status_suspended(&pdev->dev))
538*4882a593Smuzhiyun tegra30_i2s_runtime_suspend(&pdev->dev);
539*4882a593Smuzhiyun
540*4882a593Smuzhiyun tegra_pcm_platform_unregister(&pdev->dev);
541*4882a593Smuzhiyun snd_soc_unregister_component(&pdev->dev);
542*4882a593Smuzhiyun
543*4882a593Smuzhiyun tegra30_ahub_unset_rx_cif_source(i2s->capture_fifo_cif);
544*4882a593Smuzhiyun tegra30_ahub_free_rx_fifo(i2s->capture_fifo_cif);
545*4882a593Smuzhiyun
546*4882a593Smuzhiyun tegra30_ahub_unset_rx_cif_source(i2s->playback_i2s_cif);
547*4882a593Smuzhiyun tegra30_ahub_free_tx_fifo(i2s->playback_fifo_cif);
548*4882a593Smuzhiyun
549*4882a593Smuzhiyun clk_put(i2s->clk_i2s);
550*4882a593Smuzhiyun
551*4882a593Smuzhiyun return 0;
552*4882a593Smuzhiyun }
553*4882a593Smuzhiyun
554*4882a593Smuzhiyun #ifdef CONFIG_PM_SLEEP
tegra30_i2s_suspend(struct device * dev)555*4882a593Smuzhiyun static int tegra30_i2s_suspend(struct device *dev)
556*4882a593Smuzhiyun {
557*4882a593Smuzhiyun struct tegra30_i2s *i2s = dev_get_drvdata(dev);
558*4882a593Smuzhiyun
559*4882a593Smuzhiyun regcache_mark_dirty(i2s->regmap);
560*4882a593Smuzhiyun
561*4882a593Smuzhiyun return 0;
562*4882a593Smuzhiyun }
563*4882a593Smuzhiyun
tegra30_i2s_resume(struct device * dev)564*4882a593Smuzhiyun static int tegra30_i2s_resume(struct device *dev)
565*4882a593Smuzhiyun {
566*4882a593Smuzhiyun struct tegra30_i2s *i2s = dev_get_drvdata(dev);
567*4882a593Smuzhiyun int ret;
568*4882a593Smuzhiyun
569*4882a593Smuzhiyun ret = pm_runtime_get_sync(dev);
570*4882a593Smuzhiyun if (ret < 0) {
571*4882a593Smuzhiyun pm_runtime_put(dev);
572*4882a593Smuzhiyun return ret;
573*4882a593Smuzhiyun }
574*4882a593Smuzhiyun ret = regcache_sync(i2s->regmap);
575*4882a593Smuzhiyun pm_runtime_put(dev);
576*4882a593Smuzhiyun
577*4882a593Smuzhiyun return ret;
578*4882a593Smuzhiyun }
579*4882a593Smuzhiyun #endif
580*4882a593Smuzhiyun
581*4882a593Smuzhiyun static const struct dev_pm_ops tegra30_i2s_pm_ops = {
582*4882a593Smuzhiyun SET_RUNTIME_PM_OPS(tegra30_i2s_runtime_suspend,
583*4882a593Smuzhiyun tegra30_i2s_runtime_resume, NULL)
584*4882a593Smuzhiyun SET_SYSTEM_SLEEP_PM_OPS(tegra30_i2s_suspend, tegra30_i2s_resume)
585*4882a593Smuzhiyun };
586*4882a593Smuzhiyun
587*4882a593Smuzhiyun static struct platform_driver tegra30_i2s_driver = {
588*4882a593Smuzhiyun .driver = {
589*4882a593Smuzhiyun .name = DRV_NAME,
590*4882a593Smuzhiyun .of_match_table = tegra30_i2s_of_match,
591*4882a593Smuzhiyun .pm = &tegra30_i2s_pm_ops,
592*4882a593Smuzhiyun },
593*4882a593Smuzhiyun .probe = tegra30_i2s_platform_probe,
594*4882a593Smuzhiyun .remove = tegra30_i2s_platform_remove,
595*4882a593Smuzhiyun };
596*4882a593Smuzhiyun module_platform_driver(tegra30_i2s_driver);
597*4882a593Smuzhiyun
598*4882a593Smuzhiyun MODULE_AUTHOR("Stephen Warren <swarren@nvidia.com>");
599*4882a593Smuzhiyun MODULE_DESCRIPTION("Tegra30 I2S ASoC driver");
600*4882a593Smuzhiyun MODULE_LICENSE("GPL");
601*4882a593Smuzhiyun MODULE_ALIAS("platform:" DRV_NAME);
602*4882a593Smuzhiyun MODULE_DEVICE_TABLE(of, tegra30_i2s_of_match);
603