xref: /openbmc/linux/sound/soc/img/img-i2s-in.c (revision e983940270f10fe8551baf0098be76ea478294a3)
1 /*
2  * IMG I2S input controller driver
3  *
4  * Copyright (C) 2015 Imagination Technologies Ltd.
5  *
6  * Author: Damien Horsley <Damien.Horsley@imgtec.com>
7  *
8  * This program is free software; you can redistribute it and/or modify it
9  * under the terms and conditions of the GNU General Public License,
10  * version 2, as published by the Free Software Foundation.
11  */
12 
13 #include <linux/clk.h>
14 #include <linux/init.h>
15 #include <linux/kernel.h>
16 #include <linux/module.h>
17 #include <linux/of.h>
18 #include <linux/platform_device.h>
19 #include <linux/reset.h>
20 
21 #include <sound/core.h>
22 #include <sound/dmaengine_pcm.h>
23 #include <sound/initval.h>
24 #include <sound/pcm.h>
25 #include <sound/pcm_params.h>
26 #include <sound/soc.h>
27 
28 #define IMG_I2S_IN_RX_FIFO			0x0
29 
30 #define IMG_I2S_IN_CTL				0x4
31 #define IMG_I2S_IN_CTL_ACTIVE_CHAN_MASK		0xfffffffc
32 #define IMG_I2S_IN_CTL_ACTIVE_CH_SHIFT		2
33 #define IMG_I2S_IN_CTL_16PACK_MASK		BIT(1)
34 #define IMG_I2S_IN_CTL_ME_MASK			BIT(0)
35 
36 #define IMG_I2S_IN_CH_CTL			0x4
37 #define IMG_I2S_IN_CH_CTL_CCDEL_MASK		0x38000
38 #define IMG_I2S_IN_CH_CTL_CCDEL_SHIFT		15
39 #define IMG_I2S_IN_CH_CTL_FEN_MASK		BIT(14)
40 #define IMG_I2S_IN_CH_CTL_FMODE_MASK		BIT(13)
41 #define IMG_I2S_IN_CH_CTL_16PACK_MASK		BIT(12)
42 #define IMG_I2S_IN_CH_CTL_JUST_MASK		BIT(10)
43 #define IMG_I2S_IN_CH_CTL_PACKH_MASK		BIT(9)
44 #define IMG_I2S_IN_CH_CTL_CLK_TRANS_MASK	BIT(8)
45 #define IMG_I2S_IN_CH_CTL_BLKP_MASK		BIT(7)
46 #define IMG_I2S_IN_CH_CTL_FIFO_FLUSH_MASK	BIT(6)
47 #define IMG_I2S_IN_CH_CTL_LRD_MASK		BIT(3)
48 #define IMG_I2S_IN_CH_CTL_FW_MASK		BIT(2)
49 #define IMG_I2S_IN_CH_CTL_SW_MASK		BIT(1)
50 #define IMG_I2S_IN_CH_CTL_ME_MASK		BIT(0)
51 
52 #define IMG_I2S_IN_CH_STRIDE			0x20
53 
54 struct img_i2s_in {
55 	void __iomem *base;
56 	struct clk *clk_sys;
57 	struct snd_dmaengine_dai_dma_data dma_data;
58 	struct device *dev;
59 	unsigned int max_i2s_chan;
60 	void __iomem *channel_base;
61 	unsigned int active_channels;
62 	struct snd_soc_dai_driver dai_driver;
63 };
64 
65 static inline void img_i2s_in_writel(struct img_i2s_in *i2s, u32 val, u32 reg)
66 {
67 	writel(val, i2s->base + reg);
68 }
69 
70 static inline u32 img_i2s_in_readl(struct img_i2s_in *i2s, u32 reg)
71 {
72 	return readl(i2s->base + reg);
73 }
74 
75 static inline void img_i2s_in_ch_writel(struct img_i2s_in *i2s, u32 chan,
76 					u32 val, u32 reg)
77 {
78 	writel(val, i2s->channel_base + (chan * IMG_I2S_IN_CH_STRIDE) + reg);
79 }
80 
81 static inline u32 img_i2s_in_ch_readl(struct img_i2s_in *i2s, u32 chan,
82 					u32 reg)
83 {
84 	return readl(i2s->channel_base + (chan * IMG_I2S_IN_CH_STRIDE) + reg);
85 }
86 
87 static inline void img_i2s_in_ch_disable(struct img_i2s_in *i2s, u32 chan)
88 {
89 	u32 reg;
90 
91 	reg = img_i2s_in_ch_readl(i2s, chan, IMG_I2S_IN_CH_CTL);
92 	reg &= ~IMG_I2S_IN_CH_CTL_ME_MASK;
93 	img_i2s_in_ch_writel(i2s, chan, reg, IMG_I2S_IN_CH_CTL);
94 }
95 
96 static inline void img_i2s_in_ch_enable(struct img_i2s_in *i2s, u32 chan)
97 {
98 	u32 reg;
99 
100 	reg = img_i2s_in_ch_readl(i2s, chan, IMG_I2S_IN_CH_CTL);
101 	reg |= IMG_I2S_IN_CH_CTL_ME_MASK;
102 	img_i2s_in_ch_writel(i2s, chan, reg, IMG_I2S_IN_CH_CTL);
103 }
104 
105 static inline void img_i2s_in_disable(struct img_i2s_in *i2s)
106 {
107 	u32 reg;
108 
109 	reg = img_i2s_in_readl(i2s, IMG_I2S_IN_CTL);
110 	reg &= ~IMG_I2S_IN_CTL_ME_MASK;
111 	img_i2s_in_writel(i2s, reg, IMG_I2S_IN_CTL);
112 }
113 
114 static inline void img_i2s_in_enable(struct img_i2s_in *i2s)
115 {
116 	u32 reg;
117 
118 	reg = img_i2s_in_readl(i2s, IMG_I2S_IN_CTL);
119 	reg |= IMG_I2S_IN_CTL_ME_MASK;
120 	img_i2s_in_writel(i2s, reg, IMG_I2S_IN_CTL);
121 }
122 
123 static inline void img_i2s_in_flush(struct img_i2s_in *i2s)
124 {
125 	int i;
126 	u32 reg;
127 
128 	for (i = 0; i < i2s->active_channels; i++) {
129 		reg = img_i2s_in_ch_readl(i2s, i, IMG_I2S_IN_CH_CTL);
130 		reg |= IMG_I2S_IN_CH_CTL_FIFO_FLUSH_MASK;
131 		img_i2s_in_ch_writel(i2s, i, reg, IMG_I2S_IN_CH_CTL);
132 		reg &= ~IMG_I2S_IN_CH_CTL_FIFO_FLUSH_MASK;
133 		img_i2s_in_ch_writel(i2s, i, reg, IMG_I2S_IN_CH_CTL);
134 	}
135 }
136 
137 static int img_i2s_in_trigger(struct snd_pcm_substream *substream, int cmd,
138 	struct snd_soc_dai *dai)
139 {
140 	struct img_i2s_in *i2s = snd_soc_dai_get_drvdata(dai);
141 
142 	switch (cmd) {
143 	case SNDRV_PCM_TRIGGER_START:
144 	case SNDRV_PCM_TRIGGER_RESUME:
145 	case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
146 		img_i2s_in_enable(i2s);
147 		break;
148 
149 	case SNDRV_PCM_TRIGGER_STOP:
150 	case SNDRV_PCM_TRIGGER_SUSPEND:
151 	case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
152 		img_i2s_in_disable(i2s);
153 		break;
154 	default:
155 		return -EINVAL;
156 	}
157 
158 	return 0;
159 }
160 
161 static int img_i2s_in_check_rate(struct img_i2s_in *i2s,
162 		unsigned int sample_rate, unsigned int frame_size,
163 		unsigned int *bclk_filter_enable,
164 		unsigned int *bclk_filter_value)
165 {
166 	unsigned int bclk_freq, cur_freq;
167 
168 	bclk_freq = sample_rate * frame_size;
169 
170 	cur_freq = clk_get_rate(i2s->clk_sys);
171 
172 	if (cur_freq >= bclk_freq * 8) {
173 		*bclk_filter_enable = 1;
174 		*bclk_filter_value = 0;
175 	} else if (cur_freq >= bclk_freq * 7) {
176 		*bclk_filter_enable = 1;
177 		*bclk_filter_value = 1;
178 	} else if (cur_freq >= bclk_freq * 6) {
179 		*bclk_filter_enable = 0;
180 		*bclk_filter_value = 0;
181 	} else {
182 		dev_err(i2s->dev,
183 			"Sys clock rate %u insufficient for sample rate %u\n",
184 			cur_freq, sample_rate);
185 		return -EINVAL;
186 	}
187 
188 	return 0;
189 }
190 
191 static int img_i2s_in_hw_params(struct snd_pcm_substream *substream,
192 	struct snd_pcm_hw_params *params, struct snd_soc_dai *dai)
193 {
194 	struct img_i2s_in *i2s = snd_soc_dai_get_drvdata(dai);
195 	unsigned int rate, channels, i2s_channels, frame_size;
196 	unsigned int bclk_filter_enable, bclk_filter_value;
197 	int i, ret = 0;
198 	u32 reg, control_mask, chan_control_mask;
199 	u32 control_set = 0, chan_control_set = 0;
200 	snd_pcm_format_t format;
201 
202 	rate = params_rate(params);
203 	format = params_format(params);
204 	channels = params_channels(params);
205 	i2s_channels = channels / 2;
206 
207 	switch (format) {
208 	case SNDRV_PCM_FORMAT_S32_LE:
209 		frame_size = 64;
210 		chan_control_set |= IMG_I2S_IN_CH_CTL_SW_MASK;
211 		chan_control_set |= IMG_I2S_IN_CH_CTL_FW_MASK;
212 		chan_control_set |= IMG_I2S_IN_CH_CTL_PACKH_MASK;
213 		break;
214 	case SNDRV_PCM_FORMAT_S24_LE:
215 		frame_size = 64;
216 		chan_control_set |= IMG_I2S_IN_CH_CTL_SW_MASK;
217 		chan_control_set |= IMG_I2S_IN_CH_CTL_FW_MASK;
218 		break;
219 	case SNDRV_PCM_FORMAT_S16_LE:
220 		frame_size = 32;
221 		control_set |= IMG_I2S_IN_CTL_16PACK_MASK;
222 		chan_control_set |= IMG_I2S_IN_CH_CTL_16PACK_MASK;
223 		break;
224 	default:
225 		return -EINVAL;
226 	}
227 
228 	if ((channels < 2) ||
229 	    (channels > (i2s->max_i2s_chan * 2)) ||
230 	    (channels % 2))
231 		return -EINVAL;
232 
233 	control_set |= ((i2s_channels - 1) << IMG_I2S_IN_CTL_ACTIVE_CH_SHIFT);
234 
235 	ret = img_i2s_in_check_rate(i2s, rate, frame_size,
236 			&bclk_filter_enable, &bclk_filter_value);
237 	if (ret < 0)
238 		return ret;
239 
240 	if (bclk_filter_enable)
241 		chan_control_set |= IMG_I2S_IN_CH_CTL_FEN_MASK;
242 
243 	if (bclk_filter_value)
244 		chan_control_set |= IMG_I2S_IN_CH_CTL_FMODE_MASK;
245 
246 	control_mask = IMG_I2S_IN_CTL_16PACK_MASK |
247 		       IMG_I2S_IN_CTL_ACTIVE_CHAN_MASK;
248 
249 	chan_control_mask = IMG_I2S_IN_CH_CTL_16PACK_MASK |
250 			    IMG_I2S_IN_CH_CTL_FEN_MASK |
251 			    IMG_I2S_IN_CH_CTL_FMODE_MASK |
252 			    IMG_I2S_IN_CH_CTL_SW_MASK |
253 			    IMG_I2S_IN_CH_CTL_FW_MASK |
254 			    IMG_I2S_IN_CH_CTL_PACKH_MASK;
255 
256 	reg = img_i2s_in_readl(i2s, IMG_I2S_IN_CTL);
257 	reg = (reg & ~control_mask) | control_set;
258 	img_i2s_in_writel(i2s, reg, IMG_I2S_IN_CTL);
259 
260 	for (i = 0; i < i2s->active_channels; i++)
261 		img_i2s_in_ch_disable(i2s, i);
262 
263 	for (i = 0; i < i2s->max_i2s_chan; i++) {
264 		reg = img_i2s_in_ch_readl(i2s, i, IMG_I2S_IN_CH_CTL);
265 		reg = (reg & ~chan_control_mask) | chan_control_set;
266 		img_i2s_in_ch_writel(i2s, i, reg, IMG_I2S_IN_CH_CTL);
267 	}
268 
269 	i2s->active_channels = i2s_channels;
270 
271 	img_i2s_in_flush(i2s);
272 
273 	for (i = 0; i < i2s->active_channels; i++)
274 		img_i2s_in_ch_enable(i2s, i);
275 
276 	return 0;
277 }
278 
279 static int img_i2s_in_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
280 {
281 	struct img_i2s_in *i2s = snd_soc_dai_get_drvdata(dai);
282 	int i;
283 	u32 chan_control_mask, lrd_set = 0, blkp_set = 0, chan_control_set = 0;
284 	u32 reg;
285 
286 	switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
287 	case SND_SOC_DAIFMT_NB_NF:
288 		lrd_set |= IMG_I2S_IN_CH_CTL_LRD_MASK;
289 		break;
290 	case SND_SOC_DAIFMT_NB_IF:
291 		break;
292 	case SND_SOC_DAIFMT_IB_NF:
293 		lrd_set |= IMG_I2S_IN_CH_CTL_LRD_MASK;
294 		blkp_set |= IMG_I2S_IN_CH_CTL_BLKP_MASK;
295 		break;
296 	case SND_SOC_DAIFMT_IB_IF:
297 		blkp_set |= IMG_I2S_IN_CH_CTL_BLKP_MASK;
298 		break;
299 	default:
300 		return -EINVAL;
301 	}
302 
303 	switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
304 	case SND_SOC_DAIFMT_I2S:
305 		chan_control_set |= IMG_I2S_IN_CH_CTL_CLK_TRANS_MASK;
306 		break;
307 	case SND_SOC_DAIFMT_LEFT_J:
308 		break;
309 	default:
310 		return -EINVAL;
311 	}
312 
313 	switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
314 	case SND_SOC_DAIFMT_CBM_CFM:
315 		break;
316 	default:
317 		return -EINVAL;
318 	}
319 
320 	chan_control_mask = IMG_I2S_IN_CH_CTL_CLK_TRANS_MASK;
321 
322 	for (i = 0; i < i2s->active_channels; i++)
323 		img_i2s_in_ch_disable(i2s, i);
324 
325 	/*
326 	 * BLKP and LRD must be set during separate register writes
327 	 */
328 	for (i = 0; i < i2s->max_i2s_chan; i++) {
329 		reg = img_i2s_in_ch_readl(i2s, i, IMG_I2S_IN_CH_CTL);
330 		reg = (reg & ~chan_control_mask) | chan_control_set;
331 		img_i2s_in_ch_writel(i2s, i, reg, IMG_I2S_IN_CH_CTL);
332 		reg = (reg & ~IMG_I2S_IN_CH_CTL_BLKP_MASK) | blkp_set;
333 		img_i2s_in_ch_writel(i2s, i, reg, IMG_I2S_IN_CH_CTL);
334 		reg = (reg & ~IMG_I2S_IN_CH_CTL_LRD_MASK) | lrd_set;
335 		img_i2s_in_ch_writel(i2s, i, reg, IMG_I2S_IN_CH_CTL);
336 	}
337 
338 	for (i = 0; i < i2s->active_channels; i++)
339 		img_i2s_in_ch_enable(i2s, i);
340 
341 	return 0;
342 }
343 
344 static const struct snd_soc_dai_ops img_i2s_in_dai_ops = {
345 	.trigger = img_i2s_in_trigger,
346 	.hw_params = img_i2s_in_hw_params,
347 	.set_fmt = img_i2s_in_set_fmt
348 };
349 
350 static int img_i2s_in_dai_probe(struct snd_soc_dai *dai)
351 {
352 	struct img_i2s_in *i2s = snd_soc_dai_get_drvdata(dai);
353 
354 	snd_soc_dai_init_dma_data(dai, NULL, &i2s->dma_data);
355 
356 	return 0;
357 }
358 
359 static const struct snd_soc_component_driver img_i2s_in_component = {
360 	.name = "img-i2s-in"
361 };
362 
363 static int img_i2s_in_dma_prepare_slave_config(struct snd_pcm_substream *st,
364 	struct snd_pcm_hw_params *params, struct dma_slave_config *sc)
365 {
366 	unsigned int i2s_channels = params_channels(params) / 2;
367 	struct snd_soc_pcm_runtime *rtd = st->private_data;
368 	struct snd_dmaengine_dai_dma_data *dma_data;
369 	int ret;
370 
371 	dma_data = snd_soc_dai_get_dma_data(rtd->cpu_dai, st);
372 
373 	ret = snd_hwparams_to_dma_slave_config(st, params, sc);
374 	if (ret)
375 		return ret;
376 
377 	sc->src_addr = dma_data->addr;
378 	sc->src_addr_width = dma_data->addr_width;
379 	sc->src_maxburst = 4 * i2s_channels;
380 
381 	return 0;
382 }
383 
384 static const struct snd_dmaengine_pcm_config img_i2s_in_dma_config = {
385 	.prepare_slave_config = img_i2s_in_dma_prepare_slave_config
386 };
387 
388 static int img_i2s_in_probe(struct platform_device *pdev)
389 {
390 	struct img_i2s_in *i2s;
391 	struct resource *res;
392 	void __iomem *base;
393 	int ret, i;
394 	struct reset_control *rst;
395 	unsigned int max_i2s_chan_pow_2;
396 	struct device *dev = &pdev->dev;
397 
398 	i2s = devm_kzalloc(dev, sizeof(*i2s), GFP_KERNEL);
399 	if (!i2s)
400 		return -ENOMEM;
401 
402 	platform_set_drvdata(pdev, i2s);
403 
404 	i2s->dev = dev;
405 
406 	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
407 	base = devm_ioremap_resource(dev, res);
408 	if (IS_ERR(base))
409 		return PTR_ERR(base);
410 
411 	i2s->base = base;
412 
413 	if (of_property_read_u32(pdev->dev.of_node, "img,i2s-channels",
414 			&i2s->max_i2s_chan)) {
415 		dev_err(dev, "No img,i2s-channels property\n");
416 		return -EINVAL;
417 	}
418 
419 	max_i2s_chan_pow_2 = 1 << get_count_order(i2s->max_i2s_chan);
420 
421 	i2s->channel_base = base + (max_i2s_chan_pow_2 * 0x20);
422 
423 	i2s->clk_sys = devm_clk_get(dev, "sys");
424 	if (IS_ERR(i2s->clk_sys)) {
425 		if (PTR_ERR(i2s->clk_sys) != -EPROBE_DEFER)
426 			dev_err(dev, "Failed to acquire clock 'sys'\n");
427 		return PTR_ERR(i2s->clk_sys);
428 	}
429 
430 	ret = clk_prepare_enable(i2s->clk_sys);
431 	if (ret)
432 		return ret;
433 
434 	i2s->active_channels = 1;
435 	i2s->dma_data.addr = res->start + IMG_I2S_IN_RX_FIFO;
436 	i2s->dma_data.addr_width = 4;
437 
438 	i2s->dai_driver.probe = img_i2s_in_dai_probe;
439 	i2s->dai_driver.capture.channels_min = 2;
440 	i2s->dai_driver.capture.channels_max = i2s->max_i2s_chan * 2;
441 	i2s->dai_driver.capture.rates = SNDRV_PCM_RATE_8000_192000;
442 	i2s->dai_driver.capture.formats = SNDRV_PCM_FMTBIT_S32_LE |
443 		SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S16_LE;
444 	i2s->dai_driver.ops = &img_i2s_in_dai_ops;
445 
446 	rst = devm_reset_control_get(dev, "rst");
447 	if (IS_ERR(rst)) {
448 		if (PTR_ERR(rst) == -EPROBE_DEFER) {
449 			ret = -EPROBE_DEFER;
450 			goto err_clk_disable;
451 		}
452 
453 		dev_dbg(dev, "No top level reset found\n");
454 
455 		img_i2s_in_disable(i2s);
456 
457 		for (i = 0; i < i2s->max_i2s_chan; i++)
458 			img_i2s_in_ch_disable(i2s, i);
459 	} else {
460 		reset_control_assert(rst);
461 		reset_control_deassert(rst);
462 	}
463 
464 	img_i2s_in_writel(i2s, 0, IMG_I2S_IN_CTL);
465 
466 	for (i = 0; i < i2s->max_i2s_chan; i++)
467 		img_i2s_in_ch_writel(i2s, i,
468 			(4 << IMG_I2S_IN_CH_CTL_CCDEL_SHIFT) |
469 			IMG_I2S_IN_CH_CTL_JUST_MASK |
470 			IMG_I2S_IN_CH_CTL_FW_MASK, IMG_I2S_IN_CH_CTL);
471 
472 	ret = devm_snd_soc_register_component(dev, &img_i2s_in_component,
473 						&i2s->dai_driver, 1);
474 	if (ret)
475 		goto err_clk_disable;
476 
477 	ret = devm_snd_dmaengine_pcm_register(dev, &img_i2s_in_dma_config, 0);
478 	if (ret)
479 		goto err_clk_disable;
480 
481 	return 0;
482 
483 err_clk_disable:
484 	clk_disable_unprepare(i2s->clk_sys);
485 
486 	return ret;
487 }
488 
489 static int img_i2s_in_dev_remove(struct platform_device *pdev)
490 {
491 	struct img_i2s_in *i2s = platform_get_drvdata(pdev);
492 
493 	clk_disable_unprepare(i2s->clk_sys);
494 
495 	return 0;
496 }
497 
498 static const struct of_device_id img_i2s_in_of_match[] = {
499 	{ .compatible = "img,i2s-in" },
500 	{}
501 };
502 MODULE_DEVICE_TABLE(of, img_i2s_in_of_match);
503 
504 static struct platform_driver img_i2s_in_driver = {
505 	.driver = {
506 		.name = "img-i2s-in",
507 		.of_match_table = img_i2s_in_of_match
508 	},
509 	.probe = img_i2s_in_probe,
510 	.remove = img_i2s_in_dev_remove
511 };
512 module_platform_driver(img_i2s_in_driver);
513 
514 MODULE_AUTHOR("Damien Horsley <Damien.Horsley@imgtec.com>");
515 MODULE_DESCRIPTION("IMG I2S Input Driver");
516 MODULE_LICENSE("GPL v2");
517