xref: /OK3568_Linux_fs/kernel/sound/soc/tegra/tegra30_i2s.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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