aboutsummaryrefslogtreecommitdiffstats
path: root/sound/arm
diff options
context:
space:
mode:
authorAndrea Bastoni <bastoni@cs.unc.edu>2010-05-30 19:16:45 -0400
committerAndrea Bastoni <bastoni@cs.unc.edu>2010-05-30 19:16:45 -0400
commitada47b5fe13d89735805b566185f4885f5a3f750 (patch)
tree644b88f8a71896307d71438e9b3af49126ffb22b /sound/arm
parent43e98717ad40a4ae64545b5ba047c7b86aa44f4f (diff)
parent3280f21d43ee541f97f8cda5792150d2dbec20d5 (diff)
Merge branch 'wip-2.6.34' into old-private-masterarchived-private-master
Diffstat (limited to 'sound/arm')
-rw-r--r--sound/arm/Makefile2
-rw-r--r--sound/arm/aaci.c210
-rw-r--r--sound/arm/aaci.h2
-rw-r--r--sound/arm/devdma.c80
-rw-r--r--sound/arm/devdma.h3
-rw-r--r--sound/arm/pxa2xx-ac97-lib.c68
-rw-r--r--sound/arm/pxa2xx-ac97.c2
-rw-r--r--sound/arm/pxa2xx-pcm-lib.c2
8 files changed, 80 insertions, 289 deletions
diff --git a/sound/arm/Makefile b/sound/arm/Makefile
index 5a549ed6c8aa..8c0c851d4641 100644
--- a/sound/arm/Makefile
+++ b/sound/arm/Makefile
@@ -3,7 +3,7 @@
3# 3#
4 4
5obj-$(CONFIG_SND_ARMAACI) += snd-aaci.o 5obj-$(CONFIG_SND_ARMAACI) += snd-aaci.o
6snd-aaci-objs := aaci.o devdma.o 6snd-aaci-objs := aaci.o
7 7
8obj-$(CONFIG_SND_PXA2XX_PCM) += snd-pxa2xx-pcm.o 8obj-$(CONFIG_SND_PXA2XX_PCM) += snd-pxa2xx-pcm.o
9snd-pxa2xx-pcm-objs := pxa2xx-pcm.o 9snd-pxa2xx-pcm-objs := pxa2xx-pcm.o
diff --git a/sound/arm/aaci.c b/sound/arm/aaci.c
index 6c160a038b23..91acc9a243ec 100644
--- a/sound/arm/aaci.c
+++ b/sound/arm/aaci.c
@@ -18,10 +18,7 @@
18#include <linux/interrupt.h> 18#include <linux/interrupt.h>
19#include <linux/err.h> 19#include <linux/err.h>
20#include <linux/amba/bus.h> 20#include <linux/amba/bus.h>
21 21#include <linux/io.h>
22#include <asm/io.h>
23#include <asm/irq.h>
24#include <asm/sizes.h>
25 22
26#include <sound/core.h> 23#include <sound/core.h>
27#include <sound/initval.h> 24#include <sound/initval.h>
@@ -30,7 +27,6 @@
30#include <sound/pcm_params.h> 27#include <sound/pcm_params.h>
31 28
32#include "aaci.h" 29#include "aaci.h"
33#include "devdma.h"
34 30
35#define DRIVER_NAME "aaci-pl041" 31#define DRIVER_NAME "aaci-pl041"
36 32
@@ -176,14 +172,15 @@ static unsigned short aaci_ac97_read(struct snd_ac97 *ac97, unsigned short reg)
176 return v; 172 return v;
177} 173}
178 174
179static inline void aaci_chan_wait_ready(struct aaci_runtime *aacirun) 175static inline void
176aaci_chan_wait_ready(struct aaci_runtime *aacirun, unsigned long mask)
180{ 177{
181 u32 val; 178 u32 val;
182 int timeout = 5000; 179 int timeout = 5000;
183 180
184 do { 181 do {
185 val = readl(aacirun->base + AACI_SR); 182 val = readl(aacirun->base + AACI_SR);
186 } while (val & (SR_TXB|SR_RXB) && timeout--); 183 } while (val & mask && timeout--);
187} 184}
188 185
189 186
@@ -212,8 +209,10 @@ static void aaci_fifo_irq(struct aaci *aaci, int channel, u32 mask)
212 writel(0, aacirun->base + AACI_IE); 209 writel(0, aacirun->base + AACI_IE);
213 return; 210 return;
214 } 211 }
215 ptr = aacirun->ptr;
216 212
213 spin_lock(&aacirun->lock);
214
215 ptr = aacirun->ptr;
217 do { 216 do {
218 unsigned int len = aacirun->fifosz; 217 unsigned int len = aacirun->fifosz;
219 u32 val; 218 u32 val;
@@ -221,9 +220,9 @@ static void aaci_fifo_irq(struct aaci *aaci, int channel, u32 mask)
221 if (aacirun->bytes <= 0) { 220 if (aacirun->bytes <= 0) {
222 aacirun->bytes += aacirun->period; 221 aacirun->bytes += aacirun->period;
223 aacirun->ptr = ptr; 222 aacirun->ptr = ptr;
224 spin_unlock(&aaci->lock); 223 spin_unlock(&aacirun->lock);
225 snd_pcm_period_elapsed(aacirun->substream); 224 snd_pcm_period_elapsed(aacirun->substream);
226 spin_lock(&aaci->lock); 225 spin_lock(&aacirun->lock);
227 } 226 }
228 if (!(aacirun->cr & CR_EN)) 227 if (!(aacirun->cr & CR_EN))
229 break; 228 break;
@@ -249,7 +248,10 @@ static void aaci_fifo_irq(struct aaci *aaci, int channel, u32 mask)
249 ptr = aacirun->start; 248 ptr = aacirun->start;
250 } 249 }
251 } while(1); 250 } while(1);
251
252 aacirun->ptr = ptr; 252 aacirun->ptr = ptr;
253
254 spin_unlock(&aacirun->lock);
253 } 255 }
254 256
255 if (mask & ISR_URINTR) { 257 if (mask & ISR_URINTR) {
@@ -267,6 +269,8 @@ static void aaci_fifo_irq(struct aaci *aaci, int channel, u32 mask)
267 return; 269 return;
268 } 270 }
269 271
272 spin_lock(&aacirun->lock);
273
270 ptr = aacirun->ptr; 274 ptr = aacirun->ptr;
271 do { 275 do {
272 unsigned int len = aacirun->fifosz; 276 unsigned int len = aacirun->fifosz;
@@ -275,9 +279,9 @@ static void aaci_fifo_irq(struct aaci *aaci, int channel, u32 mask)
275 if (aacirun->bytes <= 0) { 279 if (aacirun->bytes <= 0) {
276 aacirun->bytes += aacirun->period; 280 aacirun->bytes += aacirun->period;
277 aacirun->ptr = ptr; 281 aacirun->ptr = ptr;
278 spin_unlock(&aaci->lock); 282 spin_unlock(&aacirun->lock);
279 snd_pcm_period_elapsed(aacirun->substream); 283 snd_pcm_period_elapsed(aacirun->substream);
280 spin_lock(&aaci->lock); 284 spin_lock(&aacirun->lock);
281 } 285 }
282 if (!(aacirun->cr & CR_EN)) 286 if (!(aacirun->cr & CR_EN))
283 break; 287 break;
@@ -305,6 +309,8 @@ static void aaci_fifo_irq(struct aaci *aaci, int channel, u32 mask)
305 } while (1); 309 } while (1);
306 310
307 aacirun->ptr = ptr; 311 aacirun->ptr = ptr;
312
313 spin_unlock(&aacirun->lock);
308 } 314 }
309} 315}
310 316
@@ -314,7 +320,6 @@ static irqreturn_t aaci_irq(int irq, void *devid)
314 u32 mask; 320 u32 mask;
315 int i; 321 int i;
316 322
317 spin_lock(&aaci->lock);
318 mask = readl(aaci->base + AACI_ALLINTS); 323 mask = readl(aaci->base + AACI_ALLINTS);
319 if (mask) { 324 if (mask) {
320 u32 m = mask; 325 u32 m = mask;
@@ -324,7 +329,6 @@ static irqreturn_t aaci_irq(int irq, void *devid)
324 } 329 }
325 } 330 }
326 } 331 }
327 spin_unlock(&aaci->lock);
328 332
329 return mask ? IRQ_HANDLED : IRQ_NONE; 333 return mask ? IRQ_HANDLED : IRQ_NONE;
330} 334}
@@ -334,63 +338,6 @@ static irqreturn_t aaci_irq(int irq, void *devid)
334/* 338/*
335 * ALSA support. 339 * ALSA support.
336 */ 340 */
337
338struct aaci_stream {
339 unsigned char codec_idx;
340 unsigned char rate_idx;
341};
342
343static struct aaci_stream aaci_streams[] = {
344 [ACSTREAM_FRONT] = {
345 .codec_idx = 0,
346 .rate_idx = AC97_RATES_FRONT_DAC,
347 },
348 [ACSTREAM_SURROUND] = {
349 .codec_idx = 0,
350 .rate_idx = AC97_RATES_SURR_DAC,
351 },
352 [ACSTREAM_LFE] = {
353 .codec_idx = 0,
354 .rate_idx = AC97_RATES_LFE_DAC,
355 },
356};
357
358static inline unsigned int aaci_rate_mask(struct aaci *aaci, int streamid)
359{
360 struct aaci_stream *s = aaci_streams + streamid;
361 return aaci->ac97_bus->codec[s->codec_idx]->rates[s->rate_idx];
362}
363
364static unsigned int rate_list[] = {
365 5512, 8000, 11025, 16000, 22050, 32000, 44100,
366 48000, 64000, 88200, 96000, 176400, 192000
367};
368
369/*
370 * Double-rate rule: we can support double rate iff channels == 2
371 * (unimplemented)
372 */
373static int
374aaci_rule_rate_by_channels(struct snd_pcm_hw_params *p, struct snd_pcm_hw_rule *rule)
375{
376 struct aaci *aaci = rule->private;
377 unsigned int rate_mask = SNDRV_PCM_RATE_8000_48000|SNDRV_PCM_RATE_5512;
378 struct snd_interval *c = hw_param_interval(p, SNDRV_PCM_HW_PARAM_CHANNELS);
379
380 switch (c->max) {
381 case 6:
382 rate_mask &= aaci_rate_mask(aaci, ACSTREAM_LFE);
383 case 4:
384 rate_mask &= aaci_rate_mask(aaci, ACSTREAM_SURROUND);
385 case 2:
386 rate_mask &= aaci_rate_mask(aaci, ACSTREAM_FRONT);
387 }
388
389 return snd_interval_list(hw_param_interval(p, rule->var),
390 ARRAY_SIZE(rate_list), rate_list,
391 rate_mask);
392}
393
394static struct snd_pcm_hardware aaci_hw_info = { 341static struct snd_pcm_hardware aaci_hw_info = {
395 .info = SNDRV_PCM_INFO_MMAP | 342 .info = SNDRV_PCM_INFO_MMAP |
396 SNDRV_PCM_INFO_MMAP_VALID | 343 SNDRV_PCM_INFO_MMAP_VALID |
@@ -404,10 +351,7 @@ static struct snd_pcm_hardware aaci_hw_info = {
404 */ 351 */
405 .formats = SNDRV_PCM_FMTBIT_S16_LE, 352 .formats = SNDRV_PCM_FMTBIT_S16_LE,
406 353
407 /* should this be continuous or knot? */ 354 /* rates are setup from the AC'97 codec */
408 .rates = SNDRV_PCM_RATE_CONTINUOUS,
409 .rate_max = 48000,
410 .rate_min = 4000,
411 .channels_min = 2, 355 .channels_min = 2,
412 .channels_max = 6, 356 .channels_max = 6,
413 .buffer_bytes_max = 64 * 1024, 357 .buffer_bytes_max = 64 * 1024,
@@ -427,6 +371,12 @@ static int __aaci_pcm_open(struct aaci *aaci,
427 aacirun->substream = substream; 371 aacirun->substream = substream;
428 runtime->private_data = aacirun; 372 runtime->private_data = aacirun;
429 runtime->hw = aaci_hw_info; 373 runtime->hw = aaci_hw_info;
374 runtime->hw.rates = aacirun->pcm->rates;
375 snd_pcm_limit_hw_rates(runtime);
376
377 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK &&
378 aacirun->pcm->r[1].slots)
379 snd_ac97_pcm_double_rate_rules(runtime);
430 380
431 /* 381 /*
432 * FIXME: ALSA specifies fifo_size in bytes. If we're in normal 382 * FIXME: ALSA specifies fifo_size in bytes. If we're in normal
@@ -437,17 +387,6 @@ static int __aaci_pcm_open(struct aaci *aaci,
437 */ 387 */
438 runtime->hw.fifo_size = aaci->fifosize * 2; 388 runtime->hw.fifo_size = aaci->fifosize * 2;
439 389
440 /*
441 * Add rule describing hardware rate dependency
442 * on the number of channels.
443 */
444 ret = snd_pcm_hw_rule_add(runtime, 0, SNDRV_PCM_HW_PARAM_RATE,
445 aaci_rule_rate_by_channels, aaci,
446 SNDRV_PCM_HW_PARAM_CHANNELS,
447 SNDRV_PCM_HW_PARAM_RATE, -1);
448 if (ret)
449 goto out;
450
451 ret = request_irq(aaci->dev->irq[0], aaci_irq, IRQF_SHARED|IRQF_DISABLED, 390 ret = request_irq(aaci->dev->irq[0], aaci_irq, IRQF_SHARED|IRQF_DISABLED,
452 DRIVER_NAME, aaci); 391 DRIVER_NAME, aaci);
453 if (ret) 392 if (ret)
@@ -492,7 +431,7 @@ static int aaci_pcm_hw_free(struct snd_pcm_substream *substream)
492 /* 431 /*
493 * Clear out the DMA and any allocated buffers. 432 * Clear out the DMA and any allocated buffers.
494 */ 433 */
495 devdma_hw_free(NULL, substream); 434 snd_pcm_lib_free_pages(substream);
496 435
497 return 0; 436 return 0;
498} 437}
@@ -502,6 +441,7 @@ static int aaci_pcm_hw_params(struct snd_pcm_substream *substream,
502 struct snd_pcm_hw_params *params) 441 struct snd_pcm_hw_params *params)
503{ 442{
504 int err; 443 int err;
444 struct aaci *aaci = substream->private_data;
505 445
506 aaci_pcm_hw_free(substream); 446 aaci_pcm_hw_free(substream);
507 if (aacirun->pcm_open) { 447 if (aacirun->pcm_open) {
@@ -509,26 +449,24 @@ static int aaci_pcm_hw_params(struct snd_pcm_substream *substream,
509 aacirun->pcm_open = 0; 449 aacirun->pcm_open = 0;
510 } 450 }
511 451
512 err = devdma_hw_alloc(NULL, substream, 452 err = snd_pcm_lib_malloc_pages(substream,
513 params_buffer_bytes(params)); 453 params_buffer_bytes(params));
514 if (err < 0) 454 if (err >= 0) {
515 goto out; 455 unsigned int rate = params_rate(params);
456 int dbl = rate > 48000;
516 457
517 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) 458 err = snd_ac97_pcm_open(aacirun->pcm, rate,
518 err = snd_ac97_pcm_open(aacirun->pcm, params_rate(params),
519 params_channels(params), 459 params_channels(params),
520 aacirun->pcm->r[0].slots); 460 aacirun->pcm->r[dbl].slots);
521 else
522 err = snd_ac97_pcm_open(aacirun->pcm, params_rate(params),
523 params_channels(params),
524 aacirun->pcm->r[0].slots);
525 461
526 if (err) 462 aacirun->pcm_open = err == 0;
527 goto out; 463 aacirun->cr = CR_FEN | CR_COMPACT | CR_SZ16;
464 aacirun->fifosz = aaci->fifosize * 4;
528 465
529 aacirun->pcm_open = 1; 466 if (aacirun->cr & CR_COMPACT)
467 aacirun->fifosz >>= 1;
468 }
530 469
531 out:
532 return err; 470 return err;
533} 471}
534 472
@@ -537,8 +475,8 @@ static int aaci_pcm_prepare(struct snd_pcm_substream *substream)
537 struct snd_pcm_runtime *runtime = substream->runtime; 475 struct snd_pcm_runtime *runtime = substream->runtime;
538 struct aaci_runtime *aacirun = runtime->private_data; 476 struct aaci_runtime *aacirun = runtime->private_data;
539 477
540 aacirun->start = (void *)runtime->dma_area; 478 aacirun->start = runtime->dma_area;
541 aacirun->end = aacirun->start + runtime->dma_bytes; 479 aacirun->end = aacirun->start + snd_pcm_lib_buffer_bytes(substream);
542 aacirun->ptr = aacirun->start; 480 aacirun->ptr = aacirun->start;
543 aacirun->period = 481 aacirun->period =
544 aacirun->bytes = frames_to_bytes(runtime, runtime->period_size); 482 aacirun->bytes = frames_to_bytes(runtime, runtime->period_size);
@@ -555,11 +493,6 @@ static snd_pcm_uframes_t aaci_pcm_pointer(struct snd_pcm_substream *substream)
555 return bytes_to_frames(runtime, bytes); 493 return bytes_to_frames(runtime, bytes);
556} 494}
557 495
558static int aaci_pcm_mmap(struct snd_pcm_substream *substream, struct vm_area_struct *vma)
559{
560 return devdma_mmap(NULL, substream, vma);
561}
562
563 496
564/* 497/*
565 * Playback specific ALSA stuff 498 * Playback specific ALSA stuff
@@ -628,7 +561,6 @@ static int aaci_pcm_open(struct snd_pcm_substream *substream)
628static int aaci_pcm_playback_hw_params(struct snd_pcm_substream *substream, 561static int aaci_pcm_playback_hw_params(struct snd_pcm_substream *substream,
629 struct snd_pcm_hw_params *params) 562 struct snd_pcm_hw_params *params)
630{ 563{
631 struct aaci *aaci = substream->private_data;
632 struct aaci_runtime *aacirun = substream->runtime->private_data; 564 struct aaci_runtime *aacirun = substream->runtime->private_data;
633 unsigned int channels = params_channels(params); 565 unsigned int channels = params_channels(params);
634 int ret; 566 int ret;
@@ -642,14 +574,9 @@ static int aaci_pcm_playback_hw_params(struct snd_pcm_substream *substream,
642 * Enable FIFO, compact mode, 16 bits per sample. 574 * Enable FIFO, compact mode, 16 bits per sample.
643 * FIXME: double rate slots? 575 * FIXME: double rate slots?
644 */ 576 */
645 if (ret >= 0) { 577 if (ret >= 0)
646 aacirun->cr = CR_FEN | CR_COMPACT | CR_SZ16;
647 aacirun->cr |= channels_to_txmask[channels]; 578 aacirun->cr |= channels_to_txmask[channels];
648 579
649 aacirun->fifosz = aaci->fifosize * 4;
650 if (aacirun->cr & CR_COMPACT)
651 aacirun->fifosz >>= 1;
652 }
653 return ret; 580 return ret;
654} 581}
655 582
@@ -661,7 +588,7 @@ static void aaci_pcm_playback_stop(struct aaci_runtime *aacirun)
661 ie &= ~(IE_URIE|IE_TXIE); 588 ie &= ~(IE_URIE|IE_TXIE);
662 writel(ie, aacirun->base + AACI_IE); 589 writel(ie, aacirun->base + AACI_IE);
663 aacirun->cr &= ~CR_EN; 590 aacirun->cr &= ~CR_EN;
664 aaci_chan_wait_ready(aacirun); 591 aaci_chan_wait_ready(aacirun, SR_TXB);
665 writel(aacirun->cr, aacirun->base + AACI_TXCR); 592 writel(aacirun->cr, aacirun->base + AACI_TXCR);
666} 593}
667 594
@@ -669,7 +596,7 @@ static void aaci_pcm_playback_start(struct aaci_runtime *aacirun)
669{ 596{
670 u32 ie; 597 u32 ie;
671 598
672 aaci_chan_wait_ready(aacirun); 599 aaci_chan_wait_ready(aacirun, SR_TXB);
673 aacirun->cr |= CR_EN; 600 aacirun->cr |= CR_EN;
674 601
675 ie = readl(aacirun->base + AACI_IE); 602 ie = readl(aacirun->base + AACI_IE);
@@ -680,12 +607,12 @@ static void aaci_pcm_playback_start(struct aaci_runtime *aacirun)
680 607
681static int aaci_pcm_playback_trigger(struct snd_pcm_substream *substream, int cmd) 608static int aaci_pcm_playback_trigger(struct snd_pcm_substream *substream, int cmd)
682{ 609{
683 struct aaci *aaci = substream->private_data;
684 struct aaci_runtime *aacirun = substream->runtime->private_data; 610 struct aaci_runtime *aacirun = substream->runtime->private_data;
685 unsigned long flags; 611 unsigned long flags;
686 int ret = 0; 612 int ret = 0;
687 613
688 spin_lock_irqsave(&aaci->lock, flags); 614 spin_lock_irqsave(&aacirun->lock, flags);
615
689 switch (cmd) { 616 switch (cmd) {
690 case SNDRV_PCM_TRIGGER_START: 617 case SNDRV_PCM_TRIGGER_START:
691 aaci_pcm_playback_start(aacirun); 618 aaci_pcm_playback_start(aacirun);
@@ -712,7 +639,8 @@ static int aaci_pcm_playback_trigger(struct snd_pcm_substream *substream, int cm
712 default: 639 default:
713 ret = -EINVAL; 640 ret = -EINVAL;
714 } 641 }
715 spin_unlock_irqrestore(&aaci->lock, flags); 642
643 spin_unlock_irqrestore(&aacirun->lock, flags);
716 644
717 return ret; 645 return ret;
718} 646}
@@ -726,29 +654,19 @@ static struct snd_pcm_ops aaci_playback_ops = {
726 .prepare = aaci_pcm_prepare, 654 .prepare = aaci_pcm_prepare,
727 .trigger = aaci_pcm_playback_trigger, 655 .trigger = aaci_pcm_playback_trigger,
728 .pointer = aaci_pcm_pointer, 656 .pointer = aaci_pcm_pointer,
729 .mmap = aaci_pcm_mmap,
730}; 657};
731 658
732static int aaci_pcm_capture_hw_params(struct snd_pcm_substream *substream, 659static int aaci_pcm_capture_hw_params(struct snd_pcm_substream *substream,
733 struct snd_pcm_hw_params *params) 660 struct snd_pcm_hw_params *params)
734{ 661{
735 struct aaci *aaci = substream->private_data;
736 struct aaci_runtime *aacirun = substream->runtime->private_data; 662 struct aaci_runtime *aacirun = substream->runtime->private_data;
737 int ret; 663 int ret;
738 664
739 ret = aaci_pcm_hw_params(substream, aacirun, params); 665 ret = aaci_pcm_hw_params(substream, aacirun, params);
740 666 if (ret >= 0)
741 if (ret >= 0) {
742 aacirun->cr = CR_FEN | CR_COMPACT | CR_SZ16;
743
744 /* Line in record: slot 3 and 4 */ 667 /* Line in record: slot 3 and 4 */
745 aacirun->cr |= CR_SL3 | CR_SL4; 668 aacirun->cr |= CR_SL3 | CR_SL4;
746 669
747 aacirun->fifosz = aaci->fifosize * 4;
748
749 if (aacirun->cr & CR_COMPACT)
750 aacirun->fifosz >>= 1;
751 }
752 return ret; 670 return ret;
753} 671}
754 672
@@ -756,7 +674,7 @@ static void aaci_pcm_capture_stop(struct aaci_runtime *aacirun)
756{ 674{
757 u32 ie; 675 u32 ie;
758 676
759 aaci_chan_wait_ready(aacirun); 677 aaci_chan_wait_ready(aacirun, SR_RXB);
760 678
761 ie = readl(aacirun->base + AACI_IE); 679 ie = readl(aacirun->base + AACI_IE);
762 ie &= ~(IE_ORIE | IE_RXIE); 680 ie &= ~(IE_ORIE | IE_RXIE);
@@ -771,7 +689,7 @@ static void aaci_pcm_capture_start(struct aaci_runtime *aacirun)
771{ 689{
772 u32 ie; 690 u32 ie;
773 691
774 aaci_chan_wait_ready(aacirun); 692 aaci_chan_wait_ready(aacirun, SR_RXB);
775 693
776#ifdef DEBUG 694#ifdef DEBUG
777 /* RX Timeout value: bits 28:17 in RXCR */ 695 /* RX Timeout value: bits 28:17 in RXCR */
@@ -788,12 +706,11 @@ static void aaci_pcm_capture_start(struct aaci_runtime *aacirun)
788 706
789static int aaci_pcm_capture_trigger(struct snd_pcm_substream *substream, int cmd) 707static int aaci_pcm_capture_trigger(struct snd_pcm_substream *substream, int cmd)
790{ 708{
791 struct aaci *aaci = substream->private_data;
792 struct aaci_runtime *aacirun = substream->runtime->private_data; 709 struct aaci_runtime *aacirun = substream->runtime->private_data;
793 unsigned long flags; 710 unsigned long flags;
794 int ret = 0; 711 int ret = 0;
795 712
796 spin_lock_irqsave(&aaci->lock, flags); 713 spin_lock_irqsave(&aacirun->lock, flags);
797 714
798 switch (cmd) { 715 switch (cmd) {
799 case SNDRV_PCM_TRIGGER_START: 716 case SNDRV_PCM_TRIGGER_START:
@@ -822,7 +739,7 @@ static int aaci_pcm_capture_trigger(struct snd_pcm_substream *substream, int cmd
822 ret = -EINVAL; 739 ret = -EINVAL;
823 } 740 }
824 741
825 spin_unlock_irqrestore(&aaci->lock, flags); 742 spin_unlock_irqrestore(&aacirun->lock, flags);
826 743
827 return ret; 744 return ret;
828} 745}
@@ -854,7 +771,6 @@ static struct snd_pcm_ops aaci_capture_ops = {
854 .prepare = aaci_pcm_capture_prepare, 771 .prepare = aaci_pcm_capture_prepare,
855 .trigger = aaci_pcm_capture_trigger, 772 .trigger = aaci_pcm_capture_trigger,
856 .pointer = aaci_pcm_pointer, 773 .pointer = aaci_pcm_pointer,
857 .mmap = aaci_pcm_mmap,
858}; 774};
859 775
860/* 776/*
@@ -906,6 +822,12 @@ static struct ac97_pcm ac97_defs[] __devinitdata = {
906 (1 << AC97_SLOT_PCM_SRIGHT) | 822 (1 << AC97_SLOT_PCM_SRIGHT) |
907 (1 << AC97_SLOT_LFE), 823 (1 << AC97_SLOT_LFE),
908 }, 824 },
825 [1] = {
826 .slots = (1 << AC97_SLOT_PCM_LEFT) |
827 (1 << AC97_SLOT_PCM_RIGHT) |
828 (1 << AC97_SLOT_PCM_LEFT_0) |
829 (1 << AC97_SLOT_PCM_RIGHT_0),
830 },
909 }, 831 },
910 }, 832 },
911 [1] = { /* PCM in */ 833 [1] = { /* PCM in */
@@ -941,7 +863,6 @@ static int __devinit aaci_probe_ac97(struct aaci *aaci)
941 struct snd_ac97 *ac97; 863 struct snd_ac97 *ac97;
942 int ret; 864 int ret;
943 865
944 writel(0, aaci->base + AC97_POWERDOWN);
945 /* 866 /*
946 * Assert AACIRESET for 2us 867 * Assert AACIRESET for 2us
947 */ 868 */
@@ -1018,7 +939,6 @@ static struct aaci * __devinit aaci_init_card(struct amba_device *dev)
1018 939
1019 aaci = card->private_data; 940 aaci = card->private_data;
1020 mutex_init(&aaci->ac97_sem); 941 mutex_init(&aaci->ac97_sem);
1021 spin_lock_init(&aaci->lock);
1022 aaci->card = card; 942 aaci->card = card;
1023 aaci->dev = dev; 943 aaci->dev = dev;
1024 944
@@ -1044,6 +964,8 @@ static int __devinit aaci_init_pcm(struct aaci *aaci)
1044 964
1045 snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &aaci_playback_ops); 965 snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &aaci_playback_ops);
1046 snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &aaci_capture_ops); 966 snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &aaci_capture_ops);
967 snd_pcm_lib_preallocate_pages_for_all(pcm, SNDRV_DMA_TYPE_DEV,
968 NULL, 0, 64 * 1024);
1047 } 969 }
1048 970
1049 return ret; 971 return ret;
@@ -1103,12 +1025,14 @@ static int __devinit aaci_probe(struct amba_device *dev, struct amba_id *id)
1103 /* 1025 /*
1104 * Playback uses AACI channel 0 1026 * Playback uses AACI channel 0
1105 */ 1027 */
1028 spin_lock_init(&aaci->playback.lock);
1106 aaci->playback.base = aaci->base + AACI_CSCH1; 1029 aaci->playback.base = aaci->base + AACI_CSCH1;
1107 aaci->playback.fifo = aaci->base + AACI_DR1; 1030 aaci->playback.fifo = aaci->base + AACI_DR1;
1108 1031
1109 /* 1032 /*
1110 * Capture uses AACI channel 0 1033 * Capture uses AACI channel 0
1111 */ 1034 */
1035 spin_lock_init(&aaci->capture.lock);
1112 aaci->capture.base = aaci->base + AACI_CSCH1; 1036 aaci->capture.base = aaci->base + AACI_CSCH1;
1113 aaci->capture.fifo = aaci->base + AACI_DR1; 1037 aaci->capture.fifo = aaci->base + AACI_DR1;
1114 1038
@@ -1122,7 +1046,11 @@ static int __devinit aaci_probe(struct amba_device *dev, struct amba_id *id)
1122 1046
1123 writel(0x1fff, aaci->base + AACI_INTCLR); 1047 writel(0x1fff, aaci->base + AACI_INTCLR);
1124 writel(aaci->maincr, aaci->base + AACI_MAINCR); 1048 writel(aaci->maincr, aaci->base + AACI_MAINCR);
1125 1049 /*
1050 * Fix: ac97 read back fail errors by reading
1051 * from any arbitrary aaci register.
1052 */
1053 readl(aaci->base + AACI_CSCH1);
1126 ret = aaci_probe_ac97(aaci); 1054 ret = aaci_probe_ac97(aaci);
1127 if (ret) 1055 if (ret)
1128 goto out; 1056 goto out;
diff --git a/sound/arm/aaci.h b/sound/arm/aaci.h
index 924f69c1c44c..6a4a2eebdda1 100644
--- a/sound/arm/aaci.h
+++ b/sound/arm/aaci.h
@@ -202,6 +202,7 @@
202struct aaci_runtime { 202struct aaci_runtime {
203 void __iomem *base; 203 void __iomem *base;
204 void __iomem *fifo; 204 void __iomem *fifo;
205 spinlock_t lock;
205 206
206 struct ac97_pcm *pcm; 207 struct ac97_pcm *pcm;
207 int pcm_open; 208 int pcm_open;
@@ -232,7 +233,6 @@ struct aaci {
232 struct snd_ac97 *ac97; 233 struct snd_ac97 *ac97;
233 234
234 u32 maincr; 235 u32 maincr;
235 spinlock_t lock;
236 236
237 struct aaci_runtime playback; 237 struct aaci_runtime playback;
238 struct aaci_runtime capture; 238 struct aaci_runtime capture;
diff --git a/sound/arm/devdma.c b/sound/arm/devdma.c
deleted file mode 100644
index 9d1e6665b546..000000000000
--- a/sound/arm/devdma.c
+++ /dev/null
@@ -1,80 +0,0 @@
1/*
2 * linux/sound/arm/devdma.c
3 *
4 * Copyright (C) 2003-2004 Russell King, All rights reserved.
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
9 *
10 * ARM DMA shim for ALSA.
11 */
12#include <linux/device.h>
13#include <linux/dma-mapping.h>
14
15#include <sound/core.h>
16#include <sound/pcm.h>
17
18#include "devdma.h"
19
20void devdma_hw_free(struct device *dev, struct snd_pcm_substream *substream)
21{
22 struct snd_pcm_runtime *runtime = substream->runtime;
23 struct snd_dma_buffer *buf = runtime->dma_buffer_p;
24
25 if (runtime->dma_area == NULL)
26 return;
27
28 if (buf != &substream->dma_buffer) {
29 dma_free_coherent(buf->dev.dev, buf->bytes, buf->area, buf->addr);
30 kfree(runtime->dma_buffer_p);
31 }
32
33 snd_pcm_set_runtime_buffer(substream, NULL);
34}
35
36int devdma_hw_alloc(struct device *dev, struct snd_pcm_substream *substream, size_t size)
37{
38 struct snd_pcm_runtime *runtime = substream->runtime;
39 struct snd_dma_buffer *buf = runtime->dma_buffer_p;
40 int ret = 0;
41
42 if (buf) {
43 if (buf->bytes >= size)
44 goto out;
45 devdma_hw_free(dev, substream);
46 }
47
48 if (substream->dma_buffer.area != NULL && substream->dma_buffer.bytes >= size) {
49 buf = &substream->dma_buffer;
50 } else {
51 buf = kmalloc(sizeof(struct snd_dma_buffer), GFP_KERNEL);
52 if (!buf)
53 goto nomem;
54
55 buf->dev.type = SNDRV_DMA_TYPE_DEV;
56 buf->dev.dev = dev;
57 buf->area = dma_alloc_coherent(dev, size, &buf->addr, GFP_KERNEL);
58 buf->bytes = size;
59 buf->private_data = NULL;
60
61 if (!buf->area)
62 goto free;
63 }
64 snd_pcm_set_runtime_buffer(substream, buf);
65 ret = 1;
66 out:
67 runtime->dma_bytes = size;
68 return ret;
69
70 free:
71 kfree(buf);
72 nomem:
73 return -ENOMEM;
74}
75
76int devdma_mmap(struct device *dev, struct snd_pcm_substream *substream, struct vm_area_struct *vma)
77{
78 struct snd_pcm_runtime *runtime = substream->runtime;
79 return dma_mmap_coherent(dev, vma, runtime->dma_area, runtime->dma_addr, runtime->dma_bytes);
80}
diff --git a/sound/arm/devdma.h b/sound/arm/devdma.h
deleted file mode 100644
index d025329c8a0f..000000000000
--- a/sound/arm/devdma.h
+++ /dev/null
@@ -1,3 +0,0 @@
1void devdma_hw_free(struct device *dev, struct snd_pcm_substream *substream);
2int devdma_hw_alloc(struct device *dev, struct snd_pcm_substream *substream, size_t size);
3int devdma_mmap(struct device *dev, struct snd_pcm_substream *substream, struct vm_area_struct *vma);
diff --git a/sound/arm/pxa2xx-ac97-lib.c b/sound/arm/pxa2xx-ac97-lib.c
index 6fdca97186e7..88eec3847df2 100644
--- a/sound/arm/pxa2xx-ac97-lib.c
+++ b/sound/arm/pxa2xx-ac97-lib.c
@@ -22,7 +22,6 @@
22 22
23#include <asm/irq.h> 23#include <asm/irq.h>
24#include <mach/regs-ac97.h> 24#include <mach/regs-ac97.h>
25#include <mach/pxa2xx-gpio.h>
26#include <mach/audio.h> 25#include <mach/audio.h>
27 26
28static DEFINE_MUTEX(car_mutex); 27static DEFINE_MUTEX(car_mutex);
@@ -32,6 +31,8 @@ static struct clk *ac97_clk;
32static struct clk *ac97conf_clk; 31static struct clk *ac97conf_clk;
33static int reset_gpio; 32static int reset_gpio;
34 33
34extern void pxa27x_assert_ac97reset(int reset_gpio, int on);
35
35/* 36/*
36 * Beware PXA27x bugs: 37 * Beware PXA27x bugs:
37 * 38 *
@@ -42,45 +43,6 @@ static int reset_gpio;
42 * 1 jiffy timeout if interrupt never comes). 43 * 1 jiffy timeout if interrupt never comes).
43 */ 44 */
44 45
45enum {
46 RESETGPIO_FORCE_HIGH,
47 RESETGPIO_FORCE_LOW,
48 RESETGPIO_NORMAL_ALTFUNC
49};
50
51/**
52 * set_resetgpio_mode - computes and sets the AC97_RESET gpio mode on PXA
53 * @mode: chosen action
54 *
55 * As the PXA27x CPUs suffer from a AC97 bug, a manual control of the reset line
56 * must be done to insure proper work of AC97 reset line. This function
57 * computes the correct gpio_mode for further use by reset functions, and
58 * applied the change through pxa_gpio_mode.
59 */
60static void set_resetgpio_mode(int resetgpio_action)
61{
62 int mode = 0;
63
64 if (reset_gpio)
65 switch (resetgpio_action) {
66 case RESETGPIO_NORMAL_ALTFUNC:
67 if (reset_gpio == 113)
68 mode = 113 | GPIO_ALT_FN_2_OUT;
69 if (reset_gpio == 95)
70 mode = 95 | GPIO_ALT_FN_1_OUT;
71 break;
72 case RESETGPIO_FORCE_LOW:
73 mode = reset_gpio | GPIO_OUT | GPIO_DFLT_LOW;
74 break;
75 case RESETGPIO_FORCE_HIGH:
76 mode = reset_gpio | GPIO_OUT | GPIO_DFLT_HIGH;
77 break;
78 };
79
80 if (mode)
81 pxa_gpio_mode(mode);
82}
83
84unsigned short pxa2xx_ac97_read(struct snd_ac97 *ac97, unsigned short reg) 46unsigned short pxa2xx_ac97_read(struct snd_ac97 *ac97, unsigned short reg)
85{ 47{
86 unsigned short val = -1; 48 unsigned short val = -1;
@@ -174,12 +136,11 @@ static inline void pxa_ac97_warm_pxa27x(void)
174{ 136{
175 gsr_bits = 0; 137 gsr_bits = 0;
176 138
177 /* warm reset broken on Bulverde, 139 /* warm reset broken on Bulverde, so manually keep AC97 reset high */
178 so manually keep AC97 reset high */ 140 pxa27x_assert_ac97reset(reset_gpio, 1);
179 set_resetgpio_mode(RESETGPIO_FORCE_HIGH);
180 udelay(10); 141 udelay(10);
181 GCR |= GCR_WARM_RST; 142 GCR |= GCR_WARM_RST;
182 set_resetgpio_mode(RESETGPIO_NORMAL_ALTFUNC); 143 pxa27x_assert_ac97reset(reset_gpio, 0);
183 udelay(500); 144 udelay(500);
184} 145}
185 146
@@ -345,16 +306,6 @@ EXPORT_SYMBOL_GPL(pxa2xx_ac97_hw_suspend);
345 306
346int pxa2xx_ac97_hw_resume(void) 307int pxa2xx_ac97_hw_resume(void)
347{ 308{
348 if (cpu_is_pxa25x() || cpu_is_pxa27x()) {
349 pxa_gpio_mode(GPIO31_SYNC_AC97_MD);
350 pxa_gpio_mode(GPIO30_SDATA_OUT_AC97_MD);
351 pxa_gpio_mode(GPIO28_BITCLK_AC97_MD);
352 pxa_gpio_mode(GPIO29_SDATA_IN_AC97_MD);
353 }
354 if (cpu_is_pxa27x()) {
355 /* Use GPIO 113 or 95 as AC97 Reset on Bulverde */
356 set_resetgpio_mode(RESETGPIO_NORMAL_ALTFUNC);
357 }
358 clk_enable(ac97_clk); 309 clk_enable(ac97_clk);
359 return 0; 310 return 0;
360} 311}
@@ -386,16 +337,9 @@ int __devinit pxa2xx_ac97_hw_probe(struct platform_device *dev)
386 reset_gpio = 113; 337 reset_gpio = 113;
387 } 338 }
388 339
389 if (cpu_is_pxa25x() || cpu_is_pxa27x()) {
390 pxa_gpio_mode(GPIO31_SYNC_AC97_MD);
391 pxa_gpio_mode(GPIO30_SDATA_OUT_AC97_MD);
392 pxa_gpio_mode(GPIO28_BITCLK_AC97_MD);
393 pxa_gpio_mode(GPIO29_SDATA_IN_AC97_MD);
394 }
395
396 if (cpu_is_pxa27x()) { 340 if (cpu_is_pxa27x()) {
397 /* Use GPIO 113 as AC97 Reset on Bulverde */ 341 /* Use GPIO 113 as AC97 Reset on Bulverde */
398 set_resetgpio_mode(RESETGPIO_NORMAL_ALTFUNC); 342 pxa27x_assert_ac97reset(reset_gpio, 0);
399 ac97conf_clk = clk_get(&dev->dev, "AC97CONFCLK"); 343 ac97conf_clk = clk_get(&dev->dev, "AC97CONFCLK");
400 if (IS_ERR(ac97conf_clk)) { 344 if (IS_ERR(ac97conf_clk)) {
401 ret = PTR_ERR(ac97conf_clk); 345 ret = PTR_ERR(ac97conf_clk);
diff --git a/sound/arm/pxa2xx-ac97.c b/sound/arm/pxa2xx-ac97.c
index b4b48afb6de6..5d9411839cd7 100644
--- a/sound/arm/pxa2xx-ac97.c
+++ b/sound/arm/pxa2xx-ac97.c
@@ -159,7 +159,7 @@ static int pxa2xx_ac97_resume(struct device *dev)
159 return ret; 159 return ret;
160} 160}
161 161
162static struct dev_pm_ops pxa2xx_ac97_pm_ops = { 162static const struct dev_pm_ops pxa2xx_ac97_pm_ops = {
163 .suspend = pxa2xx_ac97_suspend, 163 .suspend = pxa2xx_ac97_suspend,
164 .resume = pxa2xx_ac97_resume, 164 .resume = pxa2xx_ac97_resume,
165}; 165};
diff --git a/sound/arm/pxa2xx-pcm-lib.c b/sound/arm/pxa2xx-pcm-lib.c
index 743ac6a29065..8808b82311b1 100644
--- a/sound/arm/pxa2xx-pcm-lib.c
+++ b/sound/arm/pxa2xx-pcm-lib.c
@@ -4,6 +4,7 @@
4 * published by the Free Software Foundation. 4 * published by the Free Software Foundation.
5 */ 5 */
6 6
7#include <linux/slab.h>
7#include <linux/module.h> 8#include <linux/module.h>
8#include <linux/dma-mapping.h> 9#include <linux/dma-mapping.h>
9 10
@@ -205,6 +206,7 @@ int __pxa2xx_pcm_open(struct snd_pcm_substream *substream)
205 if (!rtd->dma_desc_array) 206 if (!rtd->dma_desc_array)
206 goto err1; 207 goto err1;
207 208
209 rtd->dma_ch = -1;
208 runtime->private_data = rtd; 210 runtime->private_data = rtd;
209 return 0; 211 return 0;
210 212