mpc5200_dma.c 15.8 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 11 12 13 14 15 16 17 18 19 20 21 22
 */

#include <linux/module.h>
#include <linux/of_device.h>

#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
 */
23
static irqreturn_t psc_dma_status_irq(int irq, void *_psc_dma)
24
{
25 26
	struct psc_dma *psc_dma = _psc_dma;
	struct mpc52xx_psc __iomem *regs = psc_dma->psc_regs;
27 28 29 30 31
	u16 isr;

	isr = in_be16(&regs->mpc52xx_psc_isr);

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

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

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

	return IRQ_HANDLED;
}

/**
45
 * psc_dma_bcom_enqueue_next_buffer - Enqueue another audio buffer
46 47 48 49 50 51 52 53
 * @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
 */
54
static void psc_dma_bcom_enqueue_next_buffer(struct psc_dma_stream *s)
55 56 57 58 59 60 61 62 63 64 65 66 67 68 69
{
	struct bcom_bd *bd;

	/* Prepare and enqueue the next buffer descriptor */
	bd = bcom_prepare_next_buffer(s->bcom_task);
	bd->status = s->period_bytes;
	bd->data[0] = s->period_next_pt;
	bcom_submit_next_buffer(s->bcom_task, NULL);

	/* Update for next period */
	s->period_next_pt += s->period_bytes;
	if (s->period_next_pt >= s->period_end)
		s->period_next_pt = s->period_start;
}

70 71
static void psc_dma_bcom_enqueue_tx(struct psc_dma_stream *s)
{
72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88
	if (s->appl_ptr > s->runtime->control->appl_ptr) {
		/*
		 * In this case s->runtime->control->appl_ptr has wrapped around.
		 * Play the data to the end of the boundary, then wrap our own
		 * appl_ptr back around.
		 */
		while (s->appl_ptr < s->runtime->boundary) {
			if (bcom_queue_full(s->bcom_task))
				return;

			s->appl_ptr += s->period_size;

			psc_dma_bcom_enqueue_next_buffer(s);
		}
		s->appl_ptr -= s->runtime->boundary;
	}

89 90 91 92 93 94 95 96 97 98 99
	while (s->appl_ptr < s->runtime->control->appl_ptr) {

		if (bcom_queue_full(s->bcom_task))
			return;

		s->appl_ptr += s->period_size;

		psc_dma_bcom_enqueue_next_buffer(s);
	}
}

100
/* Bestcomm DMA irq handler */
101
static irqreturn_t psc_dma_bcom_irq_tx(int irq, void *_psc_dma_stream)
102
{
103
	struct psc_dma_stream *s = _psc_dma_stream;
104

105
	spin_lock(&s->psc_dma->lock);
106 107 108 109
	/* 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);
110

111 112 113 114
		s->period_current_pt += s->period_bytes;
		if (s->period_current_pt >= s->period_end)
			s->period_current_pt = s->period_start;
	}
115 116
	psc_dma_bcom_enqueue_tx(s);
	spin_unlock(&s->psc_dma->lock);
117 118 119 120 121 122 123 124 125

	/* 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;
}

126
static irqreturn_t psc_dma_bcom_irq_rx(int irq, void *_psc_dma_stream)
127
{
128
	struct psc_dma_stream *s = _psc_dma_stream;
129

130 131 132 133 134
	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);
135

136 137 138 139 140
		s->period_current_pt += s->period_bytes;
		if (s->period_current_pt >= s->period_end)
			s->period_current_pt = s->period_start;

		psc_dma_bcom_enqueue_next_buffer(s);
141
	}
142
	spin_unlock(&s->psc_dma->lock);
143

144 145 146 147 148 149
	/* 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;
150 151
}

152
static int psc_dma_hw_free(struct snd_pcm_substream *substream)
153 154 155 156 157 158
{
	snd_pcm_set_runtime_buffer(substream, NULL);
	return 0;
}

/**
159
 * psc_dma_trigger: start and stop the DMA transfer.
160 161 162 163
 *
 * This function is called by ALSA to start, stop, pause, and resume the DMA
 * transfer of data.
 */
164
static int psc_dma_trigger(struct snd_pcm_substream *substream, int cmd)
165 166
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
167
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
168
	struct snd_pcm_runtime *runtime = substream->runtime;
169 170
	struct psc_dma_stream *s;
	struct mpc52xx_psc __iomem *regs = psc_dma->psc_regs;
171 172
	u16 imr;
	unsigned long flags;
173
	int i;
174 175

	if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE)
176
		s = &psc_dma->capture;
177
	else
178
		s = &psc_dma->playback;
179

180
	dev_dbg(psc_dma->dev, "psc_dma_trigger(substream=%p, cmd=%i)"
181 182 183 184 185 186 187 188 189 190 191 192
		" stream_id=%i\n",
		substream, cmd, substream->pstr->stream);

	switch (cmd) {
	case SNDRV_PCM_TRIGGER_START:
		s->period_bytes = frames_to_bytes(runtime,
						  runtime->period_size);
		s->period_start = virt_to_phys(runtime->dma_area);
		s->period_end = s->period_start +
				(s->period_bytes * runtime->periods);
		s->period_next_pt = s->period_start;
		s->period_current_pt = s->period_start;
193
		s->period_size = runtime->period_size;
194 195
		s->active = 1;

196 197 198 199 200 201 202 203 204 205 206 207
		/* track appl_ptr so that we have a better chance of detecting
		 * end of stream and not over running it.
		 */
		s->runtime = runtime;
		s->appl_ptr = s->runtime->control->appl_ptr -
				(runtime->period_size * runtime->periods);

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

208
		if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE) {
209 210 211 212
			bcom_gen_bd_rx_reset(s->bcom_task);
			for (i = 0; i < runtime->periods; i++)
				if (!bcom_queue_full(s->bcom_task))
					psc_dma_bcom_enqueue_next_buffer(s);
213
		} else {
214 215
			bcom_gen_bd_tx_reset(s->bcom_task);
			psc_dma_bcom_enqueue_tx(s);
216 217 218
		}

		bcom_enable(s->bcom_task);
219
		spin_unlock_irqrestore(&psc_dma->lock, flags);
220

221 222
		out_8(&regs->command, MPC52xx_PSC_RST_ERR_STAT);

223 224 225 226 227
		break;

	case SNDRV_PCM_TRIGGER_STOP:
		s->active = 0;

228
		spin_lock_irqsave(&psc_dma->lock, flags);
229
		bcom_disable(s->bcom_task);
230 231 232 233 234
		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);
235 236 237 238

		break;

	default:
239
		dev_dbg(psc_dma->dev, "invalid command\n");
240 241 242 243 244
		return -EINVAL;
	}

	/* Update interrupt enable settings */
	imr = 0;
245
	if (psc_dma->playback.active)
246
		imr |= MPC52xx_PSC_IMR_TXEMP;
247
	if (psc_dma->capture.active)
248
		imr |= MPC52xx_PSC_IMR_ORERR;
249
	out_be16(&regs->isr_imr.imr, psc_dma->imr | imr);
250 251 252 253 254 255 256 257 258 259 260 261 262

	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
 */

263
static const struct snd_pcm_hardware psc_dma_hardware = {
264 265 266 267
	.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 |
268
		SNDRV_PCM_FMTBIT_S24_BE | SNDRV_PCM_FMTBIT_S32_BE,
269 270
	.rate_min = 8000,
	.rate_max = 48000,
271
	.channels_min = 1,
272 273 274 275 276 277
	.channels_max = 2,
	.period_bytes_max	= 1024 * 1024,
	.period_bytes_min	= 32,
	.periods_min		= 2,
	.periods_max		= 256,
	.buffer_bytes_max	= 2 * 1024 * 1024,
278
	.fifo_size		= 512,
279 280
};

281
static int psc_dma_open(struct snd_pcm_substream *substream)
282
{
283
	struct snd_pcm_runtime *runtime = substream->runtime;
284
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
285 286
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
	struct psc_dma_stream *s;
287
	int rc;
288

289
	dev_dbg(psc_dma->dev, "psc_dma_open(substream=%p)\n", substream);
290 291

	if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE)
292
		s = &psc_dma->capture;
293
	else
294
		s = &psc_dma->playback;
295

296 297 298 299 300 301 302 303
	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;
	}
304 305 306 307 308

	s->stream = substream;
	return 0;
}

309
static int psc_dma_close(struct snd_pcm_substream *substream)
310 311
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
312 313
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
	struct psc_dma_stream *s;
314

315
	dev_dbg(psc_dma->dev, "psc_dma_close(substream=%p)\n", substream);
316 317

	if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE)
318
		s = &psc_dma->capture;
319
	else
320
		s = &psc_dma->playback;
321

322 323 324 325 326 327 328
	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 */
	}
329 330 331 332 333
	s->stream = NULL;
	return 0;
}

static snd_pcm_uframes_t
334
psc_dma_pointer(struct snd_pcm_substream *substream)
335 336
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
337 338
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
	struct psc_dma_stream *s;
339 340 341
	dma_addr_t count;

	if (substream->pstr->stream == SNDRV_PCM_STREAM_CAPTURE)
342
		s = &psc_dma->capture;
343
	else
344
		s = &psc_dma->playback;
345 346 347 348 349 350

	count = s->period_current_pt - s->period_start;

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

351 352 353 354 355 356 357 358 359 360 361 362 363
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,
364
	.ioctl		= snd_pcm_lib_ioctl,
365 366 367
	.pointer	= psc_dma_pointer,
	.trigger	= psc_dma_trigger,
	.hw_params	= psc_dma_hw_params,
368 369
};

370 371
static u64 psc_dma_dmamask = 0xffffffff;
static int psc_dma_new(struct snd_card *card, struct snd_soc_dai *dai,
372 373 374
			   struct snd_pcm *pcm)
{
	struct snd_soc_pcm_runtime *rtd = pcm->private_data;
375 376
	struct psc_dma *psc_dma = rtd->dai->cpu_dai->private_data;
	size_t size = psc_dma_hardware.buffer_bytes_max;
377 378
	int rc = 0;

379
	dev_dbg(rtd->socdev->dev, "psc_dma_new(card=%p, dai=%p, pcm=%p)\n",
380 381 382
		card, dai, pcm);

	if (!card->dev->dma_mask)
383
		card->dev->dma_mask = &psc_dma_dmamask;
384 385 386 387
	if (!card->dev->coherent_dma_mask)
		card->dev->coherent_dma_mask = 0xffffffff;

	if (pcm->streams[0].substream) {
388 389
		rc = snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV, pcm->card->dev,
				size, &pcm->streams[0].substream->dma_buffer);
390 391 392 393 394
		if (rc)
			goto playback_alloc_err;
	}

	if (pcm->streams[1].substream) {
395 396
		rc = snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV, pcm->card->dev,
				size, &pcm->streams[1].substream->dma_buffer);
397 398 399 400
		if (rc)
			goto capture_alloc_err;
	}

401 402 403
	if (rtd->socdev->card->codec->ac97)
		rtd->socdev->card->codec->ac97->private_data = psc_dma;

404 405 406 407 408
	return 0;

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

410 411
 playback_alloc_err:
	dev_err(card->dev, "Cannot allocate buffer(s)\n");
412

413 414 415
	return -ENOMEM;
}

416
static void psc_dma_free(struct snd_pcm *pcm)
417 418 419 420 421
{
	struct snd_soc_pcm_runtime *rtd = pcm->private_data;
	struct snd_pcm_substream *substream;
	int stream;

422
	dev_dbg(rtd->socdev->dev, "psc_dma_free(pcm=%p)\n", pcm);
423 424 425 426 427 428 429 430 431 432 433

	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;
		}
	}
}

434
struct snd_soc_platform mpc5200_audio_dma_platform = {
435
	.name		= "mpc5200-psc-audio",
436 437 438
	.pcm_ops	= &psc_dma_ops,
	.pcm_new	= &psc_dma_new,
	.pcm_free	= &psc_dma_free,
439
};
440 441 442 443 444 445 446 447 448 449
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;
450
	int ret;
451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466

	/* Fetch the registers and IRQ of the PSC */
	irq = irq_of_parse_and_map(op->node, 0);
	if (of_address_to_resource(op->node, 0, &res)) {
		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) {
467 468
		ret = -ENOMEM;
		goto out_unmap;
469 470 471 472
	}

	/* Get the PSC ID */
	prop = of_get_property(op->node, "cell-index", &size);
473 474 475 476
	if (!prop || size < sizeof *prop) {
		ret = -ENODEV;
		goto out_free;
	}
477 478

	spin_lock_init(&psc_dma->lock);
479
	mutex_init(&psc_dma->mutex);
480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498
	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");
499 500
		ret = -ENODEV;
		goto out_free;
501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541
	}

	/* 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);
	rc |= request_irq(psc_dma->capture.irq,
			  &psc_dma_bcom_irq_rx, IRQF_SHARED,
			  "psc-dma-capture", &psc_dma->capture);
	rc |= request_irq(psc_dma->playback.irq,
			  &psc_dma_bcom_irq_tx, IRQF_SHARED,
			  "psc-dma-playback", &psc_dma->playback);
	if (rc) {
542 543
		ret = -ENODEV;
		goto out_irq;
544
	}
545

546 547 548 549 550
	/* 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);
551 552 553 554 555 556 557 558 559
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;
560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589
}
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");