xref: /openbmc/linux/sound/soc/tegra/tegra30_ahub.c (revision be944d42)
1be944d42SStephen Warren /*
2be944d42SStephen Warren  * tegra30_ahub.c - Tegra30 AHUB driver
3be944d42SStephen Warren  *
4be944d42SStephen Warren  * Copyright (c) 2011,2012, NVIDIA CORPORATION.  All rights reserved.
5be944d42SStephen Warren  *
6be944d42SStephen Warren  * This program is free software; you can redistribute it and/or modify it
7be944d42SStephen Warren  * under the terms and conditions of the GNU General Public License,
8be944d42SStephen Warren  * version 2, as published by the Free Software Foundation.
9be944d42SStephen Warren  *
10be944d42SStephen Warren  * This program is distributed in the hope it will be useful, but WITHOUT
11be944d42SStephen Warren  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12be944d42SStephen Warren  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
13be944d42SStephen Warren  * more details.
14be944d42SStephen Warren  *
15be944d42SStephen Warren  * You should have received a copy of the GNU General Public License
16be944d42SStephen Warren  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
17be944d42SStephen Warren  */
18be944d42SStephen Warren 
19be944d42SStephen Warren #include <linux/clk.h>
20be944d42SStephen Warren #include <linux/device.h>
21be944d42SStephen Warren #include <linux/io.h>
22be944d42SStephen Warren #include <linux/module.h>
23be944d42SStephen Warren #include <linux/of_platform.h>
24be944d42SStephen Warren #include <linux/platform_device.h>
25be944d42SStephen Warren #include <linux/pm_runtime.h>
26be944d42SStephen Warren #include <linux/regmap.h>
27be944d42SStephen Warren #include <linux/slab.h>
28be944d42SStephen Warren #include <mach/clk.h>
29be944d42SStephen Warren #include <mach/dma.h>
30be944d42SStephen Warren #include <sound/soc.h>
31be944d42SStephen Warren #include "tegra30_ahub.h"
32be944d42SStephen Warren 
33be944d42SStephen Warren #define DRV_NAME "tegra30-ahub"
34be944d42SStephen Warren 
35be944d42SStephen Warren static struct tegra30_ahub *ahub;
36be944d42SStephen Warren 
37be944d42SStephen Warren static inline void tegra30_apbif_write(u32 reg, u32 val)
38be944d42SStephen Warren {
39be944d42SStephen Warren 	regmap_write(ahub->regmap_apbif, reg, val);
40be944d42SStephen Warren }
41be944d42SStephen Warren 
42be944d42SStephen Warren static inline u32 tegra30_apbif_read(u32 reg)
43be944d42SStephen Warren {
44be944d42SStephen Warren 	u32 val;
45be944d42SStephen Warren 	regmap_read(ahub->regmap_apbif, reg, &val);
46be944d42SStephen Warren 	return val;
47be944d42SStephen Warren }
48be944d42SStephen Warren 
49be944d42SStephen Warren static inline void tegra30_audio_write(u32 reg, u32 val)
50be944d42SStephen Warren {
51be944d42SStephen Warren 	regmap_write(ahub->regmap_ahub, reg, val);
52be944d42SStephen Warren }
53be944d42SStephen Warren 
54be944d42SStephen Warren static int tegra30_ahub_runtime_suspend(struct device *dev)
55be944d42SStephen Warren {
56be944d42SStephen Warren 	regcache_cache_only(ahub->regmap_apbif, true);
57be944d42SStephen Warren 	regcache_cache_only(ahub->regmap_ahub, true);
58be944d42SStephen Warren 
59be944d42SStephen Warren 	clk_disable(ahub->clk_apbif);
60be944d42SStephen Warren 	clk_disable(ahub->clk_d_audio);
61be944d42SStephen Warren 
62be944d42SStephen Warren 	return 0;
63be944d42SStephen Warren }
64be944d42SStephen Warren 
65be944d42SStephen Warren /*
66be944d42SStephen Warren  * clk_apbif isn't required for an I2S<->I2S configuration where no PCM data
67be944d42SStephen Warren  * is read from or sent to memory. However, that's not something the rest of
68be944d42SStephen Warren  * the driver supports right now, so we'll just treat the two clocks as one
69be944d42SStephen Warren  * for now.
70be944d42SStephen Warren  *
71be944d42SStephen Warren  * These functions should not be a plain ref-count. Instead, each active stream
72be944d42SStephen Warren  * contributes some requirement to the minimum clock rate, so starting or
73be944d42SStephen Warren  * stopping streams should dynamically adjust the clock as required.  However,
74be944d42SStephen Warren  * this is not yet implemented.
75be944d42SStephen Warren  */
76be944d42SStephen Warren static int tegra30_ahub_runtime_resume(struct device *dev)
77be944d42SStephen Warren {
78be944d42SStephen Warren 	int ret;
79be944d42SStephen Warren 
80be944d42SStephen Warren 	ret = clk_enable(ahub->clk_d_audio);
81be944d42SStephen Warren 	if (ret) {
82be944d42SStephen Warren 		dev_err(dev, "clk_enable d_audio failed: %d\n", ret);
83be944d42SStephen Warren 		return ret;
84be944d42SStephen Warren 	}
85be944d42SStephen Warren 	ret = clk_enable(ahub->clk_apbif);
86be944d42SStephen Warren 	if (ret) {
87be944d42SStephen Warren 		dev_err(dev, "clk_enable apbif failed: %d\n", ret);
88be944d42SStephen Warren 		clk_disable(ahub->clk_d_audio);
89be944d42SStephen Warren 		return ret;
90be944d42SStephen Warren 	}
91be944d42SStephen Warren 
92be944d42SStephen Warren 	regcache_cache_only(ahub->regmap_apbif, false);
93be944d42SStephen Warren 	regcache_cache_only(ahub->regmap_ahub, false);
94be944d42SStephen Warren 
95be944d42SStephen Warren 	return 0;
96be944d42SStephen Warren }
97be944d42SStephen Warren 
98be944d42SStephen Warren int tegra30_ahub_allocate_rx_fifo(enum tegra30_ahub_rxcif *rxcif,
99be944d42SStephen Warren 				  unsigned long *fiforeg,
100be944d42SStephen Warren 				  unsigned long *reqsel)
101be944d42SStephen Warren {
102be944d42SStephen Warren 	int channel;
103be944d42SStephen Warren 	u32 reg, val;
104be944d42SStephen Warren 
105be944d42SStephen Warren 	channel = find_first_zero_bit(ahub->rx_usage,
106be944d42SStephen Warren 				      TEGRA30_AHUB_CHANNEL_CTRL_COUNT);
107be944d42SStephen Warren 	if (channel >= TEGRA30_AHUB_CHANNEL_CTRL_COUNT)
108be944d42SStephen Warren 		return -EBUSY;
109be944d42SStephen Warren 
110be944d42SStephen Warren 	__set_bit(channel, ahub->rx_usage);
111be944d42SStephen Warren 
112be944d42SStephen Warren 	*rxcif = TEGRA30_AHUB_RXCIF_APBIF_RX0 + channel;
113be944d42SStephen Warren 	*fiforeg = ahub->apbif_addr + TEGRA30_AHUB_CHANNEL_RXFIFO +
114be944d42SStephen Warren 		   (channel * TEGRA30_AHUB_CHANNEL_RXFIFO_STRIDE);
115be944d42SStephen Warren 	*reqsel = ahub->dma_sel + channel;
116be944d42SStephen Warren 
117be944d42SStephen Warren 	reg = TEGRA30_AHUB_CHANNEL_CTRL +
118be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_CHANNEL_CTRL_STRIDE);
119be944d42SStephen Warren 	val = tegra30_apbif_read(reg);
120be944d42SStephen Warren 	val &= ~(TEGRA30_AHUB_CHANNEL_CTRL_RX_THRESHOLD_MASK |
121be944d42SStephen Warren 		 TEGRA30_AHUB_CHANNEL_CTRL_RX_PACK_MASK);
122be944d42SStephen Warren 	val |= (7 << TEGRA30_AHUB_CHANNEL_CTRL_RX_THRESHOLD_SHIFT) |
123be944d42SStephen Warren 	       TEGRA30_AHUB_CHANNEL_CTRL_RX_PACK_EN |
124be944d42SStephen Warren 	       TEGRA30_AHUB_CHANNEL_CTRL_RX_PACK_16;
125be944d42SStephen Warren 	tegra30_apbif_write(reg, val);
126be944d42SStephen Warren 
127be944d42SStephen Warren 	reg = TEGRA30_AHUB_CIF_RX_CTRL +
128be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_CIF_RX_CTRL_STRIDE);
129be944d42SStephen Warren 	val = (0 << TEGRA30_AUDIOCIF_CTRL_FIFO_THRESHOLD_SHIFT) |
130be944d42SStephen Warren 	      (1 << TEGRA30_AUDIOCIF_CTRL_AUDIO_CHANNELS_SHIFT) |
131be944d42SStephen Warren 	      (1 << TEGRA30_AUDIOCIF_CTRL_CLIENT_CHANNELS_SHIFT) |
132be944d42SStephen Warren 	      TEGRA30_AUDIOCIF_CTRL_AUDIO_BITS_16 |
133be944d42SStephen Warren 	      TEGRA30_AUDIOCIF_CTRL_CLIENT_BITS_16 |
134be944d42SStephen Warren 	      TEGRA30_AUDIOCIF_CTRL_DIRECTION_RX;
135be944d42SStephen Warren 	tegra30_apbif_write(reg, val);
136be944d42SStephen Warren 
137be944d42SStephen Warren 	return 0;
138be944d42SStephen Warren }
139be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_allocate_rx_fifo);
140be944d42SStephen Warren 
141be944d42SStephen Warren int tegra30_ahub_enable_rx_fifo(enum tegra30_ahub_rxcif rxcif)
142be944d42SStephen Warren {
143be944d42SStephen Warren 	int channel = rxcif - TEGRA30_AHUB_RXCIF_APBIF_RX0;
144be944d42SStephen Warren 	int reg, val;
145be944d42SStephen Warren 
146be944d42SStephen Warren 	reg = TEGRA30_AHUB_CHANNEL_CTRL +
147be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_CHANNEL_CTRL_STRIDE);
148be944d42SStephen Warren 	val = tegra30_apbif_read(reg);
149be944d42SStephen Warren 	val |= TEGRA30_AHUB_CHANNEL_CTRL_RX_EN;
150be944d42SStephen Warren 	tegra30_apbif_write(reg, val);
151be944d42SStephen Warren 
152be944d42SStephen Warren 	return 0;
153be944d42SStephen Warren }
154be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_enable_rx_fifo);
155be944d42SStephen Warren 
156be944d42SStephen Warren int tegra30_ahub_disable_rx_fifo(enum tegra30_ahub_rxcif rxcif)
157be944d42SStephen Warren {
158be944d42SStephen Warren 	int channel = rxcif - TEGRA30_AHUB_RXCIF_APBIF_RX0;
159be944d42SStephen Warren 	int reg, val;
160be944d42SStephen Warren 
161be944d42SStephen Warren 	reg = TEGRA30_AHUB_CHANNEL_CTRL +
162be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_CHANNEL_CTRL_STRIDE);
163be944d42SStephen Warren 	val = tegra30_apbif_read(reg);
164be944d42SStephen Warren 	val &= ~TEGRA30_AHUB_CHANNEL_CTRL_RX_EN;
165be944d42SStephen Warren 	tegra30_apbif_write(reg, val);
166be944d42SStephen Warren 
167be944d42SStephen Warren 	return 0;
168be944d42SStephen Warren }
169be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_disable_rx_fifo);
170be944d42SStephen Warren 
171be944d42SStephen Warren int tegra30_ahub_free_rx_fifo(enum tegra30_ahub_rxcif rxcif)
172be944d42SStephen Warren {
173be944d42SStephen Warren 	int channel = rxcif - TEGRA30_AHUB_RXCIF_APBIF_RX0;
174be944d42SStephen Warren 
175be944d42SStephen Warren 	__clear_bit(channel, ahub->rx_usage);
176be944d42SStephen Warren 
177be944d42SStephen Warren 	return 0;
178be944d42SStephen Warren }
179be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_free_rx_fifo);
180be944d42SStephen Warren 
181be944d42SStephen Warren int tegra30_ahub_allocate_tx_fifo(enum tegra30_ahub_txcif *txcif,
182be944d42SStephen Warren 				  unsigned long *fiforeg,
183be944d42SStephen Warren 				  unsigned long *reqsel)
184be944d42SStephen Warren {
185be944d42SStephen Warren 	int channel;
186be944d42SStephen Warren 	u32 reg, val;
187be944d42SStephen Warren 
188be944d42SStephen Warren 	channel = find_first_zero_bit(ahub->tx_usage,
189be944d42SStephen Warren 				      TEGRA30_AHUB_CHANNEL_CTRL_COUNT);
190be944d42SStephen Warren 	if (channel >= TEGRA30_AHUB_CHANNEL_CTRL_COUNT)
191be944d42SStephen Warren 		return -EBUSY;
192be944d42SStephen Warren 
193be944d42SStephen Warren 	__set_bit(channel, ahub->tx_usage);
194be944d42SStephen Warren 
195be944d42SStephen Warren 	*txcif = TEGRA30_AHUB_TXCIF_APBIF_TX0 + channel;
196be944d42SStephen Warren 	*fiforeg = ahub->apbif_addr + TEGRA30_AHUB_CHANNEL_TXFIFO +
197be944d42SStephen Warren 		   (channel * TEGRA30_AHUB_CHANNEL_TXFIFO_STRIDE);
198be944d42SStephen Warren 	*reqsel = ahub->dma_sel + channel;
199be944d42SStephen Warren 
200be944d42SStephen Warren 	reg = TEGRA30_AHUB_CHANNEL_CTRL +
201be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_CHANNEL_CTRL_STRIDE);
202be944d42SStephen Warren 	val = tegra30_apbif_read(reg);
203be944d42SStephen Warren 	val &= ~(TEGRA30_AHUB_CHANNEL_CTRL_TX_THRESHOLD_MASK |
204be944d42SStephen Warren 		 TEGRA30_AHUB_CHANNEL_CTRL_TX_PACK_MASK);
205be944d42SStephen Warren 	val |= (7 << TEGRA30_AHUB_CHANNEL_CTRL_TX_THRESHOLD_SHIFT) |
206be944d42SStephen Warren 	       TEGRA30_AHUB_CHANNEL_CTRL_TX_PACK_EN |
207be944d42SStephen Warren 	       TEGRA30_AHUB_CHANNEL_CTRL_TX_PACK_16;
208be944d42SStephen Warren 	tegra30_apbif_write(reg, val);
209be944d42SStephen Warren 
210be944d42SStephen Warren 	reg = TEGRA30_AHUB_CIF_TX_CTRL +
211be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_CIF_TX_CTRL_STRIDE);
212be944d42SStephen Warren 	val = (0 << TEGRA30_AUDIOCIF_CTRL_FIFO_THRESHOLD_SHIFT) |
213be944d42SStephen Warren 	      (1 << TEGRA30_AUDIOCIF_CTRL_AUDIO_CHANNELS_SHIFT) |
214be944d42SStephen Warren 	      (1 << TEGRA30_AUDIOCIF_CTRL_CLIENT_CHANNELS_SHIFT) |
215be944d42SStephen Warren 	      TEGRA30_AUDIOCIF_CTRL_AUDIO_BITS_16 |
216be944d42SStephen Warren 	      TEGRA30_AUDIOCIF_CTRL_CLIENT_BITS_16 |
217be944d42SStephen Warren 	      TEGRA30_AUDIOCIF_CTRL_DIRECTION_TX;
218be944d42SStephen Warren 	tegra30_apbif_write(reg, val);
219be944d42SStephen Warren 
220be944d42SStephen Warren 	return 0;
221be944d42SStephen Warren }
222be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_allocate_tx_fifo);
223be944d42SStephen Warren 
224be944d42SStephen Warren int tegra30_ahub_enable_tx_fifo(enum tegra30_ahub_txcif txcif)
225be944d42SStephen Warren {
226be944d42SStephen Warren 	int channel = txcif - TEGRA30_AHUB_TXCIF_APBIF_TX0;
227be944d42SStephen Warren 	int reg, val;
228be944d42SStephen Warren 
229be944d42SStephen Warren 	reg = TEGRA30_AHUB_CHANNEL_CTRL +
230be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_CHANNEL_CTRL_STRIDE);
231be944d42SStephen Warren 	val = tegra30_apbif_read(reg);
232be944d42SStephen Warren 	val |= TEGRA30_AHUB_CHANNEL_CTRL_TX_EN;
233be944d42SStephen Warren 	tegra30_apbif_write(reg, val);
234be944d42SStephen Warren 
235be944d42SStephen Warren 	return 0;
236be944d42SStephen Warren }
237be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_enable_tx_fifo);
238be944d42SStephen Warren 
239be944d42SStephen Warren int tegra30_ahub_disable_tx_fifo(enum tegra30_ahub_txcif txcif)
240be944d42SStephen Warren {
241be944d42SStephen Warren 	int channel = txcif - TEGRA30_AHUB_TXCIF_APBIF_TX0;
242be944d42SStephen Warren 	int reg, val;
243be944d42SStephen Warren 
244be944d42SStephen Warren 	reg = TEGRA30_AHUB_CHANNEL_CTRL +
245be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_CHANNEL_CTRL_STRIDE);
246be944d42SStephen Warren 	val = tegra30_apbif_read(reg);
247be944d42SStephen Warren 	val &= ~TEGRA30_AHUB_CHANNEL_CTRL_TX_EN;
248be944d42SStephen Warren 	tegra30_apbif_write(reg, val);
249be944d42SStephen Warren 
250be944d42SStephen Warren 	return 0;
251be944d42SStephen Warren }
252be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_disable_tx_fifo);
253be944d42SStephen Warren 
254be944d42SStephen Warren int tegra30_ahub_free_tx_fifo(enum tegra30_ahub_txcif txcif)
255be944d42SStephen Warren {
256be944d42SStephen Warren 	int channel = txcif - TEGRA30_AHUB_TXCIF_APBIF_TX0;
257be944d42SStephen Warren 
258be944d42SStephen Warren 	__clear_bit(channel, ahub->tx_usage);
259be944d42SStephen Warren 
260be944d42SStephen Warren 	return 0;
261be944d42SStephen Warren }
262be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_free_tx_fifo);
263be944d42SStephen Warren 
264be944d42SStephen Warren int tegra30_ahub_set_rx_cif_source(enum tegra30_ahub_rxcif rxcif,
265be944d42SStephen Warren 				   enum tegra30_ahub_txcif txcif)
266be944d42SStephen Warren {
267be944d42SStephen Warren 	int channel = rxcif - TEGRA30_AHUB_RXCIF_APBIF_RX0;
268be944d42SStephen Warren 	int reg;
269be944d42SStephen Warren 
270be944d42SStephen Warren 	reg = TEGRA30_AHUB_AUDIO_RX +
271be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_AUDIO_RX_STRIDE);
272be944d42SStephen Warren 	tegra30_audio_write(reg, 1 << txcif);
273be944d42SStephen Warren 
274be944d42SStephen Warren 	return 0;
275be944d42SStephen Warren }
276be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_set_rx_cif_source);
277be944d42SStephen Warren 
278be944d42SStephen Warren int tegra30_ahub_unset_rx_cif_source(enum tegra30_ahub_rxcif rxcif)
279be944d42SStephen Warren {
280be944d42SStephen Warren 	int channel = rxcif - TEGRA30_AHUB_RXCIF_APBIF_RX0;
281be944d42SStephen Warren 	int reg;
282be944d42SStephen Warren 
283be944d42SStephen Warren 	reg = TEGRA30_AHUB_AUDIO_RX +
284be944d42SStephen Warren 	      (channel * TEGRA30_AHUB_AUDIO_RX_STRIDE);
285be944d42SStephen Warren 	tegra30_audio_write(reg, 0);
286be944d42SStephen Warren 
287be944d42SStephen Warren 	return 0;
288be944d42SStephen Warren }
289be944d42SStephen Warren EXPORT_SYMBOL_GPL(tegra30_ahub_unset_rx_cif_source);
290be944d42SStephen Warren 
291be944d42SStephen Warren static const char * const configlink_clocks[] __devinitconst = {
292be944d42SStephen Warren 	"i2s0",
293be944d42SStephen Warren 	"i2s1",
294be944d42SStephen Warren 	"i2s2",
295be944d42SStephen Warren 	"i2s3",
296be944d42SStephen Warren 	"i2s4",
297be944d42SStephen Warren 	"dam0",
298be944d42SStephen Warren 	"dam1",
299be944d42SStephen Warren 	"dam2",
300be944d42SStephen Warren 	"spdif_in",
301be944d42SStephen Warren };
302be944d42SStephen Warren 
303be944d42SStephen Warren struct of_dev_auxdata ahub_auxdata[] __devinitdata = {
304be944d42SStephen Warren 	OF_DEV_AUXDATA("nvidia,tegra30-i2s", 0x70080300, "tegra30-i2s.0", NULL),
305be944d42SStephen Warren 	OF_DEV_AUXDATA("nvidia,tegra30-i2s", 0x70080400, "tegra30-i2s.1", NULL),
306be944d42SStephen Warren 	OF_DEV_AUXDATA("nvidia,tegra30-i2s", 0x70080500, "tegra30-i2s.2", NULL),
307be944d42SStephen Warren 	OF_DEV_AUXDATA("nvidia,tegra30-i2s", 0x70080600, "tegra30-i2s.3", NULL),
308be944d42SStephen Warren 	OF_DEV_AUXDATA("nvidia,tegra30-i2s", 0x70080700, "tegra30-i2s.4", NULL),
309be944d42SStephen Warren 	{}
310be944d42SStephen Warren };
311be944d42SStephen Warren 
312be944d42SStephen Warren #define LAST_REG(name) \
313be944d42SStephen Warren 	(TEGRA30_AHUB_##name + \
314be944d42SStephen Warren 	 (TEGRA30_AHUB_##name##_STRIDE * TEGRA30_AHUB_##name##_COUNT) - 4)
315be944d42SStephen Warren 
316be944d42SStephen Warren #define REG_IN_ARRAY(reg, name) \
317be944d42SStephen Warren 	((reg >= TEGRA30_AHUB_##name) && \
318be944d42SStephen Warren 	 (reg <= LAST_REG(name) && \
319be944d42SStephen Warren 	 (!((reg - TEGRA30_AHUB_##name) % TEGRA30_AHUB_##name##_STRIDE))))
320be944d42SStephen Warren 
321be944d42SStephen Warren static bool tegra30_ahub_apbif_wr_rd_reg(struct device *dev, unsigned int reg)
322be944d42SStephen Warren {
323be944d42SStephen Warren 	switch (reg) {
324be944d42SStephen Warren 	case TEGRA30_AHUB_CONFIG_LINK_CTRL:
325be944d42SStephen Warren 	case TEGRA30_AHUB_MISC_CTRL:
326be944d42SStephen Warren 	case TEGRA30_AHUB_APBDMA_LIVE_STATUS:
327be944d42SStephen Warren 	case TEGRA30_AHUB_I2S_LIVE_STATUS:
328be944d42SStephen Warren 	case TEGRA30_AHUB_SPDIF_LIVE_STATUS:
329be944d42SStephen Warren 	case TEGRA30_AHUB_I2S_INT_MASK:
330be944d42SStephen Warren 	case TEGRA30_AHUB_DAM_INT_MASK:
331be944d42SStephen Warren 	case TEGRA30_AHUB_SPDIF_INT_MASK:
332be944d42SStephen Warren 	case TEGRA30_AHUB_APBIF_INT_MASK:
333be944d42SStephen Warren 	case TEGRA30_AHUB_I2S_INT_STATUS:
334be944d42SStephen Warren 	case TEGRA30_AHUB_DAM_INT_STATUS:
335be944d42SStephen Warren 	case TEGRA30_AHUB_SPDIF_INT_STATUS:
336be944d42SStephen Warren 	case TEGRA30_AHUB_APBIF_INT_STATUS:
337be944d42SStephen Warren 	case TEGRA30_AHUB_I2S_INT_SOURCE:
338be944d42SStephen Warren 	case TEGRA30_AHUB_DAM_INT_SOURCE:
339be944d42SStephen Warren 	case TEGRA30_AHUB_SPDIF_INT_SOURCE:
340be944d42SStephen Warren 	case TEGRA30_AHUB_APBIF_INT_SOURCE:
341be944d42SStephen Warren 	case TEGRA30_AHUB_I2S_INT_SET:
342be944d42SStephen Warren 	case TEGRA30_AHUB_DAM_INT_SET:
343be944d42SStephen Warren 	case TEGRA30_AHUB_SPDIF_INT_SET:
344be944d42SStephen Warren 	case TEGRA30_AHUB_APBIF_INT_SET:
345be944d42SStephen Warren 		return true;
346be944d42SStephen Warren 	default:
347be944d42SStephen Warren 		break;
348be944d42SStephen Warren 	};
349be944d42SStephen Warren 
350be944d42SStephen Warren 	if (REG_IN_ARRAY(reg, CHANNEL_CTRL) ||
351be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CHANNEL_CLEAR) ||
352be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CHANNEL_STATUS) ||
353be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CHANNEL_TXFIFO) ||
354be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CHANNEL_RXFIFO) ||
355be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CIF_TX_CTRL) ||
356be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CIF_RX_CTRL) ||
357be944d42SStephen Warren 	    REG_IN_ARRAY(reg, DAM_LIVE_STATUS))
358be944d42SStephen Warren 		return true;
359be944d42SStephen Warren 
360be944d42SStephen Warren 	return false;
361be944d42SStephen Warren }
362be944d42SStephen Warren 
363be944d42SStephen Warren static bool tegra30_ahub_apbif_volatile_reg(struct device *dev,
364be944d42SStephen Warren 					    unsigned int reg)
365be944d42SStephen Warren {
366be944d42SStephen Warren 	switch (reg) {
367be944d42SStephen Warren 	case TEGRA30_AHUB_CONFIG_LINK_CTRL:
368be944d42SStephen Warren 	case TEGRA30_AHUB_MISC_CTRL:
369be944d42SStephen Warren 	case TEGRA30_AHUB_APBDMA_LIVE_STATUS:
370be944d42SStephen Warren 	case TEGRA30_AHUB_I2S_LIVE_STATUS:
371be944d42SStephen Warren 	case TEGRA30_AHUB_SPDIF_LIVE_STATUS:
372be944d42SStephen Warren 	case TEGRA30_AHUB_I2S_INT_STATUS:
373be944d42SStephen Warren 	case TEGRA30_AHUB_DAM_INT_STATUS:
374be944d42SStephen Warren 	case TEGRA30_AHUB_SPDIF_INT_STATUS:
375be944d42SStephen Warren 	case TEGRA30_AHUB_APBIF_INT_STATUS:
376be944d42SStephen Warren 	case TEGRA30_AHUB_I2S_INT_SET:
377be944d42SStephen Warren 	case TEGRA30_AHUB_DAM_INT_SET:
378be944d42SStephen Warren 	case TEGRA30_AHUB_SPDIF_INT_SET:
379be944d42SStephen Warren 	case TEGRA30_AHUB_APBIF_INT_SET:
380be944d42SStephen Warren 		return true;
381be944d42SStephen Warren 	default:
382be944d42SStephen Warren 		break;
383be944d42SStephen Warren 	};
384be944d42SStephen Warren 
385be944d42SStephen Warren 	if (REG_IN_ARRAY(reg, CHANNEL_CLEAR) ||
386be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CHANNEL_STATUS) ||
387be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CHANNEL_TXFIFO) ||
388be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CHANNEL_RXFIFO) ||
389be944d42SStephen Warren 	    REG_IN_ARRAY(reg, DAM_LIVE_STATUS))
390be944d42SStephen Warren 		return true;
391be944d42SStephen Warren 
392be944d42SStephen Warren 	return false;
393be944d42SStephen Warren }
394be944d42SStephen Warren 
395be944d42SStephen Warren static bool tegra30_ahub_apbif_precious_reg(struct device *dev,
396be944d42SStephen Warren 					    unsigned int reg)
397be944d42SStephen Warren {
398be944d42SStephen Warren 	if (REG_IN_ARRAY(reg, CHANNEL_TXFIFO) ||
399be944d42SStephen Warren 	    REG_IN_ARRAY(reg, CHANNEL_RXFIFO))
400be944d42SStephen Warren 		return true;
401be944d42SStephen Warren 
402be944d42SStephen Warren 	return false;
403be944d42SStephen Warren }
404be944d42SStephen Warren 
405be944d42SStephen Warren static const struct regmap_config tegra30_ahub_apbif_regmap_config = {
406be944d42SStephen Warren 	.name = "apbif",
407be944d42SStephen Warren 	.reg_bits = 32,
408be944d42SStephen Warren 	.val_bits = 32,
409be944d42SStephen Warren 	.reg_stride = 4,
410be944d42SStephen Warren 	.max_register = TEGRA30_AHUB_APBIF_INT_SET,
411be944d42SStephen Warren 	.writeable_reg = tegra30_ahub_apbif_wr_rd_reg,
412be944d42SStephen Warren 	.readable_reg = tegra30_ahub_apbif_wr_rd_reg,
413be944d42SStephen Warren 	.volatile_reg = tegra30_ahub_apbif_volatile_reg,
414be944d42SStephen Warren 	.precious_reg = tegra30_ahub_apbif_precious_reg,
415be944d42SStephen Warren 	.cache_type = REGCACHE_RBTREE,
416be944d42SStephen Warren };
417be944d42SStephen Warren 
418be944d42SStephen Warren static bool tegra30_ahub_ahub_wr_rd_reg(struct device *dev, unsigned int reg)
419be944d42SStephen Warren {
420be944d42SStephen Warren 	if (REG_IN_ARRAY(reg, AUDIO_RX))
421be944d42SStephen Warren 		return true;
422be944d42SStephen Warren 
423be944d42SStephen Warren 	return false;
424be944d42SStephen Warren }
425be944d42SStephen Warren 
426be944d42SStephen Warren static const struct regmap_config tegra30_ahub_ahub_regmap_config = {
427be944d42SStephen Warren 	.name = "ahub",
428be944d42SStephen Warren 	.reg_bits = 32,
429be944d42SStephen Warren 	.val_bits = 32,
430be944d42SStephen Warren 	.reg_stride = 4,
431be944d42SStephen Warren 	.max_register = LAST_REG(AUDIO_RX),
432be944d42SStephen Warren 	.writeable_reg = tegra30_ahub_ahub_wr_rd_reg,
433be944d42SStephen Warren 	.readable_reg = tegra30_ahub_ahub_wr_rd_reg,
434be944d42SStephen Warren 	.cache_type = REGCACHE_RBTREE,
435be944d42SStephen Warren };
436be944d42SStephen Warren 
437be944d42SStephen Warren static int __devinit tegra30_ahub_probe(struct platform_device *pdev)
438be944d42SStephen Warren {
439be944d42SStephen Warren 	struct clk *clk;
440be944d42SStephen Warren 	int i;
441be944d42SStephen Warren 	struct resource *res0, *res1, *region;
442be944d42SStephen Warren 	u32 of_dma[2];
443be944d42SStephen Warren 	void __iomem *regs_apbif, *regs_ahub;
444be944d42SStephen Warren 	int ret = 0;
445be944d42SStephen Warren 
446be944d42SStephen Warren 	if (ahub)
447be944d42SStephen Warren 		return -ENODEV;
448be944d42SStephen Warren 
449be944d42SStephen Warren 	/*
450be944d42SStephen Warren 	 * The AHUB hosts a register bus: the "configlink". For this to
451be944d42SStephen Warren 	 * operate correctly, all devices on this bus must be out of reset.
452be944d42SStephen Warren 	 * Ensure that here.
453be944d42SStephen Warren 	 */
454be944d42SStephen Warren 	for (i = 0; i < ARRAY_SIZE(configlink_clocks); i++) {
455be944d42SStephen Warren 		clk = clk_get_sys(NULL, configlink_clocks[i]);
456be944d42SStephen Warren 		if (IS_ERR(clk)) {
457be944d42SStephen Warren 			dev_err(&pdev->dev, "Can't get clock %s\n",
458be944d42SStephen Warren 				configlink_clocks[i]);
459be944d42SStephen Warren 			ret = PTR_ERR(clk);
460be944d42SStephen Warren 			goto err;
461be944d42SStephen Warren 		}
462be944d42SStephen Warren 		tegra_periph_reset_deassert(clk);
463be944d42SStephen Warren 		clk_put(clk);
464be944d42SStephen Warren 	}
465be944d42SStephen Warren 
466be944d42SStephen Warren 	ahub = devm_kzalloc(&pdev->dev, sizeof(struct tegra30_ahub),
467be944d42SStephen Warren 			    GFP_KERNEL);
468be944d42SStephen Warren 	if (!ahub) {
469be944d42SStephen Warren 		dev_err(&pdev->dev, "Can't allocate tegra30_ahub\n");
470be944d42SStephen Warren 		ret = -ENOMEM;
471be944d42SStephen Warren 		goto err;
472be944d42SStephen Warren 	}
473be944d42SStephen Warren 	dev_set_drvdata(&pdev->dev, ahub);
474be944d42SStephen Warren 
475be944d42SStephen Warren 	ahub->dev = &pdev->dev;
476be944d42SStephen Warren 
477be944d42SStephen Warren 	ahub->clk_d_audio = clk_get(&pdev->dev, "d_audio");
478be944d42SStephen Warren 	if (IS_ERR(ahub->clk_d_audio)) {
479be944d42SStephen Warren 		dev_err(&pdev->dev, "Can't retrieve ahub d_audio clock\n");
480be944d42SStephen Warren 		ret = PTR_ERR(ahub->clk_d_audio);
481be944d42SStephen Warren 		goto err;
482be944d42SStephen Warren 	}
483be944d42SStephen Warren 
484be944d42SStephen Warren 	ahub->clk_apbif = clk_get(&pdev->dev, "apbif");
485be944d42SStephen Warren 	if (IS_ERR(ahub->clk_apbif)) {
486be944d42SStephen Warren 		dev_err(&pdev->dev, "Can't retrieve ahub apbif clock\n");
487be944d42SStephen Warren 		ret = PTR_ERR(ahub->clk_apbif);
488be944d42SStephen Warren 		goto err_clk_put_d_audio;
489be944d42SStephen Warren 	}
490be944d42SStephen Warren 
491be944d42SStephen Warren 	if (of_property_read_u32_array(pdev->dev.of_node,
492be944d42SStephen Warren 				"nvidia,dma-request-selector",
493be944d42SStephen Warren 				of_dma, 2) < 0) {
494be944d42SStephen Warren 		dev_err(&pdev->dev,
495be944d42SStephen Warren 			"Missing property nvidia,dma-request-selector\n");
496be944d42SStephen Warren 		ret = -ENODEV;
497be944d42SStephen Warren 		goto err_clk_put_d_audio;
498be944d42SStephen Warren 	}
499be944d42SStephen Warren 	ahub->dma_sel = of_dma[1];
500be944d42SStephen Warren 
501be944d42SStephen Warren 	res0 = platform_get_resource(pdev, IORESOURCE_MEM, 0);
502be944d42SStephen Warren 	if (!res0) {
503be944d42SStephen Warren 		dev_err(&pdev->dev, "No apbif memory resource\n");
504be944d42SStephen Warren 		ret = -ENODEV;
505be944d42SStephen Warren 		goto err_clk_put_apbif;
506be944d42SStephen Warren 	}
507be944d42SStephen Warren 
508be944d42SStephen Warren 	region = devm_request_mem_region(&pdev->dev, res0->start,
509be944d42SStephen Warren 					 resource_size(res0), DRV_NAME);
510be944d42SStephen Warren 	if (!region) {
511be944d42SStephen Warren 		dev_err(&pdev->dev, "request region apbif failed\n");
512be944d42SStephen Warren 		ret = -EBUSY;
513be944d42SStephen Warren 		goto err_clk_put_apbif;
514be944d42SStephen Warren 	}
515be944d42SStephen Warren 	ahub->apbif_addr = res0->start;
516be944d42SStephen Warren 
517be944d42SStephen Warren 	regs_apbif = devm_ioremap(&pdev->dev, res0->start,
518be944d42SStephen Warren 				  resource_size(res0));
519be944d42SStephen Warren 	if (!regs_apbif) {
520be944d42SStephen Warren 		dev_err(&pdev->dev, "ioremap apbif failed\n");
521be944d42SStephen Warren 		ret = -ENOMEM;
522be944d42SStephen Warren 		goto err_clk_put_apbif;
523be944d42SStephen Warren 	}
524be944d42SStephen Warren 
525be944d42SStephen Warren 	ahub->regmap_apbif = devm_regmap_init_mmio(&pdev->dev, regs_apbif,
526be944d42SStephen Warren 					&tegra30_ahub_apbif_regmap_config);
527be944d42SStephen Warren 	if (IS_ERR(ahub->regmap_apbif)) {
528be944d42SStephen Warren 		dev_err(&pdev->dev, "apbif regmap init failed\n");
529be944d42SStephen Warren 		ret = PTR_ERR(ahub->regmap_apbif);
530be944d42SStephen Warren 		goto err_clk_put_apbif;
531be944d42SStephen Warren 	}
532be944d42SStephen Warren 	regcache_cache_only(ahub->regmap_apbif, true);
533be944d42SStephen Warren 
534be944d42SStephen Warren 	res1 = platform_get_resource(pdev, IORESOURCE_MEM, 1);
535be944d42SStephen Warren 	if (!res1) {
536be944d42SStephen Warren 		dev_err(&pdev->dev, "No ahub memory resource\n");
537be944d42SStephen Warren 		ret = -ENODEV;
538be944d42SStephen Warren 		goto err_clk_put_apbif;
539be944d42SStephen Warren 	}
540be944d42SStephen Warren 
541be944d42SStephen Warren 	region = devm_request_mem_region(&pdev->dev, res1->start,
542be944d42SStephen Warren 					 resource_size(res1), DRV_NAME);
543be944d42SStephen Warren 	if (!region) {
544be944d42SStephen Warren 		dev_err(&pdev->dev, "request region ahub failed\n");
545be944d42SStephen Warren 		ret = -EBUSY;
546be944d42SStephen Warren 		goto err_clk_put_apbif;
547be944d42SStephen Warren 	}
548be944d42SStephen Warren 
549be944d42SStephen Warren 	regs_ahub = devm_ioremap(&pdev->dev, res1->start,
550be944d42SStephen Warren 				 resource_size(res1));
551be944d42SStephen Warren 	if (!regs_ahub) {
552be944d42SStephen Warren 		dev_err(&pdev->dev, "ioremap ahub failed\n");
553be944d42SStephen Warren 		ret = -ENOMEM;
554be944d42SStephen Warren 		goto err_clk_put_apbif;
555be944d42SStephen Warren 	}
556be944d42SStephen Warren 
557be944d42SStephen Warren 	ahub->regmap_ahub = devm_regmap_init_mmio(&pdev->dev, regs_ahub,
558be944d42SStephen Warren 					&tegra30_ahub_ahub_regmap_config);
559be944d42SStephen Warren 	if (IS_ERR(ahub->regmap_ahub)) {
560be944d42SStephen Warren 		dev_err(&pdev->dev, "ahub regmap init failed\n");
561be944d42SStephen Warren 		ret = PTR_ERR(ahub->regmap_ahub);
562be944d42SStephen Warren 		goto err_clk_put_apbif;
563be944d42SStephen Warren 	}
564be944d42SStephen Warren 	regcache_cache_only(ahub->regmap_ahub, true);
565be944d42SStephen Warren 
566be944d42SStephen Warren 	pm_runtime_enable(&pdev->dev);
567be944d42SStephen Warren 	if (!pm_runtime_enabled(&pdev->dev)) {
568be944d42SStephen Warren 		ret = tegra30_ahub_runtime_resume(&pdev->dev);
569be944d42SStephen Warren 		if (ret)
570be944d42SStephen Warren 			goto err_pm_disable;
571be944d42SStephen Warren 	}
572be944d42SStephen Warren 
573be944d42SStephen Warren 	of_platform_populate(pdev->dev.of_node, NULL, ahub_auxdata,
574be944d42SStephen Warren 			     &pdev->dev);
575be944d42SStephen Warren 
576be944d42SStephen Warren 	return 0;
577be944d42SStephen Warren 
578be944d42SStephen Warren err_pm_disable:
579be944d42SStephen Warren 	pm_runtime_disable(&pdev->dev);
580be944d42SStephen Warren err_clk_put_apbif:
581be944d42SStephen Warren 	clk_put(ahub->clk_apbif);
582be944d42SStephen Warren err_clk_put_d_audio:
583be944d42SStephen Warren 	clk_put(ahub->clk_d_audio);
584be944d42SStephen Warren 	ahub = 0;
585be944d42SStephen Warren err:
586be944d42SStephen Warren 	return ret;
587be944d42SStephen Warren }
588be944d42SStephen Warren 
589be944d42SStephen Warren static int __devexit tegra30_ahub_remove(struct platform_device *pdev)
590be944d42SStephen Warren {
591be944d42SStephen Warren 	if (!ahub)
592be944d42SStephen Warren 		return -ENODEV;
593be944d42SStephen Warren 
594be944d42SStephen Warren 	pm_runtime_disable(&pdev->dev);
595be944d42SStephen Warren 	if (!pm_runtime_status_suspended(&pdev->dev))
596be944d42SStephen Warren 		tegra30_ahub_runtime_suspend(&pdev->dev);
597be944d42SStephen Warren 
598be944d42SStephen Warren 	clk_put(ahub->clk_apbif);
599be944d42SStephen Warren 	clk_put(ahub->clk_d_audio);
600be944d42SStephen Warren 
601be944d42SStephen Warren 	ahub = 0;
602be944d42SStephen Warren 
603be944d42SStephen Warren 	return 0;
604be944d42SStephen Warren }
605be944d42SStephen Warren 
606be944d42SStephen Warren static const struct of_device_id tegra30_ahub_of_match[] __devinitconst = {
607be944d42SStephen Warren 	{ .compatible = "nvidia,tegra30-ahub", },
608be944d42SStephen Warren 	{},
609be944d42SStephen Warren };
610be944d42SStephen Warren 
611be944d42SStephen Warren static const struct dev_pm_ops tegra30_ahub_pm_ops __devinitconst = {
612be944d42SStephen Warren 	SET_RUNTIME_PM_OPS(tegra30_ahub_runtime_suspend,
613be944d42SStephen Warren 			   tegra30_ahub_runtime_resume, NULL)
614be944d42SStephen Warren };
615be944d42SStephen Warren 
616be944d42SStephen Warren static struct platform_driver tegra30_ahub_driver = {
617be944d42SStephen Warren 	.probe = tegra30_ahub_probe,
618be944d42SStephen Warren 	.remove = __devexit_p(tegra30_ahub_remove),
619be944d42SStephen Warren 	.driver = {
620be944d42SStephen Warren 		.name = DRV_NAME,
621be944d42SStephen Warren 		.owner = THIS_MODULE,
622be944d42SStephen Warren 		.of_match_table = tegra30_ahub_of_match,
623be944d42SStephen Warren 		.pm = &tegra30_ahub_pm_ops,
624be944d42SStephen Warren 	},
625be944d42SStephen Warren };
626be944d42SStephen Warren module_platform_driver(tegra30_ahub_driver);
627be944d42SStephen Warren 
628be944d42SStephen Warren MODULE_AUTHOR("Stephen Warren <swarren@nvidia.com>");
629be944d42SStephen Warren MODULE_DESCRIPTION("Tegra30 AHUB driver");
630be944d42SStephen Warren MODULE_LICENSE("GPL v2");
631be944d42SStephen Warren MODULE_ALIAS("platform:" DRV_NAME);
632