mpc5200_dma.c 14.1 KB
Newer Older
1 2 3 4 5
/*
 * Freescale MPC5200 PSC DMA
 * ALSA SoC Platform driver
 *
 * Copyright (C) 2008 Secret Lab Technologies Ltd.
6
 * Copyright (C) 2009 Jon Smirl, Digispeaker
7 8 9 10
 */

#include <linux/module.h>
#include <linux/of_device.h>
11
#include <linux/slab.h>
12 13 14 15 16 17 18 19 20 21 22 23

#include <sound/soc.h>

#include <sysdev/bestcomm/bestcomm.h>
#include <sysdev/bestcomm/gen_bd.h>
#include <asm/mpc52xx_psc.h>

#include "mpc5200_dma.h"

/*
 * Interrupt handlers
 */
24
static irqreturn_t psc_dma_status_irq(int irq, void *_psc_dma)
25
{
26 27
	struct psc_dma *psc_dma = _psc_dma;
	struct mpc52xx_psc __iomem *regs = psc_dma->psc_regs;
28 29 30 31 32
	u16 isr;

	isr = in_be16(&regs->mpc52xx_psc_isr);

	/* Playback underrun error */
33 34
	if (psc_dma->playback.active && (isr & MPC52xx_PSC_IMR_TXEMP))
		psc_dma->stats.underrun_count++;
35 36

	/* Capture overrun error */
37 38
	if (psc_dma->capture.active && (isr & MPC52xx_PSC_IMR_ORERR))
		psc_dma->stats.overrun_count++;
39

40
	out_8(&regs->command, MPC52xx_PSC_RST_ERR_STAT);
41 42 43 44 45

	return IRQ_HANDLED;
}

/**
46
 * psc_dma_bcom_enqueue_next_buffer - Enqueue another audio buffer
47 48 49 50 51 52 53 54
 * @s: pointer to stream private data structure
 *
 * Enqueues another audio period buffer into the bestcomm queue.
 *
 * Note: The routine must only be called when there is space available in
 * the queue.  Otherwise the enqueue will fail and the audio ring buffer
 * will get out of sync
 */
55
static void psc_dma_bcom_enqueue_next_buffer(struct psc_dma_stream *s)
56 57 58 59 60 61
{
	struct bcom_bd *bd;

	/* Prepare and enqueue the next buffer descriptor */
	bd = bcom_prepare_next_buffer(s->bcom_task);
	bd->status = s->period_bytes;
62
	bd->data[0] = s->runtime->dma_addr + (s->period_next * s->period_bytes);
63 64 65
	bcom_submit_next_buffer(s->bcom_task, NULL);

	/* Update for next period */
66
	s->period_next = (s->period_next + 1) % s->runtime->periods;
67 68 69
}

/* Bestcomm DMA irq handler */
70
static irqreturn_t psc_dma_bcom_irq(int irq, void *_psc_dma_stream)
71
{
72
	struct psc_dma_stream *s = _psc_dma_stream;
73

74 75 76 77 78
	spin_lock(&s->psc_dma->lock);
	/* For each finished period, dequeue the completed period buffer
	 * and enqueue a new one in it's place. */
	while (bcom_buffer_done(s->bcom_task)) {
		bcom_retrieve_buffer(s->bcom_task, NULL, NULL);
79

80
		s->period_current = (s->period_current+1) % s->runtime->periods;
81
		s->period_count++;
82 83

		psc_dma_bcom_enqueue_next_buffer(s);
84
	}
85
	spin_unlock(&s->psc_dma->lock);
86

87 88 89 90 91 92
	/* If the stream is active, then also inform the PCM middle layer
	 * of the period finished event. */
	if (s->active)
		snd_pcm_period_elapsed(s->stream);

	return IRQ_HANDLED;
93 94
}

95
static int psc_dma_hw_free(struct snd_pcm_substream *substream)
96 97 98 99 100 101
{
	snd_pcm_set_runtime_buffer(substream, NULL);
	return 0;
}

/**
102
 * psc_dma_trigger: start and stop the DMA transfer.
103 104 105 106
 *
 * This function is called by ALSA to start, stop, pause, and resume the DMA
 * transfer of data.
 */
107
static int psc_dma_trigger(struct snd_pcm_substream *substream, int cmd)
108 109
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
110
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
111
	struct snd_pcm_runtime *runtime = substream->runtime;
112
	struct psc_dma_stream *s = to_psc_dma_stream(substream, psc_dma);
113
	struct mpc52xx_psc __iomem *regs = psc_dma->psc_regs;
114 115
	u16 imr;
	unsigned long flags;
116
	int i;
117 118 119

	switch (cmd) {
	case SNDRV_PCM_TRIGGER_START:
120 121 122
		dev_dbg(psc_dma->dev, "START: stream=%i fbits=%u ps=%u #p=%u\n",
			substream->pstr->stream, runtime->frame_bits,
			(int)runtime->period_size, runtime->periods);
123 124
		s->period_bytes = frames_to_bytes(runtime,
						  runtime->period_size);
125 126
		s->period_next = 0;
		s->period_current = 0;
127
		s->active = 1;
128
		s->period_count = 0;
129 130 131 132 133 134 135
		s->runtime = runtime;

		/* Fill up the bestcomm bd queue and enable DMA.
		 * This will begin filling the PSC's fifo.
		 */
		spin_lock_irqsave(&psc_dma->lock, flags);

136
		if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE)
137
			bcom_gen_bd_rx_reset(s->bcom_task);
138
		else
139
			bcom_gen_bd_tx_reset(s->bcom_task);
140 141 142 143

		for (i = 0; i < runtime->periods; i++)
			if (!bcom_queue_full(s->bcom_task))
				psc_dma_bcom_enqueue_next_buffer(s);
144 145

		bcom_enable(s->bcom_task);
146
		spin_unlock_irqrestore(&psc_dma->lock, flags);
147

148 149
		out_8(&regs->command, MPC52xx_PSC_RST_ERR_STAT);

150 151 152
		break;

	case SNDRV_PCM_TRIGGER_STOP:
153 154
		dev_dbg(psc_dma->dev, "STOP: stream=%i periods_count=%i\n",
			substream->pstr->stream, s->period_count);
155 156
		s->active = 0;

157
		spin_lock_irqsave(&psc_dma->lock, flags);
158
		bcom_disable(s->bcom_task);
159 160 161 162 163
		if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE)
			bcom_gen_bd_rx_reset(s->bcom_task);
		else
			bcom_gen_bd_tx_reset(s->bcom_task);
		spin_unlock_irqrestore(&psc_dma->lock, flags);
164 165 166 167

		break;

	default:
168 169
		dev_dbg(psc_dma->dev, "unhandled trigger: stream=%i cmd=%i\n",
			substream->pstr->stream, cmd);
170 171 172 173 174
		return -EINVAL;
	}

	/* Update interrupt enable settings */
	imr = 0;
175
	if (psc_dma->playback.active)
176
		imr |= MPC52xx_PSC_IMR_TXEMP;
177
	if (psc_dma->capture.active)
178
		imr |= MPC52xx_PSC_IMR_ORERR;
179
	out_be16(&regs->isr_imr.imr, psc_dma->imr | imr);
180 181 182 183 184 185 186 187 188 189 190 191 192

	return 0;
}


/* ---------------------------------------------------------------------
 * The PSC DMA 'ASoC platform' driver
 *
 * Can be referenced by an 'ASoC machine' driver
 * This driver only deals with the audio bus; it doesn't have any
 * interaction with the attached codec
 */

193
static const struct snd_pcm_hardware psc_dma_hardware = {
194 195 196 197
	.info = SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_MMAP_VALID |
		SNDRV_PCM_INFO_INTERLEAVED | SNDRV_PCM_INFO_BLOCK_TRANSFER |
		SNDRV_PCM_INFO_BATCH,
	.formats = SNDRV_PCM_FMTBIT_S8 | SNDRV_PCM_FMTBIT_S16_BE |
198
		SNDRV_PCM_FMTBIT_S24_BE | SNDRV_PCM_FMTBIT_S32_BE,
199 200
	.rate_min = 8000,
	.rate_max = 48000,
201
	.channels_min = 1,
202 203 204 205 206 207
	.channels_max = 2,
	.period_bytes_max	= 1024 * 1024,
	.period_bytes_min	= 32,
	.periods_min		= 2,
	.periods_max		= 256,
	.buffer_bytes_max	= 2 * 1024 * 1024,
208
	.fifo_size		= 512,
209 210
};

211
static int psc_dma_open(struct snd_pcm_substream *substream)
212
{
213
	struct snd_pcm_runtime *runtime = substream->runtime;
214
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
215 216
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
	struct psc_dma_stream *s;
217
	int rc;
218

219
	dev_dbg(psc_dma->dev, "psc_dma_open(substream=%p)\n", substream);
220 221

	if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE)
222
		s = &psc_dma->capture;
223
	else
224
		s = &psc_dma->playback;
225

226 227 228 229 230 231 232 233
	snd_soc_set_runtime_hwparams(substream, &psc_dma_hardware);

	rc = snd_pcm_hw_constraint_integer(runtime,
		SNDRV_PCM_HW_PARAM_PERIODS);
	if (rc < 0) {
		dev_err(substream->pcm->card->dev, "invalid buffer size\n");
		return rc;
	}
234 235 236 237 238

	s->stream = substream;
	return 0;
}

239
static int psc_dma_close(struct snd_pcm_substream *substream)
240 241
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
242 243
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
	struct psc_dma_stream *s;
244

245
	dev_dbg(psc_dma->dev, "psc_dma_close(substream=%p)\n", substream);
246 247

	if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE)
248
		s = &psc_dma->capture;
249
	else
250
		s = &psc_dma->playback;
251

252 253 254 255 256 257 258
	if (!psc_dma->playback.active &&
	    !psc_dma->capture.active) {

		/* Disable all interrupts and reset the PSC */
		out_be16(&psc_dma->psc_regs->isr_imr.imr, psc_dma->imr);
		out_8(&psc_dma->psc_regs->command, 4 << 4); /* reset error */
	}
259 260 261 262 263
	s->stream = NULL;
	return 0;
}

static snd_pcm_uframes_t
264
psc_dma_pointer(struct snd_pcm_substream *substream)
265 266
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
267 268
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
	struct psc_dma_stream *s;
269 270 271
	dma_addr_t count;

	if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE)
272
		s = &psc_dma->capture;
273
	else
274
		s = &psc_dma->playback;
275

276
	count = s->period_current * s->period_bytes;
277 278 279 280

	return bytes_to_frames(substream->runtime, count);
}

281 282 283 284 285 286 287 288 289 290 291 292 293
static int
psc_dma_hw_params(struct snd_pcm_substream *substream,
			 struct snd_pcm_hw_params *params)
{
	snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer);

	return 0;
}

static struct snd_pcm_ops psc_dma_ops = {
	.open		= psc_dma_open,
	.close		= psc_dma_close,
	.hw_free	= psc_dma_hw_free,
294
	.ioctl		= snd_pcm_lib_ioctl,
295 296 297
	.pointer	= psc_dma_pointer,
	.trigger	= psc_dma_trigger,
	.hw_params	= psc_dma_hw_params,
298 299
};

300 301
static u64 psc_dma_dmamask = 0xffffffff;
static int psc_dma_new(struct snd_card *card, struct snd_soc_dai *dai,
302 303 304
			   struct snd_pcm *pcm)
{
	struct snd_soc_pcm_runtime *rtd = pcm->private_data;
305 306
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
	size_t size = psc_dma_hardware.buffer_bytes_max;
307 308
	int rc = 0;

309
	dev_dbg(rtd->socdev->dev, "psc_dma_new(card=%p, dai=%p, pcm=%p)\n",
310 311 312
		card, dai, pcm);

	if (!card->dev->dma_mask)
313
		card->dev->dma_mask = &psc_dma_dmamask;
314 315 316 317
	if (!card->dev->coherent_dma_mask)
		card->dev->coherent_dma_mask = 0xffffffff;

	if (pcm->streams[0].substream) {
318 319
		rc = snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV, pcm->card->dev,
				size, &pcm->streams[0].substream->dma_buffer);
320 321 322 323 324
		if (rc)
			goto playback_alloc_err;
	}

	if (pcm->streams[1].substream) {
325 326
		rc = snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV, pcm->card->dev,
				size, &pcm->streams[1].substream->dma_buffer);
327 328 329 330
		if (rc)
			goto capture_alloc_err;
	}

331 332 333
	if (rtd->socdev->card->codec->ac97)
		rtd->socdev->card->codec->ac97->private_data = psc_dma;

334 335 336 337 338
	return 0;

 capture_alloc_err:
	if (pcm->streams[0].substream)
		snd_dma_free_pages(&pcm->streams[0].substream->dma_buffer);
339

340 341
 playback_alloc_err:
	dev_err(card->dev, "Cannot allocate buffer(s)\n");
342

343 344 345
	return -ENOMEM;
}

346
static void psc_dma_free(struct snd_pcm *pcm)
347 348 349 350 351
{
	struct snd_soc_pcm_runtime *rtd = pcm->private_data;
	struct snd_pcm_substream *substream;
	int stream;

352
	dev_dbg(rtd->socdev->dev, "psc_dma_free(pcm=%p)\n", pcm);
353 354 355 356 357 358 359 360 361 362 363

	for (stream = 0; stream < 2; stream++) {
		substream = pcm->streams[stream].substream;
		if (substream) {
			snd_dma_free_pages(&substream->dma_buffer);
			substream->dma_buffer.area = NULL;
			substream->dma_buffer.addr = 0;
		}
	}
}

364
struct snd_soc_platform mpc5200_audio_dma_platform = {
365
	.name		= "mpc5200-psc-audio",
366 367 368
	.pcm_ops	= &psc_dma_ops,
	.pcm_new	= &psc_dma_new,
	.pcm_free	= &psc_dma_free,
369
};
370 371 372 373 374 375 376 377 378 379
EXPORT_SYMBOL_GPL(mpc5200_audio_dma_platform);

int mpc5200_audio_dma_create(struct of_device *op)
{
	phys_addr_t fifo;
	struct psc_dma *psc_dma;
	struct resource res;
	int size, irq, rc;
	const __be32 *prop;
	void __iomem *regs;
380
	int ret;
381 382

	/* Fetch the registers and IRQ of the PSC */
383 384
	irq = irq_of_parse_and_map(op->dev.of_node, 0);
	if (of_address_to_resource(op->dev.of_node, 0, &res)) {
385 386 387 388 389 390 391 392 393 394 395 396
		dev_err(&op->dev, "Missing reg property\n");
		return -ENODEV;
	}
	regs = ioremap(res.start, 1 + res.end - res.start);
	if (!regs) {
		dev_err(&op->dev, "Could not map registers\n");
		return -ENODEV;
	}

	/* Allocate and initialize the driver private data */
	psc_dma = kzalloc(sizeof *psc_dma, GFP_KERNEL);
	if (!psc_dma) {
397 398
		ret = -ENOMEM;
		goto out_unmap;
399 400 401
	}

	/* Get the PSC ID */
402
	prop = of_get_property(op->dev.of_node, "cell-index", &size);
403 404 405 406
	if (!prop || size < sizeof *prop) {
		ret = -ENODEV;
		goto out_free;
	}
407 408

	spin_lock_init(&psc_dma->lock);
409
	mutex_init(&psc_dma->mutex);
410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428
	psc_dma->id = be32_to_cpu(*prop);
	psc_dma->irq = irq;
	psc_dma->psc_regs = regs;
	psc_dma->fifo_regs = regs + sizeof *psc_dma->psc_regs;
	psc_dma->dev = &op->dev;
	psc_dma->playback.psc_dma = psc_dma;
	psc_dma->capture.psc_dma = psc_dma;
	snprintf(psc_dma->name, sizeof psc_dma->name, "PSC%u", psc_dma->id);

	/* Find the address of the fifo data registers and setup the
	 * DMA tasks */
	fifo = res.start + offsetof(struct mpc52xx_psc, buffer.buffer_32);
	psc_dma->capture.bcom_task =
		bcom_psc_gen_bd_rx_init(psc_dma->id, 10, fifo, 512);
	psc_dma->playback.bcom_task =
		bcom_psc_gen_bd_tx_init(psc_dma->id, 10, fifo);
	if (!psc_dma->capture.bcom_task ||
	    !psc_dma->playback.bcom_task) {
		dev_err(&op->dev, "Could not allocate bestcomm tasks\n");
429 430
		ret = -ENODEV;
		goto out_free;
431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464
	}

	/* Disable all interrupts and reset the PSC */
	out_be16(&psc_dma->psc_regs->isr_imr.imr, psc_dma->imr);
	 /* reset receiver */
	out_8(&psc_dma->psc_regs->command, MPC52xx_PSC_RST_RX);
	 /* reset transmitter */
	out_8(&psc_dma->psc_regs->command, MPC52xx_PSC_RST_TX);
	 /* reset error */
	out_8(&psc_dma->psc_regs->command, MPC52xx_PSC_RST_ERR_STAT);
	 /* reset mode */
	out_8(&psc_dma->psc_regs->command, MPC52xx_PSC_SEL_MODE_REG_1);

	/* Set up mode register;
	 * First write: RxRdy (FIFO Alarm) generates rx FIFO irq
	 * Second write: register Normal mode for non loopback
	 */
	out_8(&psc_dma->psc_regs->mode, 0);
	out_8(&psc_dma->psc_regs->mode, 0);

	/* Set the TX and RX fifo alarm thresholds */
	out_be16(&psc_dma->fifo_regs->rfalarm, 0x100);
	out_8(&psc_dma->fifo_regs->rfcntl, 0x4);
	out_be16(&psc_dma->fifo_regs->tfalarm, 0x100);
	out_8(&psc_dma->fifo_regs->tfcntl, 0x7);

	/* Lookup the IRQ numbers */
	psc_dma->playback.irq =
		bcom_get_task_irq(psc_dma->playback.bcom_task);
	psc_dma->capture.irq =
		bcom_get_task_irq(psc_dma->capture.bcom_task);

	rc = request_irq(psc_dma->irq, &psc_dma_status_irq, IRQF_SHARED,
			 "psc-dma-status", psc_dma);
465
	rc |= request_irq(psc_dma->capture.irq, &psc_dma_bcom_irq, IRQF_SHARED,
466
			  "psc-dma-capture", &psc_dma->capture);
467
	rc |= request_irq(psc_dma->playback.irq, &psc_dma_bcom_irq, IRQF_SHARED,
468 469
			  "psc-dma-playback", &psc_dma->playback);
	if (rc) {
470 471
		ret = -ENODEV;
		goto out_irq;
472
	}
473

474 475 476 477 478
	/* Save what we've done so it can be found again later */
	dev_set_drvdata(&op->dev, psc_dma);

	/* Tell the ASoC OF helpers about it */
	return snd_soc_register_platform(&mpc5200_audio_dma_platform);
479 480 481 482 483 484 485 486 487
out_irq:
	free_irq(psc_dma->irq, psc_dma);
	free_irq(psc_dma->capture.irq, &psc_dma->capture);
	free_irq(psc_dma->playback.irq, &psc_dma->playback);
out_free:
	kfree(psc_dma);
out_unmap:
	iounmap(regs);
	return ret;
488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517
}
EXPORT_SYMBOL_GPL(mpc5200_audio_dma_create);

int mpc5200_audio_dma_destroy(struct of_device *op)
{
	struct psc_dma *psc_dma = dev_get_drvdata(&op->dev);

	dev_dbg(&op->dev, "mpc5200_audio_dma_destroy()\n");

	snd_soc_unregister_platform(&mpc5200_audio_dma_platform);

	bcom_gen_bd_rx_release(psc_dma->capture.bcom_task);
	bcom_gen_bd_tx_release(psc_dma->playback.bcom_task);

	/* Release irqs */
	free_irq(psc_dma->irq, psc_dma);
	free_irq(psc_dma->capture.irq, &psc_dma->capture);
	free_irq(psc_dma->playback.irq, &psc_dma->playback);

	iounmap(psc_dma->psc_regs);
	kfree(psc_dma);
	dev_set_drvdata(&op->dev, NULL);

	return 0;
}
EXPORT_SYMBOL_GPL(mpc5200_audio_dma_destroy);

MODULE_AUTHOR("Grant Likely <grant.likely@secretlab.ca>");
MODULE_DESCRIPTION("Freescale MPC5200 PSC in DMA mode ASoC Driver");
MODULE_LICENSE("GPL");