| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 1 | /* | 
 | 2 |  * ALSA SoC I2S (McBSP) Audio Layer for TI DAVINCI processor | 
 | 3 |  * | 
| Vladimir Barinov | d6b5203 | 2008-09-29 23:14:11 +0400 | [diff] [blame] | 4 |  * Author:      Vladimir Barinov, <vbarinov@embeddedalley.com> | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 5 |  * Copyright:   (C) 2007 MontaVista Software, Inc., <source@mvista.com> | 
 | 6 |  * | 
 | 7 |  * This program is free software; you can redistribute it and/or modify | 
 | 8 |  * it under the terms of the GNU General Public License version 2 as | 
 | 9 |  * published by the Free Software Foundation. | 
 | 10 |  */ | 
 | 11 |  | 
 | 12 | #include <linux/init.h> | 
 | 13 | #include <linux/module.h> | 
 | 14 | #include <linux/device.h> | 
| Tejun Heo | 5a0e3ad | 2010-03-24 17:04:11 +0900 | [diff] [blame] | 15 | #include <linux/slab.h> | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 16 | #include <linux/delay.h> | 
 | 17 | #include <linux/io.h> | 
 | 18 | #include <linux/clk.h> | 
 | 19 |  | 
 | 20 | #include <sound/core.h> | 
 | 21 | #include <sound/pcm.h> | 
 | 22 | #include <sound/pcm_params.h> | 
 | 23 | #include <sound/initval.h> | 
 | 24 | #include <sound/soc.h> | 
 | 25 |  | 
| Mark Brown | ff7d04b | 2009-07-08 16:54:51 +0100 | [diff] [blame] | 26 | #include <mach/asp.h> | 
 | 27 |  | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 28 | #include "davinci-pcm.h" | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 29 | #include "davinci-i2s.h" | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 30 |  | 
| David Brownell | a62114c | 2009-05-14 12:47:42 -0700 | [diff] [blame] | 31 |  | 
 | 32 | /* | 
 | 33 |  * NOTE:  terminology here is confusing. | 
 | 34 |  * | 
 | 35 |  *  - This driver supports the "Audio Serial Port" (ASP), | 
 | 36 |  *    found on dm6446, dm355, and other DaVinci chips. | 
 | 37 |  * | 
 | 38 |  *  - But it labels it a "Multi-channel Buffered Serial Port" | 
 | 39 |  *    (McBSP) as on older chips like the dm642 ... which was | 
 | 40 |  *    backward-compatible, possibly explaining that confusion. | 
 | 41 |  * | 
 | 42 |  *  - OMAP chips have a controller called McBSP, which is | 
 | 43 |  *    incompatible with the DaVinci flavor of McBSP. | 
 | 44 |  * | 
 | 45 |  *  - Newer DaVinci chips have a controller called McASP, | 
 | 46 |  *    incompatible with ASP and with either McBSP. | 
 | 47 |  * | 
 | 48 |  * In short:  this uses ASP to implement I2S, not McBSP. | 
 | 49 |  * And it won't be the only DaVinci implemention of I2S. | 
 | 50 |  */ | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 51 | #define DAVINCI_MCBSP_DRR_REG	0x00 | 
 | 52 | #define DAVINCI_MCBSP_DXR_REG	0x04 | 
 | 53 | #define DAVINCI_MCBSP_SPCR_REG	0x08 | 
 | 54 | #define DAVINCI_MCBSP_RCR_REG	0x0c | 
 | 55 | #define DAVINCI_MCBSP_XCR_REG	0x10 | 
 | 56 | #define DAVINCI_MCBSP_SRGR_REG	0x14 | 
 | 57 | #define DAVINCI_MCBSP_PCR_REG	0x24 | 
 | 58 |  | 
 | 59 | #define DAVINCI_MCBSP_SPCR_RRST		(1 << 0) | 
 | 60 | #define DAVINCI_MCBSP_SPCR_RINTM(v)	((v) << 4) | 
 | 61 | #define DAVINCI_MCBSP_SPCR_XRST		(1 << 16) | 
 | 62 | #define DAVINCI_MCBSP_SPCR_XINTM(v)	((v) << 20) | 
 | 63 | #define DAVINCI_MCBSP_SPCR_GRST		(1 << 22) | 
 | 64 | #define DAVINCI_MCBSP_SPCR_FRST		(1 << 23) | 
 | 65 | #define DAVINCI_MCBSP_SPCR_FREE		(1 << 25) | 
 | 66 |  | 
 | 67 | #define DAVINCI_MCBSP_RCR_RWDLEN1(v)	((v) << 5) | 
 | 68 | #define DAVINCI_MCBSP_RCR_RFRLEN1(v)	((v) << 8) | 
 | 69 | #define DAVINCI_MCBSP_RCR_RDATDLY(v)	((v) << 16) | 
| Troy Kisky | f5cfa95 | 2009-07-04 19:29:57 -0700 | [diff] [blame] | 70 | #define DAVINCI_MCBSP_RCR_RFIG		(1 << 18) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 71 | #define DAVINCI_MCBSP_RCR_RWDLEN2(v)	((v) << 21) | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 72 | #define DAVINCI_MCBSP_RCR_RFRLEN2(v)	((v) << 24) | 
 | 73 | #define DAVINCI_MCBSP_RCR_RPHASE	BIT(31) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 74 |  | 
 | 75 | #define DAVINCI_MCBSP_XCR_XWDLEN1(v)	((v) << 5) | 
 | 76 | #define DAVINCI_MCBSP_XCR_XFRLEN1(v)	((v) << 8) | 
 | 77 | #define DAVINCI_MCBSP_XCR_XDATDLY(v)	((v) << 16) | 
 | 78 | #define DAVINCI_MCBSP_XCR_XFIG		(1 << 18) | 
 | 79 | #define DAVINCI_MCBSP_XCR_XWDLEN2(v)	((v) << 21) | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 80 | #define DAVINCI_MCBSP_XCR_XFRLEN2(v)	((v) << 24) | 
 | 81 | #define DAVINCI_MCBSP_XCR_XPHASE	BIT(31) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 82 |  | 
 | 83 | #define DAVINCI_MCBSP_SRGR_FWID(v)	((v) << 8) | 
 | 84 | #define DAVINCI_MCBSP_SRGR_FPER(v)	((v) << 16) | 
 | 85 | #define DAVINCI_MCBSP_SRGR_FSGM		(1 << 28) | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 86 | #define DAVINCI_MCBSP_SRGR_CLKSM	BIT(29) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 87 |  | 
 | 88 | #define DAVINCI_MCBSP_PCR_CLKRP		(1 << 0) | 
 | 89 | #define DAVINCI_MCBSP_PCR_CLKXP		(1 << 1) | 
 | 90 | #define DAVINCI_MCBSP_PCR_FSRP		(1 << 2) | 
 | 91 | #define DAVINCI_MCBSP_PCR_FSXP		(1 << 3) | 
| Hugo Villeneuve | b402dff | 2008-11-08 13:26:09 -0500 | [diff] [blame] | 92 | #define DAVINCI_MCBSP_PCR_SCLKME	(1 << 7) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 93 | #define DAVINCI_MCBSP_PCR_CLKRM		(1 << 8) | 
 | 94 | #define DAVINCI_MCBSP_PCR_CLKXM		(1 << 9) | 
 | 95 | #define DAVINCI_MCBSP_PCR_FSRM		(1 << 10) | 
 | 96 | #define DAVINCI_MCBSP_PCR_FSXM		(1 << 11) | 
 | 97 |  | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 98 | enum { | 
 | 99 | 	DAVINCI_MCBSP_WORD_8 = 0, | 
 | 100 | 	DAVINCI_MCBSP_WORD_12, | 
 | 101 | 	DAVINCI_MCBSP_WORD_16, | 
 | 102 | 	DAVINCI_MCBSP_WORD_20, | 
 | 103 | 	DAVINCI_MCBSP_WORD_24, | 
 | 104 | 	DAVINCI_MCBSP_WORD_32, | 
 | 105 | }; | 
 | 106 |  | 
| Troy Kisky | 0d6c977 | 2009-11-18 17:49:51 -0700 | [diff] [blame] | 107 | static const unsigned char data_type[SNDRV_PCM_FORMAT_S32_LE + 1] = { | 
 | 108 | 	[SNDRV_PCM_FORMAT_S8]		= 1, | 
 | 109 | 	[SNDRV_PCM_FORMAT_S16_LE]	= 2, | 
 | 110 | 	[SNDRV_PCM_FORMAT_S32_LE]	= 4, | 
 | 111 | }; | 
 | 112 |  | 
 | 113 | static const unsigned char asp_word_length[SNDRV_PCM_FORMAT_S32_LE + 1] = { | 
 | 114 | 	[SNDRV_PCM_FORMAT_S8]		= DAVINCI_MCBSP_WORD_8, | 
 | 115 | 	[SNDRV_PCM_FORMAT_S16_LE]	= DAVINCI_MCBSP_WORD_16, | 
 | 116 | 	[SNDRV_PCM_FORMAT_S32_LE]	= DAVINCI_MCBSP_WORD_32, | 
 | 117 | }; | 
 | 118 |  | 
 | 119 | static const unsigned char double_fmt[SNDRV_PCM_FORMAT_S32_LE + 1] = { | 
 | 120 | 	[SNDRV_PCM_FORMAT_S8]		= SNDRV_PCM_FORMAT_S16_LE, | 
 | 121 | 	[SNDRV_PCM_FORMAT_S16_LE]	= SNDRV_PCM_FORMAT_S32_LE, | 
 | 122 | }; | 
 | 123 |  | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 124 | struct davinci_mcbsp_dev { | 
| Raffaele Recalcati | ec63755 | 2010-07-06 10:39:03 +0200 | [diff] [blame] | 125 | 	struct device *dev; | 
| Troy Kisky | 92e2a6f | 2009-09-11 14:29:03 -0700 | [diff] [blame] | 126 | 	struct davinci_pcm_dma_params	dma_params[2]; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 127 | 	void __iomem			*base; | 
| Troy Kisky | f5cfa95 | 2009-07-04 19:29:57 -0700 | [diff] [blame] | 128 | #define MOD_DSP_A	0 | 
 | 129 | #define MOD_DSP_B	1 | 
 | 130 | 	int				mode; | 
| Troy Kisky | c392bec | 2009-07-04 19:29:52 -0700 | [diff] [blame] | 131 | 	u32				pcr; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 132 | 	struct clk			*clk; | 
| Troy Kisky | 0d6c977 | 2009-11-18 17:49:51 -0700 | [diff] [blame] | 133 | 	/* | 
 | 134 | 	 * Combining both channels into 1 element will at least double the | 
 | 135 | 	 * amount of time between servicing the dma channel, increase | 
 | 136 | 	 * effiency, and reduce the chance of overrun/underrun. But, | 
 | 137 | 	 * it will result in the left & right channels being swapped. | 
 | 138 | 	 * | 
 | 139 | 	 * If relabeling the left and right channels is not possible, | 
 | 140 | 	 * you may want to let the codec know to swap them back. | 
 | 141 | 	 * | 
 | 142 | 	 * It may allow x10 the amount of time to service dma requests, | 
 | 143 | 	 * if the codec is master and is using an unnecessarily fast bit clock | 
 | 144 | 	 * (ie. tlvaic23b), independent of the sample rate. So, having an | 
 | 145 | 	 * entire frame at once means it can be serviced at the sample rate | 
 | 146 | 	 * instead of the bit clock rate. | 
 | 147 | 	 * | 
 | 148 | 	 * In the now unlikely case that an underrun still | 
 | 149 | 	 * occurs, both the left and right samples will be repeated | 
 | 150 | 	 * so that no pops are heard, and the left and right channels | 
 | 151 | 	 * won't end up being swapped because of the underrun. | 
 | 152 | 	 */ | 
 | 153 | 	unsigned enable_channel_combine:1; | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 154 |  | 
 | 155 | 	unsigned int fmt; | 
 | 156 | 	int clk_div; | 
| Raffaele Recalcati | ec63755 | 2010-07-06 10:39:03 +0200 | [diff] [blame] | 157 | 	int clk_input_pin; | 
| Raffaele Recalcati | d9823ed | 2010-07-06 10:39:04 +0200 | [diff] [blame] | 158 | 	bool i2s_accurate_sck; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 159 | }; | 
 | 160 |  | 
 | 161 | static inline void davinci_mcbsp_write_reg(struct davinci_mcbsp_dev *dev, | 
 | 162 | 					   int reg, u32 val) | 
 | 163 | { | 
 | 164 | 	__raw_writel(val, dev->base + reg); | 
 | 165 | } | 
 | 166 |  | 
 | 167 | static inline u32 davinci_mcbsp_read_reg(struct davinci_mcbsp_dev *dev, int reg) | 
 | 168 | { | 
 | 169 | 	return __raw_readl(dev->base + reg); | 
 | 170 | } | 
 | 171 |  | 
| Troy Kisky | c392bec | 2009-07-04 19:29:52 -0700 | [diff] [blame] | 172 | static void toggle_clock(struct davinci_mcbsp_dev *dev, int playback) | 
 | 173 | { | 
 | 174 | 	u32 m = playback ? DAVINCI_MCBSP_PCR_CLKXP : DAVINCI_MCBSP_PCR_CLKRP; | 
 | 175 | 	/* The clock needs to toggle to complete reset. | 
 | 176 | 	 * So, fake it by toggling the clk polarity. | 
 | 177 | 	 */ | 
 | 178 | 	davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_PCR_REG, dev->pcr ^ m); | 
 | 179 | 	davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_PCR_REG, dev->pcr); | 
 | 180 | } | 
 | 181 |  | 
| Troy Kisky | f9af37c | 2009-07-04 19:29:53 -0700 | [diff] [blame] | 182 | static void davinci_mcbsp_start(struct davinci_mcbsp_dev *dev, | 
 | 183 | 		struct snd_pcm_substream *substream) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 184 | { | 
 | 185 | 	struct snd_soc_pcm_runtime *rtd = substream->private_data; | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 186 | 	struct snd_soc_platform *platform = rtd->platform; | 
| Troy Kisky | c392bec | 2009-07-04 19:29:52 -0700 | [diff] [blame] | 187 | 	int playback = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK); | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 188 | 	u32 spcr; | 
| Troy Kisky | c392bec | 2009-07-04 19:29:52 -0700 | [diff] [blame] | 189 | 	u32 mask = playback ? DAVINCI_MCBSP_SPCR_XRST : DAVINCI_MCBSP_SPCR_RRST; | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 190 | 	spcr = davinci_mcbsp_read_reg(dev, DAVINCI_MCBSP_SPCR_REG); | 
| Troy Kisky | c392bec | 2009-07-04 19:29:52 -0700 | [diff] [blame] | 191 | 	if (spcr & mask) { | 
 | 192 | 		/* start off disabled */ | 
 | 193 | 		davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SPCR_REG, | 
 | 194 | 				spcr & ~mask); | 
 | 195 | 		toggle_clock(dev, playback); | 
 | 196 | 	} | 
| Troy Kisky | 1bef449 | 2009-07-04 19:29:55 -0700 | [diff] [blame] | 197 | 	if (dev->pcr & (DAVINCI_MCBSP_PCR_FSXM | DAVINCI_MCBSP_PCR_FSRM | | 
 | 198 | 			DAVINCI_MCBSP_PCR_CLKXM | DAVINCI_MCBSP_PCR_CLKRM)) { | 
 | 199 | 		/* Start the sample generator */ | 
 | 200 | 		spcr |= DAVINCI_MCBSP_SPCR_GRST; | 
 | 201 | 		davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SPCR_REG, spcr); | 
 | 202 | 	} | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 203 |  | 
| Troy Kisky | 1bef449 | 2009-07-04 19:29:55 -0700 | [diff] [blame] | 204 | 	if (playback) { | 
| Naresh Medisetty | fb0ef64 | 2008-11-12 10:26:31 +0530 | [diff] [blame] | 205 | 		/* Stop the DMA to avoid data loss */ | 
 | 206 | 		/* while the transmitter is out of reset to handle XSYNCERR */ | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 207 | 		if (platform->driver->ops->trigger) { | 
 | 208 | 			int ret = platform->driver->ops->trigger(substream, | 
| Naresh Medisetty | fb0ef64 | 2008-11-12 10:26:31 +0530 | [diff] [blame] | 209 | 				SNDRV_PCM_TRIGGER_STOP); | 
 | 210 | 			if (ret < 0) | 
 | 211 | 				printk(KERN_DEBUG "Playback DMA stop failed\n"); | 
 | 212 | 		} | 
 | 213 |  | 
 | 214 | 		/* Enable the transmitter */ | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 215 | 		spcr = davinci_mcbsp_read_reg(dev, DAVINCI_MCBSP_SPCR_REG); | 
 | 216 | 		spcr |= DAVINCI_MCBSP_SPCR_XRST; | 
 | 217 | 		davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SPCR_REG, spcr); | 
| Naresh Medisetty | fb0ef64 | 2008-11-12 10:26:31 +0530 | [diff] [blame] | 218 |  | 
 | 219 | 		/* wait for any unexpected frame sync error to occur */ | 
 | 220 | 		udelay(100); | 
 | 221 |  | 
 | 222 | 		/* Disable the transmitter to clear any outstanding XSYNCERR */ | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 223 | 		spcr = davinci_mcbsp_read_reg(dev, DAVINCI_MCBSP_SPCR_REG); | 
 | 224 | 		spcr &= ~DAVINCI_MCBSP_SPCR_XRST; | 
 | 225 | 		davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SPCR_REG, spcr); | 
| Troy Kisky | c392bec | 2009-07-04 19:29:52 -0700 | [diff] [blame] | 226 | 		toggle_clock(dev, playback); | 
| Naresh Medisetty | fb0ef64 | 2008-11-12 10:26:31 +0530 | [diff] [blame] | 227 |  | 
 | 228 | 		/* Restart the DMA */ | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 229 | 		if (platform->driver->ops->trigger) { | 
 | 230 | 			int ret = platform->driver->ops->trigger(substream, | 
| Naresh Medisetty | fb0ef64 | 2008-11-12 10:26:31 +0530 | [diff] [blame] | 231 | 				SNDRV_PCM_TRIGGER_START); | 
 | 232 | 			if (ret < 0) | 
 | 233 | 				printk(KERN_DEBUG "Playback DMA start failed\n"); | 
 | 234 | 		} | 
| Naresh Medisetty | fb0ef64 | 2008-11-12 10:26:31 +0530 | [diff] [blame] | 235 | 	} | 
 | 236 |  | 
| Troy Kisky | 1bef449 | 2009-07-04 19:29:55 -0700 | [diff] [blame] | 237 | 	/* Enable transmitter or receiver */ | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 238 | 	spcr = davinci_mcbsp_read_reg(dev, DAVINCI_MCBSP_SPCR_REG); | 
| Troy Kisky | 1bef449 | 2009-07-04 19:29:55 -0700 | [diff] [blame] | 239 | 	spcr |= mask; | 
 | 240 |  | 
 | 241 | 	if (dev->pcr & (DAVINCI_MCBSP_PCR_FSXM | DAVINCI_MCBSP_PCR_FSRM)) { | 
 | 242 | 		/* Start frame sync */ | 
 | 243 | 		spcr |= DAVINCI_MCBSP_SPCR_FRST; | 
 | 244 | 	} | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 245 | 	davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SPCR_REG, spcr); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 246 | } | 
 | 247 |  | 
| Troy Kisky | f9af37c | 2009-07-04 19:29:53 -0700 | [diff] [blame] | 248 | static void davinci_mcbsp_stop(struct davinci_mcbsp_dev *dev, int playback) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 249 | { | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 250 | 	u32 spcr; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 251 |  | 
 | 252 | 	/* Reset transmitter/receiver and sample rate/frame sync generators */ | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 253 | 	spcr = davinci_mcbsp_read_reg(dev, DAVINCI_MCBSP_SPCR_REG); | 
 | 254 | 	spcr &= ~(DAVINCI_MCBSP_SPCR_GRST | DAVINCI_MCBSP_SPCR_FRST); | 
| Troy Kisky | c392bec | 2009-07-04 19:29:52 -0700 | [diff] [blame] | 255 | 	spcr &= playback ? ~DAVINCI_MCBSP_SPCR_XRST : ~DAVINCI_MCBSP_SPCR_RRST; | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 256 | 	davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SPCR_REG, spcr); | 
| Troy Kisky | c392bec | 2009-07-04 19:29:52 -0700 | [diff] [blame] | 257 | 	toggle_clock(dev, playback); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 258 | } | 
 | 259 |  | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 260 | #define DEFAULT_BITPERSAMPLE	16 | 
 | 261 |  | 
| Liam Girdwood | 9cb132d | 2008-07-07 16:07:42 +0100 | [diff] [blame] | 262 | static int davinci_i2s_set_dai_fmt(struct snd_soc_dai *cpu_dai, | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 263 | 				   unsigned int fmt) | 
 | 264 | { | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 265 | 	struct davinci_mcbsp_dev *dev = snd_soc_dai_get_drvdata(cpu_dai); | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 266 | 	unsigned int pcr; | 
 | 267 | 	unsigned int srgr; | 
| Jarkko Nikula | ad51f76 | 2011-09-30 10:55:33 +0300 | [diff] [blame] | 268 | 	bool inv_fs = false; | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 269 | 	/* Attention srgr is updated by hw_params! */ | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 270 | 	srgr = DAVINCI_MCBSP_SRGR_FSGM | | 
 | 271 | 		DAVINCI_MCBSP_SRGR_FPER(DEFAULT_BITPERSAMPLE * 2 - 1) | | 
 | 272 | 		DAVINCI_MCBSP_SRGR_FWID(DEFAULT_BITPERSAMPLE - 1); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 273 |  | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 274 | 	dev->fmt = fmt; | 
| Troy Kisky | f5cfa95 | 2009-07-04 19:29:57 -0700 | [diff] [blame] | 275 | 	/* set master/slave audio interface */ | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 276 | 	switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) { | 
 | 277 | 	case SND_SOC_DAIFMT_CBS_CFS: | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 278 | 		/* cpu is master */ | 
 | 279 | 		pcr = DAVINCI_MCBSP_PCR_FSXM | | 
 | 280 | 			DAVINCI_MCBSP_PCR_FSRM | | 
 | 281 | 			DAVINCI_MCBSP_PCR_CLKXM | | 
 | 282 | 			DAVINCI_MCBSP_PCR_CLKRM; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 283 | 		break; | 
| Hugo Villeneuve | b402dff | 2008-11-08 13:26:09 -0500 | [diff] [blame] | 284 | 	case SND_SOC_DAIFMT_CBM_CFS: | 
| Raffaele Recalcati | ec63755 | 2010-07-06 10:39:03 +0200 | [diff] [blame] | 285 | 		pcr = DAVINCI_MCBSP_PCR_FSRM | DAVINCI_MCBSP_PCR_FSXM; | 
 | 286 | 		/* | 
 | 287 | 		 * Selection of the clock input pin that is the | 
 | 288 | 		 * input for the Sample Rate Generator. | 
 | 289 | 		 * McBSP FSR and FSX are driven by the Sample Rate | 
 | 290 | 		 * Generator. | 
 | 291 | 		 */ | 
 | 292 | 		switch (dev->clk_input_pin) { | 
 | 293 | 		case MCBSP_CLKS: | 
 | 294 | 			pcr |= DAVINCI_MCBSP_PCR_CLKXM | | 
 | 295 | 				DAVINCI_MCBSP_PCR_CLKRM; | 
 | 296 | 			break; | 
 | 297 | 		case MCBSP_CLKR: | 
 | 298 | 			pcr |= DAVINCI_MCBSP_PCR_SCLKME; | 
 | 299 | 			break; | 
 | 300 | 		default: | 
 | 301 | 			dev_err(dev->dev, "bad clk_input_pin\n"); | 
 | 302 | 			return -EINVAL; | 
 | 303 | 		} | 
 | 304 |  | 
| Hugo Villeneuve | b402dff | 2008-11-08 13:26:09 -0500 | [diff] [blame] | 305 | 		break; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 306 | 	case SND_SOC_DAIFMT_CBM_CFM: | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 307 | 		/* codec is master */ | 
 | 308 | 		pcr = 0; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 309 | 		break; | 
 | 310 | 	default: | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 311 | 		printk(KERN_ERR "%s:bad master\n", __func__); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 312 | 		return -EINVAL; | 
 | 313 | 	} | 
 | 314 |  | 
| Troy Kisky | f5cfa95 | 2009-07-04 19:29:57 -0700 | [diff] [blame] | 315 | 	/* interface format */ | 
| Troy Kisky | 69ab820 | 2008-12-18 12:36:44 -0700 | [diff] [blame] | 316 | 	switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) { | 
| Troy Kisky | 69ab820 | 2008-12-18 12:36:44 -0700 | [diff] [blame] | 317 | 	case SND_SOC_DAIFMT_I2S: | 
| Troy Kisky | 07d8d9d | 2008-12-19 13:05:24 -0700 | [diff] [blame] | 318 | 		/* Davinci doesn't support TRUE I2S, but some codecs will have | 
 | 319 | 		 * the left and right channels contiguous. This allows | 
 | 320 | 		 * dsp_a mode to be used with an inverted normal frame clk. | 
 | 321 | 		 * If your codec is master and does not have contiguous | 
 | 322 | 		 * channels, then you will have sound on only one channel. | 
 | 323 | 		 * Try using a different mode, or codec as slave. | 
 | 324 | 		 * | 
 | 325 | 		 * The TLV320AIC33 is an example of a codec where this works. | 
 | 326 | 		 * It has a variable bit clock frequency allowing it to have | 
 | 327 | 		 * valid data on every bit clock. | 
 | 328 | 		 * | 
 | 329 | 		 * The TLV320AIC23 is an example of a codec where this does not | 
 | 330 | 		 * work. It has a fixed bit clock frequency with progressively | 
 | 331 | 		 * more empty bit clock slots between channels as the sample | 
 | 332 | 		 * rate is lowered. | 
 | 333 | 		 */ | 
| Jarkko Nikula | ad51f76 | 2011-09-30 10:55:33 +0300 | [diff] [blame] | 334 | 		inv_fs = true; | 
| Troy Kisky | 07d8d9d | 2008-12-19 13:05:24 -0700 | [diff] [blame] | 335 | 	case SND_SOC_DAIFMT_DSP_A: | 
| Troy Kisky | f5cfa95 | 2009-07-04 19:29:57 -0700 | [diff] [blame] | 336 | 		dev->mode = MOD_DSP_A; | 
 | 337 | 		break; | 
 | 338 | 	case SND_SOC_DAIFMT_DSP_B: | 
 | 339 | 		dev->mode = MOD_DSP_B; | 
| Troy Kisky | 69ab820 | 2008-12-18 12:36:44 -0700 | [diff] [blame] | 340 | 		break; | 
 | 341 | 	default: | 
 | 342 | 		printk(KERN_ERR "%s:bad format\n", __func__); | 
 | 343 | 		return -EINVAL; | 
 | 344 | 	} | 
 | 345 |  | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 346 | 	switch (fmt & SND_SOC_DAIFMT_INV_MASK) { | 
| Troy Kisky | 9e03162 | 2008-12-19 13:05:23 -0700 | [diff] [blame] | 347 | 	case SND_SOC_DAIFMT_NB_NF: | 
| Troy Kisky | 664b4af | 2008-12-18 12:36:41 -0700 | [diff] [blame] | 348 | 		/* CLKRP Receive clock polarity, | 
 | 349 | 		 *	1 - sampled on rising edge of CLKR | 
 | 350 | 		 *	valid on rising edge | 
 | 351 | 		 * CLKXP Transmit clock polarity, | 
 | 352 | 		 *	1 - clocked on falling edge of CLKX | 
 | 353 | 		 *	valid on rising edge | 
 | 354 | 		 * FSRP  Receive frame sync pol, 0 - active high | 
 | 355 | 		 * FSXP  Transmit frame sync pol, 0 - active high | 
 | 356 | 		 */ | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 357 | 		pcr |= (DAVINCI_MCBSP_PCR_CLKXP | DAVINCI_MCBSP_PCR_CLKRP); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 358 | 		break; | 
| Troy Kisky | 9e03162 | 2008-12-19 13:05:23 -0700 | [diff] [blame] | 359 | 	case SND_SOC_DAIFMT_IB_IF: | 
| Troy Kisky | 664b4af | 2008-12-18 12:36:41 -0700 | [diff] [blame] | 360 | 		/* CLKRP Receive clock polarity, | 
 | 361 | 		 *	0 - sampled on falling edge of CLKR | 
 | 362 | 		 *	valid on falling edge | 
 | 363 | 		 * CLKXP Transmit clock polarity, | 
 | 364 | 		 *	0 - clocked on rising edge of CLKX | 
 | 365 | 		 *	valid on falling edge | 
 | 366 | 		 * FSRP  Receive frame sync pol, 1 - active low | 
 | 367 | 		 * FSXP  Transmit frame sync pol, 1 - active low | 
 | 368 | 		 */ | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 369 | 		pcr |= (DAVINCI_MCBSP_PCR_FSXP | DAVINCI_MCBSP_PCR_FSRP); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 370 | 		break; | 
| Troy Kisky | 9e03162 | 2008-12-19 13:05:23 -0700 | [diff] [blame] | 371 | 	case SND_SOC_DAIFMT_NB_IF: | 
| Troy Kisky | 664b4af | 2008-12-18 12:36:41 -0700 | [diff] [blame] | 372 | 		/* CLKRP Receive clock polarity, | 
 | 373 | 		 *	1 - sampled on rising edge of CLKR | 
 | 374 | 		 *	valid on rising edge | 
 | 375 | 		 * CLKXP Transmit clock polarity, | 
 | 376 | 		 *	1 - clocked on falling edge of CLKX | 
 | 377 | 		 *	valid on rising edge | 
 | 378 | 		 * FSRP  Receive frame sync pol, 1 - active low | 
 | 379 | 		 * FSXP  Transmit frame sync pol, 1 - active low | 
 | 380 | 		 */ | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 381 | 		pcr |= (DAVINCI_MCBSP_PCR_CLKXP | DAVINCI_MCBSP_PCR_CLKRP | | 
 | 382 | 			DAVINCI_MCBSP_PCR_FSXP | DAVINCI_MCBSP_PCR_FSRP); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 383 | 		break; | 
| Troy Kisky | 9e03162 | 2008-12-19 13:05:23 -0700 | [diff] [blame] | 384 | 	case SND_SOC_DAIFMT_IB_NF: | 
| Troy Kisky | 664b4af | 2008-12-18 12:36:41 -0700 | [diff] [blame] | 385 | 		/* CLKRP Receive clock polarity, | 
 | 386 | 		 *	0 - sampled on falling edge of CLKR | 
 | 387 | 		 *	valid on falling edge | 
 | 388 | 		 * CLKXP Transmit clock polarity, | 
 | 389 | 		 *	0 - clocked on rising edge of CLKX | 
 | 390 | 		 *	valid on falling edge | 
 | 391 | 		 * FSRP  Receive frame sync pol, 0 - active high | 
 | 392 | 		 * FSXP  Transmit frame sync pol, 0 - active high | 
 | 393 | 		 */ | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 394 | 		break; | 
 | 395 | 	default: | 
 | 396 | 		return -EINVAL; | 
 | 397 | 	} | 
| Jarkko Nikula | ad51f76 | 2011-09-30 10:55:33 +0300 | [diff] [blame] | 398 | 	if (inv_fs == true) | 
 | 399 | 		pcr ^= (DAVINCI_MCBSP_PCR_FSXP | DAVINCI_MCBSP_PCR_FSRP); | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 400 | 	davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SRGR_REG, srgr); | 
| Troy Kisky | c392bec | 2009-07-04 19:29:52 -0700 | [diff] [blame] | 401 | 	dev->pcr = pcr; | 
| Troy Kisky | 21903c1 | 2008-12-18 12:36:43 -0700 | [diff] [blame] | 402 | 	davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_PCR_REG, pcr); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 403 | 	return 0; | 
 | 404 | } | 
 | 405 |  | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 406 | static int davinci_i2s_dai_set_clkdiv(struct snd_soc_dai *cpu_dai, | 
 | 407 | 				int div_id, int div) | 
 | 408 | { | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 409 | 	struct davinci_mcbsp_dev *dev = snd_soc_dai_get_drvdata(cpu_dai); | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 410 |  | 
 | 411 | 	if (div_id != DAVINCI_MCBSP_CLKGDV) | 
 | 412 | 		return -ENODEV; | 
 | 413 |  | 
 | 414 | 	dev->clk_div = div; | 
 | 415 | 	return 0; | 
 | 416 | } | 
 | 417 |  | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 418 | static int davinci_i2s_hw_params(struct snd_pcm_substream *substream, | 
| Mark Brown | dee89c4 | 2008-11-18 22:11:38 +0000 | [diff] [blame] | 419 | 				 struct snd_pcm_hw_params *params, | 
 | 420 | 				 struct snd_soc_dai *dai) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 421 | { | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 422 | 	struct davinci_mcbsp_dev *dev = snd_soc_dai_get_drvdata(dai); | 
| Troy Kisky | 81ac55a | 2009-09-11 14:29:02 -0700 | [diff] [blame] | 423 | 	struct davinci_pcm_dma_params *dma_params = | 
| Troy Kisky | 92e2a6f | 2009-09-11 14:29:03 -0700 | [diff] [blame] | 424 | 					&dev->dma_params[substream->stream]; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 425 | 	struct snd_interval *i = NULL; | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 426 | 	int mcbsp_word_length, master; | 
 | 427 | 	unsigned int rcr, xcr, srgr, clk_div, freq, framesize; | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 428 | 	u32 spcr; | 
| Troy Kisky | 0d6c977 | 2009-11-18 17:49:51 -0700 | [diff] [blame] | 429 | 	snd_pcm_format_t fmt; | 
 | 430 | 	unsigned element_cnt = 1; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 431 |  | 
 | 432 | 	/* general line settings */ | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 433 | 	spcr = davinci_mcbsp_read_reg(dev, DAVINCI_MCBSP_SPCR_REG); | 
| Naresh Medisetty | cb6e206 | 2008-11-18 11:01:03 +0530 | [diff] [blame] | 434 | 	if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) { | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 435 | 		spcr |= DAVINCI_MCBSP_SPCR_RINTM(3) | DAVINCI_MCBSP_SPCR_FREE; | 
 | 436 | 		davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SPCR_REG, spcr); | 
| Naresh Medisetty | cb6e206 | 2008-11-18 11:01:03 +0530 | [diff] [blame] | 437 | 	} else { | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 438 | 		spcr |= DAVINCI_MCBSP_SPCR_XINTM(3) | DAVINCI_MCBSP_SPCR_FREE; | 
 | 439 | 		davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SPCR_REG, spcr); | 
| Naresh Medisetty | cb6e206 | 2008-11-18 11:01:03 +0530 | [diff] [blame] | 440 | 	} | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 441 |  | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 442 | 	master = dev->fmt & SND_SOC_DAIFMT_MASTER_MASK; | 
 | 443 | 	fmt = params_format(params); | 
 | 444 | 	mcbsp_word_length = asp_word_length[fmt]; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 445 |  | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 446 | 	switch (master) { | 
 | 447 | 	case SND_SOC_DAIFMT_CBS_CFS: | 
 | 448 | 		freq = clk_get_rate(dev->clk); | 
 | 449 | 		srgr = DAVINCI_MCBSP_SRGR_FSGM | | 
 | 450 | 		       DAVINCI_MCBSP_SRGR_CLKSM; | 
 | 451 | 		srgr |= DAVINCI_MCBSP_SRGR_FWID(mcbsp_word_length * | 
 | 452 | 						8 - 1); | 
| Raffaele Recalcati | d9823ed | 2010-07-06 10:39:04 +0200 | [diff] [blame] | 453 | 		if (dev->i2s_accurate_sck) { | 
 | 454 | 			clk_div = 256; | 
 | 455 | 			do { | 
 | 456 | 				framesize = (freq / (--clk_div)) / | 
 | 457 | 				params->rate_num * | 
 | 458 | 					params->rate_den; | 
 | 459 | 			} while (((framesize < 33) || (framesize > 4095)) && | 
 | 460 | 				 (clk_div)); | 
 | 461 | 			clk_div--; | 
 | 462 | 			srgr |= DAVINCI_MCBSP_SRGR_FPER(framesize - 1); | 
 | 463 | 		} else { | 
 | 464 | 			/* symmetric waveforms */ | 
 | 465 | 			clk_div = freq / (mcbsp_word_length * 16) / | 
 | 466 | 				  params->rate_num * params->rate_den; | 
 | 467 | 			srgr |= DAVINCI_MCBSP_SRGR_FPER(mcbsp_word_length * | 
 | 468 | 							16 - 1); | 
 | 469 | 		} | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 470 | 		clk_div &= 0xFF; | 
 | 471 | 		srgr |= clk_div; | 
 | 472 | 		break; | 
 | 473 | 	case SND_SOC_DAIFMT_CBM_CFS: | 
 | 474 | 		srgr = DAVINCI_MCBSP_SRGR_FSGM; | 
 | 475 | 		clk_div = dev->clk_div - 1; | 
 | 476 | 		srgr |= DAVINCI_MCBSP_SRGR_FWID(mcbsp_word_length * 8 - 1); | 
 | 477 | 		srgr |= DAVINCI_MCBSP_SRGR_FPER(mcbsp_word_length * 16 - 1); | 
 | 478 | 		clk_div &= 0xFF; | 
 | 479 | 		srgr |= clk_div; | 
 | 480 | 		break; | 
 | 481 | 	case SND_SOC_DAIFMT_CBM_CFM: | 
 | 482 | 		/* Clock and frame sync given from external sources */ | 
 | 483 | 		i = hw_param_interval(params, SNDRV_PCM_HW_PARAM_SAMPLE_BITS); | 
 | 484 | 		srgr = DAVINCI_MCBSP_SRGR_FSGM; | 
 | 485 | 		srgr |= DAVINCI_MCBSP_SRGR_FWID(snd_interval_value(i) - 1); | 
 | 486 | 		pr_debug("%s - %d  FWID set: re-read srgr = %X\n", | 
 | 487 | 			__func__, __LINE__, snd_interval_value(i) - 1); | 
 | 488 |  | 
 | 489 | 		i = hw_param_interval(params, SNDRV_PCM_HW_PARAM_FRAME_BITS); | 
 | 490 | 		srgr |= DAVINCI_MCBSP_SRGR_FPER(snd_interval_value(i) - 1); | 
 | 491 | 		break; | 
 | 492 | 	default: | 
 | 493 | 		return -EINVAL; | 
 | 494 | 	} | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 495 | 	davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_SRGR_REG, srgr); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 496 |  | 
| Troy Kisky | f5cfa95 | 2009-07-04 19:29:57 -0700 | [diff] [blame] | 497 | 	rcr = DAVINCI_MCBSP_RCR_RFIG; | 
 | 498 | 	xcr = DAVINCI_MCBSP_XCR_XFIG; | 
 | 499 | 	if (dev->mode == MOD_DSP_B) { | 
 | 500 | 		rcr |= DAVINCI_MCBSP_RCR_RDATDLY(0); | 
 | 501 | 		xcr |= DAVINCI_MCBSP_XCR_XDATDLY(0); | 
 | 502 | 	} else { | 
 | 503 | 		rcr |= DAVINCI_MCBSP_RCR_RDATDLY(1); | 
 | 504 | 		xcr |= DAVINCI_MCBSP_XCR_XDATDLY(1); | 
 | 505 | 	} | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 506 | 	/* Determine xfer data type */ | 
| Troy Kisky | 0d6c977 | 2009-11-18 17:49:51 -0700 | [diff] [blame] | 507 | 	fmt = params_format(params); | 
 | 508 | 	if ((fmt > SNDRV_PCM_FORMAT_S32_LE) || !data_type[fmt]) { | 
| Jean Delvare | 9b6e12e | 2008-08-26 15:47:55 +0200 | [diff] [blame] | 509 | 		printk(KERN_WARNING "davinci-i2s: unsupported PCM format\n"); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 510 | 		return -EINVAL; | 
 | 511 | 	} | 
 | 512 |  | 
| Troy Kisky | 0d6c977 | 2009-11-18 17:49:51 -0700 | [diff] [blame] | 513 | 	if (params_channels(params) == 2) { | 
 | 514 | 		element_cnt = 2; | 
 | 515 | 		if (double_fmt[fmt] && dev->enable_channel_combine) { | 
 | 516 | 			element_cnt = 1; | 
 | 517 | 			fmt = double_fmt[fmt]; | 
 | 518 | 		} | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 519 | 		switch (master) { | 
 | 520 | 		case SND_SOC_DAIFMT_CBS_CFS: | 
 | 521 | 		case SND_SOC_DAIFMT_CBS_CFM: | 
 | 522 | 			rcr |= DAVINCI_MCBSP_RCR_RFRLEN2(0); | 
 | 523 | 			xcr |= DAVINCI_MCBSP_XCR_XFRLEN2(0); | 
 | 524 | 			rcr |= DAVINCI_MCBSP_RCR_RPHASE; | 
 | 525 | 			xcr |= DAVINCI_MCBSP_XCR_XPHASE; | 
 | 526 | 			break; | 
 | 527 | 		case SND_SOC_DAIFMT_CBM_CFM: | 
 | 528 | 		case SND_SOC_DAIFMT_CBM_CFS: | 
 | 529 | 			rcr |= DAVINCI_MCBSP_RCR_RFRLEN2(element_cnt - 1); | 
 | 530 | 			xcr |= DAVINCI_MCBSP_XCR_XFRLEN2(element_cnt - 1); | 
 | 531 | 			break; | 
 | 532 | 		default: | 
 | 533 | 			return -EINVAL; | 
 | 534 | 		} | 
| Troy Kisky | 0d6c977 | 2009-11-18 17:49:51 -0700 | [diff] [blame] | 535 | 	} | 
 | 536 | 	dma_params->acnt = dma_params->data_type = data_type[fmt]; | 
| Chaithrika U S | 4fa9c1a | 2009-09-30 17:32:27 -0400 | [diff] [blame] | 537 | 	dma_params->fifo_level = 0; | 
| Troy Kisky | 0d6c977 | 2009-11-18 17:49:51 -0700 | [diff] [blame] | 538 | 	mcbsp_word_length = asp_word_length[fmt]; | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 539 |  | 
 | 540 | 	switch (master) { | 
 | 541 | 	case SND_SOC_DAIFMT_CBS_CFS: | 
 | 542 | 	case SND_SOC_DAIFMT_CBS_CFM: | 
 | 543 | 		rcr |= DAVINCI_MCBSP_RCR_RFRLEN1(0); | 
 | 544 | 		xcr |= DAVINCI_MCBSP_XCR_XFRLEN1(0); | 
 | 545 | 		break; | 
 | 546 | 	case SND_SOC_DAIFMT_CBM_CFM: | 
 | 547 | 	case SND_SOC_DAIFMT_CBM_CFS: | 
 | 548 | 		rcr |= DAVINCI_MCBSP_RCR_RFRLEN1(element_cnt - 1); | 
 | 549 | 		xcr |= DAVINCI_MCBSP_XCR_XFRLEN1(element_cnt - 1); | 
 | 550 | 		break; | 
 | 551 | 	default: | 
 | 552 | 		return -EINVAL; | 
 | 553 | 	} | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 554 |  | 
| Troy Kisky | f5cfa95 | 2009-07-04 19:29:57 -0700 | [diff] [blame] | 555 | 	rcr |= DAVINCI_MCBSP_RCR_RWDLEN1(mcbsp_word_length) | | 
 | 556 | 		DAVINCI_MCBSP_RCR_RWDLEN2(mcbsp_word_length); | 
 | 557 | 	xcr |= DAVINCI_MCBSP_XCR_XWDLEN1(mcbsp_word_length) | | 
 | 558 | 		DAVINCI_MCBSP_XCR_XWDLEN2(mcbsp_word_length); | 
 | 559 |  | 
 | 560 | 	if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) | 
| Troy Kisky | 35cf635 | 2009-07-04 19:29:51 -0700 | [diff] [blame] | 561 | 		davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_XCR_REG, xcr); | 
| Troy Kisky | f5cfa95 | 2009-07-04 19:29:57 -0700 | [diff] [blame] | 562 | 	else | 
 | 563 | 		davinci_mcbsp_write_reg(dev, DAVINCI_MCBSP_RCR_REG, rcr); | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 564 |  | 
 | 565 | 	pr_debug("%s - %d  srgr=%X\n", __func__, __LINE__, srgr); | 
 | 566 | 	pr_debug("%s - %d  xcr=%X\n", __func__, __LINE__, xcr); | 
 | 567 | 	pr_debug("%s - %d  rcr=%X\n", __func__, __LINE__, rcr); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 568 | 	return 0; | 
 | 569 | } | 
 | 570 |  | 
| Troy Kisky | af0adf3 | 2009-07-04 19:29:59 -0700 | [diff] [blame] | 571 | static int davinci_i2s_prepare(struct snd_pcm_substream *substream, | 
 | 572 | 		struct snd_soc_dai *dai) | 
 | 573 | { | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 574 | 	struct davinci_mcbsp_dev *dev = snd_soc_dai_get_drvdata(dai); | 
| Troy Kisky | af0adf3 | 2009-07-04 19:29:59 -0700 | [diff] [blame] | 575 | 	int playback = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK); | 
 | 576 | 	davinci_mcbsp_stop(dev, playback); | 
| Troy Kisky | af0adf3 | 2009-07-04 19:29:59 -0700 | [diff] [blame] | 577 | 	return 0; | 
 | 578 | } | 
 | 579 |  | 
| Mark Brown | dee89c4 | 2008-11-18 22:11:38 +0000 | [diff] [blame] | 580 | static int davinci_i2s_trigger(struct snd_pcm_substream *substream, int cmd, | 
 | 581 | 			       struct snd_soc_dai *dai) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 582 | { | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 583 | 	struct davinci_mcbsp_dev *dev = snd_soc_dai_get_drvdata(dai); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 584 | 	int ret = 0; | 
| Troy Kisky | f9af37c | 2009-07-04 19:29:53 -0700 | [diff] [blame] | 585 | 	int playback = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 586 |  | 
 | 587 | 	switch (cmd) { | 
 | 588 | 	case SNDRV_PCM_TRIGGER_START: | 
 | 589 | 	case SNDRV_PCM_TRIGGER_RESUME: | 
 | 590 | 	case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: | 
| Troy Kisky | f9af37c | 2009-07-04 19:29:53 -0700 | [diff] [blame] | 591 | 		davinci_mcbsp_start(dev, substream); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 592 | 		break; | 
 | 593 | 	case SNDRV_PCM_TRIGGER_STOP: | 
 | 594 | 	case SNDRV_PCM_TRIGGER_SUSPEND: | 
 | 595 | 	case SNDRV_PCM_TRIGGER_PAUSE_PUSH: | 
| Troy Kisky | f9af37c | 2009-07-04 19:29:53 -0700 | [diff] [blame] | 596 | 		davinci_mcbsp_stop(dev, playback); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 597 | 		break; | 
 | 598 | 	default: | 
 | 599 | 		ret = -EINVAL; | 
 | 600 | 	} | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 601 | 	return ret; | 
 | 602 | } | 
 | 603 |  | 
| Chris Paulson-Ellis | bedad0c | 2010-11-16 12:27:09 +0000 | [diff] [blame] | 604 | static int davinci_i2s_startup(struct snd_pcm_substream *substream, | 
 | 605 | 			       struct snd_soc_dai *dai) | 
 | 606 | { | 
 | 607 | 	struct davinci_mcbsp_dev *dev = snd_soc_dai_get_drvdata(dai); | 
 | 608 |  | 
 | 609 | 	snd_soc_dai_set_dma_data(dai, substream, dev->dma_params); | 
 | 610 | 	return 0; | 
 | 611 | } | 
 | 612 |  | 
| Troy Kisky | af0adf3 | 2009-07-04 19:29:59 -0700 | [diff] [blame] | 613 | static void davinci_i2s_shutdown(struct snd_pcm_substream *substream, | 
 | 614 | 		struct snd_soc_dai *dai) | 
 | 615 | { | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 616 | 	struct davinci_mcbsp_dev *dev = snd_soc_dai_get_drvdata(dai); | 
| Troy Kisky | af0adf3 | 2009-07-04 19:29:59 -0700 | [diff] [blame] | 617 | 	int playback = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK); | 
 | 618 | 	davinci_mcbsp_stop(dev, playback); | 
 | 619 | } | 
 | 620 |  | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 621 | #define DAVINCI_I2S_RATES	SNDRV_PCM_RATE_8000_96000 | 
 | 622 |  | 
| Lars-Peter Clausen | 85e7652 | 2011-11-23 11:40:40 +0100 | [diff] [blame] | 623 | static const struct snd_soc_dai_ops davinci_i2s_dai_ops = { | 
| Chris Paulson-Ellis | bedad0c | 2010-11-16 12:27:09 +0000 | [diff] [blame] | 624 | 	.startup	= davinci_i2s_startup, | 
| Mark Brown | 3f405b4 | 2009-07-07 19:18:46 +0100 | [diff] [blame] | 625 | 	.shutdown	= davinci_i2s_shutdown, | 
 | 626 | 	.prepare	= davinci_i2s_prepare, | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 627 | 	.trigger	= davinci_i2s_trigger, | 
 | 628 | 	.hw_params	= davinci_i2s_hw_params, | 
 | 629 | 	.set_fmt	= davinci_i2s_set_dai_fmt, | 
| Raffaele Recalcati | a4c8ea2 | 2010-07-06 10:39:02 +0200 | [diff] [blame] | 630 | 	.set_clkdiv	= davinci_i2s_dai_set_clkdiv, | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 631 |  | 
 | 632 | }; | 
 | 633 |  | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 634 | static struct snd_soc_dai_driver davinci_i2s_dai = { | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 635 | 	.playback = { | 
 | 636 | 		.channels_min = 2, | 
 | 637 | 		.channels_max = 2, | 
 | 638 | 		.rates = DAVINCI_I2S_RATES, | 
 | 639 | 		.formats = SNDRV_PCM_FMTBIT_S16_LE,}, | 
 | 640 | 	.capture = { | 
 | 641 | 		.channels_min = 2, | 
 | 642 | 		.channels_max = 2, | 
 | 643 | 		.rates = DAVINCI_I2S_RATES, | 
 | 644 | 		.formats = SNDRV_PCM_FMTBIT_S16_LE,}, | 
 | 645 | 	.ops = &davinci_i2s_dai_ops, | 
 | 646 |  | 
 | 647 | }; | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 648 |  | 
 | 649 | static int davinci_i2s_probe(struct platform_device *pdev) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 650 | { | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 651 | 	struct snd_platform_data *pdata = pdev->dev.platform_data; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 652 | 	struct davinci_mcbsp_dev *dev; | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 653 | 	struct resource *mem, *ioarea, *res; | 
| Sekhar Nori | 48519f0 | 2010-07-19 12:31:16 +0530 | [diff] [blame] | 654 | 	enum dma_event_q asp_chan_q = EVENTQ_0; | 
 | 655 | 	enum dma_event_q ram_chan_q = EVENTQ_1; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 656 | 	int ret; | 
 | 657 |  | 
 | 658 | 	mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 
 | 659 | 	if (!mem) { | 
 | 660 | 		dev_err(&pdev->dev, "no mem resource?\n"); | 
 | 661 | 		return -ENODEV; | 
 | 662 | 	} | 
 | 663 |  | 
| Julia Lawall | cd0ff7e | 2011-12-29 17:51:22 +0100 | [diff] [blame] | 664 | 	ioarea = devm_request_mem_region(&pdev->dev, mem->start, | 
 | 665 | 					 resource_size(mem), | 
 | 666 | 					 pdev->name); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 667 | 	if (!ioarea) { | 
 | 668 | 		dev_err(&pdev->dev, "McBSP region already claimed\n"); | 
 | 669 | 		return -EBUSY; | 
 | 670 | 	} | 
 | 671 |  | 
| Julia Lawall | cd0ff7e | 2011-12-29 17:51:22 +0100 | [diff] [blame] | 672 | 	dev = devm_kzalloc(&pdev->dev, sizeof(struct davinci_mcbsp_dev), | 
 | 673 | 			   GFP_KERNEL); | 
 | 674 | 	if (!dev) | 
 | 675 | 		return -ENOMEM; | 
| Troy Kisky | 1e224f3 | 2009-11-18 17:49:53 -0700 | [diff] [blame] | 676 | 	if (pdata) { | 
| Troy Kisky | 0d6c977 | 2009-11-18 17:49:51 -0700 | [diff] [blame] | 677 | 		dev->enable_channel_combine = pdata->enable_channel_combine; | 
| Troy Kisky | 1e224f3 | 2009-11-18 17:49:53 -0700 | [diff] [blame] | 678 | 		dev->dma_params[SNDRV_PCM_STREAM_PLAYBACK].sram_size = | 
 | 679 | 			pdata->sram_size_playback; | 
 | 680 | 		dev->dma_params[SNDRV_PCM_STREAM_CAPTURE].sram_size = | 
 | 681 | 			pdata->sram_size_capture; | 
| Raffaele Recalcati | ec63755 | 2010-07-06 10:39:03 +0200 | [diff] [blame] | 682 | 		dev->clk_input_pin = pdata->clk_input_pin; | 
| Raffaele Recalcati | d9823ed | 2010-07-06 10:39:04 +0200 | [diff] [blame] | 683 | 		dev->i2s_accurate_sck = pdata->i2s_accurate_sck; | 
| Sekhar Nori | 48519f0 | 2010-07-19 12:31:16 +0530 | [diff] [blame] | 684 | 		asp_chan_q = pdata->asp_chan_q; | 
 | 685 | 		ram_chan_q = pdata->ram_chan_q; | 
| Troy Kisky | 1e224f3 | 2009-11-18 17:49:53 -0700 | [diff] [blame] | 686 | 	} | 
| Sekhar Nori | 48519f0 | 2010-07-19 12:31:16 +0530 | [diff] [blame] | 687 |  | 
 | 688 | 	dev->dma_params[SNDRV_PCM_STREAM_PLAYBACK].asp_chan_q	= asp_chan_q; | 
 | 689 | 	dev->dma_params[SNDRV_PCM_STREAM_PLAYBACK].ram_chan_q	= ram_chan_q; | 
 | 690 | 	dev->dma_params[SNDRV_PCM_STREAM_CAPTURE].asp_chan_q	= asp_chan_q; | 
 | 691 | 	dev->dma_params[SNDRV_PCM_STREAM_CAPTURE].ram_chan_q	= ram_chan_q; | 
 | 692 |  | 
| Kevin Hilman | 3e46a44 | 2009-07-15 10:42:09 -0700 | [diff] [blame] | 693 | 	dev->clk = clk_get(&pdev->dev, NULL); | 
| Julia Lawall | cd0ff7e | 2011-12-29 17:51:22 +0100 | [diff] [blame] | 694 | 	if (IS_ERR(dev->clk)) | 
 | 695 | 		return -ENODEV; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 696 | 	clk_enable(dev->clk); | 
 | 697 |  | 
| Julia Lawall | cd0ff7e | 2011-12-29 17:51:22 +0100 | [diff] [blame] | 698 | 	dev->base = devm_ioremap(&pdev->dev, mem->start, resource_size(mem)); | 
| Vaibhav Bedia | 4f82f02 | 2011-02-09 18:39:54 +0530 | [diff] [blame] | 699 | 	if (!dev->base) { | 
 | 700 | 		dev_err(&pdev->dev, "ioremap failed\n"); | 
 | 701 | 		ret = -ENOMEM; | 
 | 702 | 		goto err_release_clk; | 
 | 703 | 	} | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 704 |  | 
| Troy Kisky | 92e2a6f | 2009-09-11 14:29:03 -0700 | [diff] [blame] | 705 | 	dev->dma_params[SNDRV_PCM_STREAM_PLAYBACK].dma_addr = | 
| Vaibhav Bedia | 4f82f02 | 2011-02-09 18:39:54 +0530 | [diff] [blame] | 706 | 	    (dma_addr_t)(mem->start + DAVINCI_MCBSP_DXR_REG); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 707 |  | 
| Troy Kisky | 92e2a6f | 2009-09-11 14:29:03 -0700 | [diff] [blame] | 708 | 	dev->dma_params[SNDRV_PCM_STREAM_CAPTURE].dma_addr = | 
| Vaibhav Bedia | 4f82f02 | 2011-02-09 18:39:54 +0530 | [diff] [blame] | 709 | 	    (dma_addr_t)(mem->start + DAVINCI_MCBSP_DRR_REG); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 710 |  | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 711 | 	/* first TX, then RX */ | 
 | 712 | 	res = platform_get_resource(pdev, IORESOURCE_DMA, 0); | 
 | 713 | 	if (!res) { | 
 | 714 | 		dev_err(&pdev->dev, "no DMA resource\n"); | 
| Chaithrika U S | efd13be | 2009-06-08 06:49:41 -0400 | [diff] [blame] | 715 | 		ret = -ENXIO; | 
| Julia Lawall | cd0ff7e | 2011-12-29 17:51:22 +0100 | [diff] [blame] | 716 | 		goto err_release_clk; | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 717 | 	} | 
| Troy Kisky | 92e2a6f | 2009-09-11 14:29:03 -0700 | [diff] [blame] | 718 | 	dev->dma_params[SNDRV_PCM_STREAM_PLAYBACK].channel = res->start; | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 719 |  | 
 | 720 | 	res = platform_get_resource(pdev, IORESOURCE_DMA, 1); | 
 | 721 | 	if (!res) { | 
 | 722 | 		dev_err(&pdev->dev, "no DMA resource\n"); | 
| Chaithrika U S | efd13be | 2009-06-08 06:49:41 -0400 | [diff] [blame] | 723 | 		ret = -ENXIO; | 
| Julia Lawall | cd0ff7e | 2011-12-29 17:51:22 +0100 | [diff] [blame] | 724 | 		goto err_release_clk; | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 725 | 	} | 
| Troy Kisky | 92e2a6f | 2009-09-11 14:29:03 -0700 | [diff] [blame] | 726 | 	dev->dma_params[SNDRV_PCM_STREAM_CAPTURE].channel = res->start; | 
| Raffaele Recalcati | ec63755 | 2010-07-06 10:39:03 +0200 | [diff] [blame] | 727 | 	dev->dev = &pdev->dev; | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 728 |  | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 729 | 	dev_set_drvdata(&pdev->dev, dev); | 
 | 730 |  | 
 | 731 | 	ret = snd_soc_register_dai(&pdev->dev, &davinci_i2s_dai); | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 732 | 	if (ret != 0) | 
| Julia Lawall | cd0ff7e | 2011-12-29 17:51:22 +0100 | [diff] [blame] | 733 | 		goto err_release_clk; | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 734 |  | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 735 | 	return 0; | 
 | 736 |  | 
| Vaibhav Bedia | eef6d7b | 2011-02-09 18:39:53 +0530 | [diff] [blame] | 737 | err_release_clk: | 
 | 738 | 	clk_disable(dev->clk); | 
 | 739 | 	clk_put(dev->clk); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 740 | 	return ret; | 
 | 741 | } | 
 | 742 |  | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 743 | static int davinci_i2s_remove(struct platform_device *pdev) | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 744 | { | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 745 | 	struct davinci_mcbsp_dev *dev = dev_get_drvdata(&pdev->dev); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 746 |  | 
| Liam Girdwood | f0fba2a | 2010-03-17 20:15:21 +0000 | [diff] [blame] | 747 | 	snd_soc_unregister_dai(&pdev->dev); | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 748 | 	clk_disable(dev->clk); | 
 | 749 | 	clk_put(dev->clk); | 
 | 750 | 	dev->clk = NULL; | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 751 |  | 
 | 752 | 	return 0; | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 753 | } | 
 | 754 |  | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 755 | static struct platform_driver davinci_mcbsp_driver = { | 
 | 756 | 	.probe		= davinci_i2s_probe, | 
 | 757 | 	.remove		= davinci_i2s_remove, | 
 | 758 | 	.driver		= { | 
| Chris Paulson-Ellis | bedad0c | 2010-11-16 12:27:09 +0000 | [diff] [blame] | 759 | 		.name	= "davinci-mcbsp", | 
| Chaithrika U S | 5204d49 | 2009-06-05 06:28:23 -0400 | [diff] [blame] | 760 | 		.owner	= THIS_MODULE, | 
 | 761 | 	}, | 
| Eric Miao | 6335d05 | 2009-03-03 09:41:00 +0800 | [diff] [blame] | 762 | }; | 
 | 763 |  | 
| Axel Lin | f9b8a51 | 2011-11-25 10:09:27 +0800 | [diff] [blame] | 764 | module_platform_driver(davinci_mcbsp_driver); | 
| Mark Brown | 3f4b783 | 2008-12-03 19:26:35 +0000 | [diff] [blame] | 765 |  | 
| Vladimir Barinov | 310355c | 2008-02-18 11:40:22 +0100 | [diff] [blame] | 766 | MODULE_AUTHOR("Vladimir Barinov"); | 
 | 767 | MODULE_DESCRIPTION("TI DAVINCI I2S (McBSP) SoC Interface"); | 
 | 768 | MODULE_LICENSE("GPL"); |