diff options
Diffstat (limited to 'sound/soc/au1x')
-rw-r--r-- | sound/soc/au1x/db1200.c | 40 | ||||
-rw-r--r-- | sound/soc/au1x/dbdma2.c | 95 | ||||
-rw-r--r-- | sound/soc/au1x/psc-ac97.c | 71 | ||||
-rw-r--r-- | sound/soc/au1x/psc-i2s.c | 53 | ||||
-rw-r--r-- | sound/soc/au1x/psc.h | 10 |
5 files changed, 97 insertions, 172 deletions
diff --git a/sound/soc/au1x/db1200.c b/sound/soc/au1x/db1200.c index cdf7be1b9b91..1d3e258c9ea8 100644 --- a/sound/soc/au1x/db1200.c +++ b/sound/soc/au1x/db1200.c | |||
@@ -13,13 +13,11 @@ | |||
13 | #include <sound/core.h> | 13 | #include <sound/core.h> |
14 | #include <sound/pcm.h> | 14 | #include <sound/pcm.h> |
15 | #include <sound/soc.h> | 15 | #include <sound/soc.h> |
16 | #include <sound/soc-dapm.h> | ||
17 | #include <asm/mach-au1x00/au1000.h> | 16 | #include <asm/mach-au1x00/au1000.h> |
18 | #include <asm/mach-au1x00/au1xxx_psc.h> | 17 | #include <asm/mach-au1x00/au1xxx_psc.h> |
19 | #include <asm/mach-au1x00/au1xxx_dbdma.h> | 18 | #include <asm/mach-au1x00/au1xxx_dbdma.h> |
20 | #include <asm/mach-db1x00/bcsr.h> | 19 | #include <asm/mach-db1x00/bcsr.h> |
21 | 20 | ||
22 | #include "../codecs/ac97.h" | ||
23 | #include "../codecs/wm8731.h" | 21 | #include "../codecs/wm8731.h" |
24 | #include "psc.h" | 22 | #include "psc.h" |
25 | 23 | ||
@@ -28,20 +26,16 @@ | |||
28 | static struct snd_soc_dai_link db1200_ac97_dai = { | 26 | static struct snd_soc_dai_link db1200_ac97_dai = { |
29 | .name = "AC97", | 27 | .name = "AC97", |
30 | .stream_name = "AC97 HiFi", | 28 | .stream_name = "AC97 HiFi", |
31 | .cpu_dai = &au1xpsc_ac97_dai, | 29 | .codec_dai_name = "ac97-hifi", |
32 | .codec_dai = &ac97_dai, | 30 | .cpu_dai_name = "au1xpsc_ac97.1", |
31 | .platform_name = "au1xpsc-pcm.1", | ||
32 | .codec_name = "ac97-codec.1", | ||
33 | }; | 33 | }; |
34 | 34 | ||
35 | static struct snd_soc_card db1200_ac97_machine = { | 35 | static struct snd_soc_card db1200_ac97_machine = { |
36 | .name = "DB1200_AC97", | 36 | .name = "DB1200_AC97", |
37 | .dai_link = &db1200_ac97_dai, | 37 | .dai_link = &db1200_ac97_dai, |
38 | .num_links = 1, | 38 | .num_links = 1, |
39 | .platform = &au1xpsc_soc_platform, | ||
40 | }; | ||
41 | |||
42 | static struct snd_soc_device db1200_ac97_devdata = { | ||
43 | .card = &db1200_ac97_machine, | ||
44 | .codec_dev = &soc_codec_dev_ac97, | ||
45 | }; | 39 | }; |
46 | 40 | ||
47 | /*------------------------- I2S PART ---------------------------*/ | 41 | /*------------------------- I2S PART ---------------------------*/ |
@@ -49,12 +43,12 @@ static struct snd_soc_device db1200_ac97_devdata = { | |||
49 | static int db1200_i2s_startup(struct snd_pcm_substream *substream) | 43 | static int db1200_i2s_startup(struct snd_pcm_substream *substream) |
50 | { | 44 | { |
51 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 45 | struct snd_soc_pcm_runtime *rtd = substream->private_data; |
52 | struct snd_soc_dai *codec_dai = rtd->dai->codec_dai; | 46 | struct snd_soc_dai *codec_dai = rtd->codec_dai; |
53 | struct snd_soc_dai *cpu_dai = rtd->dai->cpu_dai; | 47 | struct snd_soc_dai *cpu_dai = rtd->cpu_dai; |
54 | int ret; | 48 | int ret; |
55 | 49 | ||
56 | /* WM8731 has its own 12MHz crystal */ | 50 | /* WM8731 has its own 12MHz crystal */ |
57 | snd_soc_dai_set_sysclk(codec_dai, WM8731_SYSCLK, | 51 | snd_soc_dai_set_sysclk(codec_dai, WM8731_SYSCLK_XTAL, |
58 | 12000000, SND_SOC_CLOCK_IN); | 52 | 12000000, SND_SOC_CLOCK_IN); |
59 | 53 | ||
60 | /* codec is bitclock and lrclk master */ | 54 | /* codec is bitclock and lrclk master */ |
@@ -80,8 +74,10 @@ static struct snd_soc_ops db1200_i2s_wm8731_ops = { | |||
80 | static struct snd_soc_dai_link db1200_i2s_dai = { | 74 | static struct snd_soc_dai_link db1200_i2s_dai = { |
81 | .name = "WM8731", | 75 | .name = "WM8731", |
82 | .stream_name = "WM8731 PCM", | 76 | .stream_name = "WM8731 PCM", |
83 | .cpu_dai = &au1xpsc_i2s_dai, | 77 | .codec_dai_name = "wm8731-hifi", |
84 | .codec_dai = &wm8731_dai, | 78 | .cpu_dai_name = "au1xpsc_i2s.1", |
79 | .platform_name = "au1xpsc-pcm.1", | ||
80 | .codec_name = "wm8731.0-001b", | ||
85 | .ops = &db1200_i2s_wm8731_ops, | 81 | .ops = &db1200_i2s_wm8731_ops, |
86 | }; | 82 | }; |
87 | 83 | ||
@@ -89,12 +85,6 @@ static struct snd_soc_card db1200_i2s_machine = { | |||
89 | .name = "DB1200_I2S", | 85 | .name = "DB1200_I2S", |
90 | .dai_link = &db1200_i2s_dai, | 86 | .dai_link = &db1200_i2s_dai, |
91 | .num_links = 1, | 87 | .num_links = 1, |
92 | .platform = &au1xpsc_soc_platform, | ||
93 | }; | ||
94 | |||
95 | static struct snd_soc_device db1200_i2s_devdata = { | ||
96 | .card = &db1200_i2s_machine, | ||
97 | .codec_dev = &soc_codec_dev_wm8731, | ||
98 | }; | 88 | }; |
99 | 89 | ||
100 | /*------------------------- COMMON PART ---------------------------*/ | 90 | /*------------------------- COMMON PART ---------------------------*/ |
@@ -106,18 +96,16 @@ static int __init db1200_audio_load(void) | |||
106 | int ret; | 96 | int ret; |
107 | 97 | ||
108 | ret = -ENOMEM; | 98 | ret = -ENOMEM; |
109 | db1200_asoc_dev = platform_device_alloc("soc-audio", -1); | 99 | db1200_asoc_dev = platform_device_alloc("soc-audio", 1); /* PSC1 */ |
110 | if (!db1200_asoc_dev) | 100 | if (!db1200_asoc_dev) |
111 | goto out; | 101 | goto out; |
112 | 102 | ||
113 | /* DB1200 board setup set PSC1MUX to preferred audio device */ | 103 | /* DB1200 board setup set PSC1MUX to preferred audio device */ |
114 | if (bcsr_read(BCSR_RESETS) & BCSR_RESETS_PSC1MUX) | 104 | if (bcsr_read(BCSR_RESETS) & BCSR_RESETS_PSC1MUX) |
115 | platform_set_drvdata(db1200_asoc_dev, &db1200_i2s_devdata); | 105 | platform_set_drvdata(db1200_asoc_dev, &db1200_i2s_machine); |
116 | else | 106 | else |
117 | platform_set_drvdata(db1200_asoc_dev, &db1200_ac97_devdata); | 107 | platform_set_drvdata(db1200_asoc_dev, &db1200_ac97_machine); |
118 | 108 | ||
119 | db1200_ac97_devdata.dev = &db1200_asoc_dev->dev; | ||
120 | db1200_i2s_devdata.dev = &db1200_asoc_dev->dev; | ||
121 | ret = platform_device_add(db1200_asoc_dev); | 109 | ret = platform_device_add(db1200_asoc_dev); |
122 | 110 | ||
123 | if (ret) { | 111 | if (ret) { |
diff --git a/sound/soc/au1x/dbdma2.c b/sound/soc/au1x/dbdma2.c index 6d9f4c624949..10fdd2854e58 100644 --- a/sound/soc/au1x/dbdma2.c +++ b/sound/soc/au1x/dbdma2.c | |||
@@ -10,9 +10,6 @@ | |||
10 | * | 10 | * |
11 | * DMA glue for Au1x-PSC audio. | 11 | * DMA glue for Au1x-PSC audio. |
12 | * | 12 | * |
13 | * NOTE: all of these drivers can only work with a SINGLE instance | ||
14 | * of a PSC. Multiple independent audio devices are impossible | ||
15 | * with ASoC v1. | ||
16 | */ | 13 | */ |
17 | 14 | ||
18 | 15 | ||
@@ -61,9 +58,6 @@ struct au1xpsc_audio_dmadata { | |||
61 | int msbits; | 58 | int msbits; |
62 | }; | 59 | }; |
63 | 60 | ||
64 | /* instance data. There can be only one, MacLeod!!!! */ | ||
65 | static struct au1xpsc_audio_dmadata *au1xpsc_audio_pcmdma[2]; | ||
66 | |||
67 | /* | 61 | /* |
68 | * These settings are somewhat okay, at least on my machine audio plays | 62 | * These settings are somewhat okay, at least on my machine audio plays |
69 | * almost skip-free. Especially the 64kB buffer seems to help a LOT. | 63 | * almost skip-free. Especially the 64kB buffer seems to help a LOT. |
@@ -199,6 +193,14 @@ out: | |||
199 | return 0; | 193 | return 0; |
200 | } | 194 | } |
201 | 195 | ||
196 | static inline struct au1xpsc_audio_dmadata *to_dmadata(struct snd_pcm_substream *ss) | ||
197 | { | ||
198 | struct snd_soc_pcm_runtime *rtd = ss->private_data; | ||
199 | struct au1xpsc_audio_dmadata *pcd = | ||
200 | snd_soc_platform_get_drvdata(rtd->platform); | ||
201 | return &pcd[SUBSTREAM_TYPE(ss)]; | ||
202 | } | ||
203 | |||
202 | static int au1xpsc_pcm_hw_params(struct snd_pcm_substream *substream, | 204 | static int au1xpsc_pcm_hw_params(struct snd_pcm_substream *substream, |
203 | struct snd_pcm_hw_params *params) | 205 | struct snd_pcm_hw_params *params) |
204 | { | 206 | { |
@@ -211,7 +213,7 @@ static int au1xpsc_pcm_hw_params(struct snd_pcm_substream *substream, | |||
211 | goto out; | 213 | goto out; |
212 | 214 | ||
213 | stype = SUBSTREAM_TYPE(substream); | 215 | stype = SUBSTREAM_TYPE(substream); |
214 | pcd = au1xpsc_audio_pcmdma[stype]; | 216 | pcd = to_dmadata(substream); |
215 | 217 | ||
216 | DBG("runtime->dma_area = 0x%08lx dma_addr_t = 0x%08lx dma_size = %d " | 218 | DBG("runtime->dma_area = 0x%08lx dma_addr_t = 0x%08lx dma_size = %d " |
217 | "runtime->min_align %d\n", | 219 | "runtime->min_align %d\n", |
@@ -249,8 +251,7 @@ static int au1xpsc_pcm_hw_free(struct snd_pcm_substream *substream) | |||
249 | 251 | ||
250 | static int au1xpsc_pcm_prepare(struct snd_pcm_substream *substream) | 252 | static int au1xpsc_pcm_prepare(struct snd_pcm_substream *substream) |
251 | { | 253 | { |
252 | struct au1xpsc_audio_dmadata *pcd = | 254 | struct au1xpsc_audio_dmadata *pcd = to_dmadata(substream); |
253 | au1xpsc_audio_pcmdma[SUBSTREAM_TYPE(substream)]; | ||
254 | 255 | ||
255 | au1xxx_dbdma_reset(pcd->ddma_chan); | 256 | au1xxx_dbdma_reset(pcd->ddma_chan); |
256 | 257 | ||
@@ -267,7 +268,7 @@ static int au1xpsc_pcm_prepare(struct snd_pcm_substream *substream) | |||
267 | 268 | ||
268 | static int au1xpsc_pcm_trigger(struct snd_pcm_substream *substream, int cmd) | 269 | static int au1xpsc_pcm_trigger(struct snd_pcm_substream *substream, int cmd) |
269 | { | 270 | { |
270 | u32 c = au1xpsc_audio_pcmdma[SUBSTREAM_TYPE(substream)]->ddma_chan; | 271 | u32 c = to_dmadata(substream)->ddma_chan; |
271 | 272 | ||
272 | switch (cmd) { | 273 | switch (cmd) { |
273 | case SNDRV_PCM_TRIGGER_START: | 274 | case SNDRV_PCM_TRIGGER_START: |
@@ -287,8 +288,7 @@ static int au1xpsc_pcm_trigger(struct snd_pcm_substream *substream, int cmd) | |||
287 | static snd_pcm_uframes_t | 288 | static snd_pcm_uframes_t |
288 | au1xpsc_pcm_pointer(struct snd_pcm_substream *substream) | 289 | au1xpsc_pcm_pointer(struct snd_pcm_substream *substream) |
289 | { | 290 | { |
290 | return bytes_to_frames(substream->runtime, | 291 | return bytes_to_frames(substream->runtime, to_dmadata(substream)->pos); |
291 | au1xpsc_audio_pcmdma[SUBSTREAM_TYPE(substream)]->pos); | ||
292 | } | 292 | } |
293 | 293 | ||
294 | static int au1xpsc_pcm_open(struct snd_pcm_substream *substream) | 294 | static int au1xpsc_pcm_open(struct snd_pcm_substream *substream) |
@@ -299,7 +299,7 @@ static int au1xpsc_pcm_open(struct snd_pcm_substream *substream) | |||
299 | 299 | ||
300 | static int au1xpsc_pcm_close(struct snd_pcm_substream *substream) | 300 | static int au1xpsc_pcm_close(struct snd_pcm_substream *substream) |
301 | { | 301 | { |
302 | au1x_pcm_dbdma_free(au1xpsc_audio_pcmdma[SUBSTREAM_TYPE(substream)]); | 302 | au1x_pcm_dbdma_free(to_dmadata(substream)); |
303 | return 0; | 303 | return 0; |
304 | } | 304 | } |
305 | 305 | ||
@@ -329,42 +329,21 @@ static int au1xpsc_pcm_new(struct snd_card *card, | |||
329 | return 0; | 329 | return 0; |
330 | } | 330 | } |
331 | 331 | ||
332 | static int au1xpsc_pcm_probe(struct platform_device *pdev) | ||
333 | { | ||
334 | if (!au1xpsc_audio_pcmdma[PCM_TX] || !au1xpsc_audio_pcmdma[PCM_RX]) | ||
335 | return -ENODEV; | ||
336 | |||
337 | return 0; | ||
338 | } | ||
339 | |||
340 | static int au1xpsc_pcm_remove(struct platform_device *pdev) | ||
341 | { | ||
342 | return 0; | ||
343 | } | ||
344 | |||
345 | /* au1xpsc audio platform */ | 332 | /* au1xpsc audio platform */ |
346 | struct snd_soc_platform au1xpsc_soc_platform = { | 333 | struct snd_soc_platform_driver au1xpsc_soc_platform = { |
347 | .name = "au1xpsc-pcm-dbdma", | 334 | .ops = &au1xpsc_pcm_ops, |
348 | .probe = au1xpsc_pcm_probe, | ||
349 | .remove = au1xpsc_pcm_remove, | ||
350 | .pcm_ops = &au1xpsc_pcm_ops, | ||
351 | .pcm_new = au1xpsc_pcm_new, | 335 | .pcm_new = au1xpsc_pcm_new, |
352 | .pcm_free = au1xpsc_pcm_free_dma_buffers, | 336 | .pcm_free = au1xpsc_pcm_free_dma_buffers, |
353 | }; | 337 | }; |
354 | EXPORT_SYMBOL_GPL(au1xpsc_soc_platform); | ||
355 | 338 | ||
356 | static int __devinit au1xpsc_pcm_drvprobe(struct platform_device *pdev) | 339 | static int __devinit au1xpsc_pcm_drvprobe(struct platform_device *pdev) |
357 | { | 340 | { |
341 | struct au1xpsc_audio_dmadata *dmadata; | ||
358 | struct resource *r; | 342 | struct resource *r; |
359 | int ret; | 343 | int ret; |
360 | 344 | ||
361 | if (au1xpsc_audio_pcmdma[PCM_TX] || au1xpsc_audio_pcmdma[PCM_RX]) | 345 | dmadata = kzalloc(2 * sizeof(struct au1xpsc_audio_dmadata), GFP_KERNEL); |
362 | return -EBUSY; | 346 | if (!dmadata) |
363 | |||
364 | /* TX DMA */ | ||
365 | au1xpsc_audio_pcmdma[PCM_TX] | ||
366 | = kzalloc(sizeof(struct au1xpsc_audio_dmadata), GFP_KERNEL); | ||
367 | if (!au1xpsc_audio_pcmdma[PCM_TX]) | ||
368 | return -ENOMEM; | 347 | return -ENOMEM; |
369 | 348 | ||
370 | r = platform_get_resource(pdev, IORESOURCE_DMA, 0); | 349 | r = platform_get_resource(pdev, IORESOURCE_DMA, 0); |
@@ -372,47 +351,33 @@ static int __devinit au1xpsc_pcm_drvprobe(struct platform_device *pdev) | |||
372 | ret = -ENODEV; | 351 | ret = -ENODEV; |
373 | goto out1; | 352 | goto out1; |
374 | } | 353 | } |
375 | (au1xpsc_audio_pcmdma[PCM_TX])->ddma_id = r->start; | 354 | dmadata[PCM_TX].ddma_id = r->start; |
376 | 355 | ||
377 | /* RX DMA */ | 356 | /* RX DMA */ |
378 | au1xpsc_audio_pcmdma[PCM_RX] | ||
379 | = kzalloc(sizeof(struct au1xpsc_audio_dmadata), GFP_KERNEL); | ||
380 | if (!au1xpsc_audio_pcmdma[PCM_RX]) | ||
381 | return -ENOMEM; | ||
382 | |||
383 | r = platform_get_resource(pdev, IORESOURCE_DMA, 1); | 357 | r = platform_get_resource(pdev, IORESOURCE_DMA, 1); |
384 | if (!r) { | 358 | if (!r) { |
385 | ret = -ENODEV; | 359 | ret = -ENODEV; |
386 | goto out2; | 360 | goto out1; |
387 | } | 361 | } |
388 | (au1xpsc_audio_pcmdma[PCM_RX])->ddma_id = r->start; | 362 | dmadata[PCM_RX].ddma_id = r->start; |
363 | |||
364 | platform_set_drvdata(pdev, dmadata); | ||
389 | 365 | ||
390 | ret = snd_soc_register_platform(&au1xpsc_soc_platform); | 366 | ret = snd_soc_register_platform(&pdev->dev, &au1xpsc_soc_platform); |
391 | if (!ret) | 367 | if (!ret) |
392 | return ret; | 368 | return ret; |
393 | 369 | ||
394 | out2: | ||
395 | kfree(au1xpsc_audio_pcmdma[PCM_RX]); | ||
396 | au1xpsc_audio_pcmdma[PCM_RX] = NULL; | ||
397 | out1: | 370 | out1: |
398 | kfree(au1xpsc_audio_pcmdma[PCM_TX]); | 371 | kfree(dmadata); |
399 | au1xpsc_audio_pcmdma[PCM_TX] = NULL; | ||
400 | return ret; | 372 | return ret; |
401 | } | 373 | } |
402 | 374 | ||
403 | static int __devexit au1xpsc_pcm_drvremove(struct platform_device *pdev) | 375 | static int __devexit au1xpsc_pcm_drvremove(struct platform_device *pdev) |
404 | { | 376 | { |
405 | int i; | 377 | struct au1xpsc_audio_dmadata *dmadata = platform_get_drvdata(pdev); |
406 | 378 | ||
407 | snd_soc_unregister_platform(&au1xpsc_soc_platform); | 379 | snd_soc_unregister_platform(&pdev->dev); |
408 | 380 | kfree(dmadata); | |
409 | for (i = 0; i < 2; i++) { | ||
410 | if (au1xpsc_audio_pcmdma[i]) { | ||
411 | au1x_pcm_dbdma_free(au1xpsc_audio_pcmdma[i]); | ||
412 | kfree(au1xpsc_audio_pcmdma[i]); | ||
413 | au1xpsc_audio_pcmdma[i] = NULL; | ||
414 | } | ||
415 | } | ||
416 | 381 | ||
417 | return 0; | 382 | return 0; |
418 | } | 383 | } |
@@ -428,8 +393,6 @@ static struct platform_driver au1xpsc_pcm_driver = { | |||
428 | 393 | ||
429 | static int __init au1xpsc_audio_dbdma_load(void) | 394 | static int __init au1xpsc_audio_dbdma_load(void) |
430 | { | 395 | { |
431 | au1xpsc_audio_pcmdma[PCM_TX] = NULL; | ||
432 | au1xpsc_audio_pcmdma[PCM_RX] = NULL; | ||
433 | return platform_driver_register(&au1xpsc_pcm_driver); | 396 | return platform_driver_register(&au1xpsc_pcm_driver); |
434 | } | 397 | } |
435 | 398 | ||
@@ -467,7 +430,7 @@ struct platform_device *au1xpsc_pcm_add(struct platform_device *pdev) | |||
467 | res[1].start = res[1].end = id[1]; | 430 | res[1].start = res[1].end = id[1]; |
468 | res[0].flags = res[1].flags = IORESOURCE_DMA; | 431 | res[0].flags = res[1].flags = IORESOURCE_DMA; |
469 | 432 | ||
470 | pd = platform_device_alloc("au1xpsc-pcm", -1); | 433 | pd = platform_device_alloc("au1xpsc-pcm", pdev->id); |
471 | if (!pd) | 434 | if (!pd) |
472 | goto out; | 435 | goto out; |
473 | 436 | ||
diff --git a/sound/soc/au1x/psc-ac97.c b/sound/soc/au1x/psc-ac97.c index d14a5a91a465..d0db66f24a00 100644 --- a/sound/soc/au1x/psc-ac97.c +++ b/sound/soc/au1x/psc-ac97.c | |||
@@ -10,9 +10,6 @@ | |||
10 | * | 10 | * |
11 | * Au1xxx-PSC AC97 glue. | 11 | * Au1xxx-PSC AC97 glue. |
12 | * | 12 | * |
13 | * NOTE: all of these drivers can only work with a SINGLE instance | ||
14 | * of a PSC. Multiple independent audio devices are impossible | ||
15 | * with ASoC v1. | ||
16 | */ | 13 | */ |
17 | 14 | ||
18 | #include <linux/init.h> | 15 | #include <linux/init.h> |
@@ -56,12 +53,29 @@ | |||
56 | /* instance data. There can be only one, MacLeod!!!! */ | 53 | /* instance data. There can be only one, MacLeod!!!! */ |
57 | static struct au1xpsc_audio_data *au1xpsc_ac97_workdata; | 54 | static struct au1xpsc_audio_data *au1xpsc_ac97_workdata; |
58 | 55 | ||
56 | #if 0 | ||
57 | |||
58 | /* this could theoretically work, but ac97->bus->card->private_data can be NULL | ||
59 | * when snd_ac97_mixer() is called; I don't know if the rest further down the | ||
60 | * chain are always valid either. | ||
61 | */ | ||
62 | static inline struct au1xpsc_audio_data *ac97_to_pscdata(struct snd_ac97 *x) | ||
63 | { | ||
64 | struct snd_soc_card *c = x->bus->card->private_data; | ||
65 | return snd_soc_dai_get_drvdata(c->rtd->cpu_dai); | ||
66 | } | ||
67 | |||
68 | #else | ||
69 | |||
70 | #define ac97_to_pscdata(x) au1xpsc_ac97_workdata | ||
71 | |||
72 | #endif | ||
73 | |||
59 | /* AC97 controller reads codec register */ | 74 | /* AC97 controller reads codec register */ |
60 | static unsigned short au1xpsc_ac97_read(struct snd_ac97 *ac97, | 75 | static unsigned short au1xpsc_ac97_read(struct snd_ac97 *ac97, |
61 | unsigned short reg) | 76 | unsigned short reg) |
62 | { | 77 | { |
63 | /* FIXME */ | 78 | struct au1xpsc_audio_data *pscdata = ac97_to_pscdata(ac97); |
64 | struct au1xpsc_audio_data *pscdata = au1xpsc_ac97_workdata; | ||
65 | unsigned short retry, tmo; | 79 | unsigned short retry, tmo; |
66 | unsigned long data; | 80 | unsigned long data; |
67 | 81 | ||
@@ -102,8 +116,7 @@ static unsigned short au1xpsc_ac97_read(struct snd_ac97 *ac97, | |||
102 | static void au1xpsc_ac97_write(struct snd_ac97 *ac97, unsigned short reg, | 116 | static void au1xpsc_ac97_write(struct snd_ac97 *ac97, unsigned short reg, |
103 | unsigned short val) | 117 | unsigned short val) |
104 | { | 118 | { |
105 | /* FIXME */ | 119 | struct au1xpsc_audio_data *pscdata = ac97_to_pscdata(ac97); |
106 | struct au1xpsc_audio_data *pscdata = au1xpsc_ac97_workdata; | ||
107 | unsigned int tmo, retry; | 120 | unsigned int tmo, retry; |
108 | 121 | ||
109 | au_writel(PSC_AC97EVNT_CD, AC97_EVNT(pscdata)); | 122 | au_writel(PSC_AC97EVNT_CD, AC97_EVNT(pscdata)); |
@@ -134,8 +147,7 @@ static void au1xpsc_ac97_write(struct snd_ac97 *ac97, unsigned short reg, | |||
134 | /* AC97 controller asserts a warm reset */ | 147 | /* AC97 controller asserts a warm reset */ |
135 | static void au1xpsc_ac97_warm_reset(struct snd_ac97 *ac97) | 148 | static void au1xpsc_ac97_warm_reset(struct snd_ac97 *ac97) |
136 | { | 149 | { |
137 | /* FIXME */ | 150 | struct au1xpsc_audio_data *pscdata = ac97_to_pscdata(ac97); |
138 | struct au1xpsc_audio_data *pscdata = au1xpsc_ac97_workdata; | ||
139 | 151 | ||
140 | au_writel(PSC_AC97RST_SNC, AC97_RST(pscdata)); | 152 | au_writel(PSC_AC97RST_SNC, AC97_RST(pscdata)); |
141 | au_sync(); | 153 | au_sync(); |
@@ -146,8 +158,7 @@ static void au1xpsc_ac97_warm_reset(struct snd_ac97 *ac97) | |||
146 | 158 | ||
147 | static void au1xpsc_ac97_cold_reset(struct snd_ac97 *ac97) | 159 | static void au1xpsc_ac97_cold_reset(struct snd_ac97 *ac97) |
148 | { | 160 | { |
149 | /* FIXME */ | 161 | struct au1xpsc_audio_data *pscdata = ac97_to_pscdata(ac97); |
150 | struct au1xpsc_audio_data *pscdata = au1xpsc_ac97_workdata; | ||
151 | int i; | 162 | int i; |
152 | 163 | ||
153 | /* disable PSC during cold reset */ | 164 | /* disable PSC during cold reset */ |
@@ -202,8 +213,7 @@ static int au1xpsc_ac97_hw_params(struct snd_pcm_substream *substream, | |||
202 | struct snd_pcm_hw_params *params, | 213 | struct snd_pcm_hw_params *params, |
203 | struct snd_soc_dai *dai) | 214 | struct snd_soc_dai *dai) |
204 | { | 215 | { |
205 | /* FIXME */ | 216 | struct au1xpsc_audio_data *pscdata = snd_soc_dai_get_drvdata(dai); |
206 | struct au1xpsc_audio_data *pscdata = au1xpsc_ac97_workdata; | ||
207 | unsigned long r, ro, stat; | 217 | unsigned long r, ro, stat; |
208 | int chans, t, stype = SUBSTREAM_TYPE(substream); | 218 | int chans, t, stype = SUBSTREAM_TYPE(substream); |
209 | 219 | ||
@@ -283,8 +293,7 @@ out: | |||
283 | static int au1xpsc_ac97_trigger(struct snd_pcm_substream *substream, | 293 | static int au1xpsc_ac97_trigger(struct snd_pcm_substream *substream, |
284 | int cmd, struct snd_soc_dai *dai) | 294 | int cmd, struct snd_soc_dai *dai) |
285 | { | 295 | { |
286 | /* FIXME */ | 296 | struct au1xpsc_audio_data *pscdata = snd_soc_dai_get_drvdata(dai); |
287 | struct au1xpsc_audio_data *pscdata = au1xpsc_ac97_workdata; | ||
288 | int ret, stype = SUBSTREAM_TYPE(substream); | 297 | int ret, stype = SUBSTREAM_TYPE(substream); |
289 | 298 | ||
290 | ret = 0; | 299 | ret = 0; |
@@ -315,27 +324,19 @@ static int au1xpsc_ac97_trigger(struct snd_pcm_substream *substream, | |||
315 | return ret; | 324 | return ret; |
316 | } | 325 | } |
317 | 326 | ||
318 | static int au1xpsc_ac97_probe(struct platform_device *pdev, | 327 | static int au1xpsc_ac97_probe(struct snd_soc_dai *dai) |
319 | struct snd_soc_dai *dai) | ||
320 | { | 328 | { |
321 | return au1xpsc_ac97_workdata ? 0 : -ENODEV; | 329 | return au1xpsc_ac97_workdata ? 0 : -ENODEV; |
322 | } | 330 | } |
323 | 331 | ||
324 | static void au1xpsc_ac97_remove(struct platform_device *pdev, | ||
325 | struct snd_soc_dai *dai) | ||
326 | { | ||
327 | } | ||
328 | |||
329 | static struct snd_soc_dai_ops au1xpsc_ac97_dai_ops = { | 332 | static struct snd_soc_dai_ops au1xpsc_ac97_dai_ops = { |
330 | .trigger = au1xpsc_ac97_trigger, | 333 | .trigger = au1xpsc_ac97_trigger, |
331 | .hw_params = au1xpsc_ac97_hw_params, | 334 | .hw_params = au1xpsc_ac97_hw_params, |
332 | }; | 335 | }; |
333 | 336 | ||
334 | struct snd_soc_dai au1xpsc_ac97_dai = { | 337 | static const struct snd_soc_dai_driver au1xpsc_ac97_dai_template = { |
335 | .name = "au1xpsc_ac97", | ||
336 | .ac97_control = 1, | 338 | .ac97_control = 1, |
337 | .probe = au1xpsc_ac97_probe, | 339 | .probe = au1xpsc_ac97_probe, |
338 | .remove = au1xpsc_ac97_remove, | ||
339 | .playback = { | 340 | .playback = { |
340 | .rates = AC97_RATES, | 341 | .rates = AC97_RATES, |
341 | .formats = AC97_FMTS, | 342 | .formats = AC97_FMTS, |
@@ -350,7 +351,6 @@ struct snd_soc_dai au1xpsc_ac97_dai = { | |||
350 | }, | 351 | }, |
351 | .ops = &au1xpsc_ac97_dai_ops, | 352 | .ops = &au1xpsc_ac97_dai_ops, |
352 | }; | 353 | }; |
353 | EXPORT_SYMBOL_GPL(au1xpsc_ac97_dai); | ||
354 | 354 | ||
355 | static int __devinit au1xpsc_ac97_drvprobe(struct platform_device *pdev) | 355 | static int __devinit au1xpsc_ac97_drvprobe(struct platform_device *pdev) |
356 | { | 356 | { |
@@ -359,9 +359,6 @@ static int __devinit au1xpsc_ac97_drvprobe(struct platform_device *pdev) | |||
359 | unsigned long sel; | 359 | unsigned long sel; |
360 | struct au1xpsc_audio_data *wd; | 360 | struct au1xpsc_audio_data *wd; |
361 | 361 | ||
362 | if (au1xpsc_ac97_workdata) | ||
363 | return -EBUSY; | ||
364 | |||
365 | wd = kzalloc(sizeof(struct au1xpsc_audio_data), GFP_KERNEL); | 362 | wd = kzalloc(sizeof(struct au1xpsc_audio_data), GFP_KERNEL); |
366 | if (!wd) | 363 | if (!wd) |
367 | return -ENOMEM; | 364 | return -ENOMEM; |
@@ -395,18 +392,24 @@ static int __devinit au1xpsc_ac97_drvprobe(struct platform_device *pdev) | |||
395 | au_writel(PSC_SEL_PS_AC97MODE | sel, PSC_SEL(wd)); | 392 | au_writel(PSC_SEL_PS_AC97MODE | sel, PSC_SEL(wd)); |
396 | au_sync(); | 393 | au_sync(); |
397 | 394 | ||
398 | ret = snd_soc_register_dai(&au1xpsc_ac97_dai); | 395 | /* name the DAI like this device instance ("au1xpsc-ac97.PSCINDEX") */ |
396 | memcpy(&wd->dai_drv, &au1xpsc_ac97_dai_template, | ||
397 | sizeof(struct snd_soc_dai_driver)); | ||
398 | wd->dai_drv.name = dev_name(&pdev->dev); | ||
399 | |||
400 | platform_set_drvdata(pdev, wd); | ||
401 | |||
402 | ret = snd_soc_register_dai(&pdev->dev, &wd->dai_drv); | ||
399 | if (ret) | 403 | if (ret) |
400 | goto out1; | 404 | goto out1; |
401 | 405 | ||
402 | wd->dmapd = au1xpsc_pcm_add(pdev); | 406 | wd->dmapd = au1xpsc_pcm_add(pdev); |
403 | if (wd->dmapd) { | 407 | if (wd->dmapd) { |
404 | platform_set_drvdata(pdev, wd); | 408 | au1xpsc_ac97_workdata = wd; |
405 | au1xpsc_ac97_workdata = wd; /* MDEV */ | ||
406 | return 0; | 409 | return 0; |
407 | } | 410 | } |
408 | 411 | ||
409 | snd_soc_unregister_dai(&au1xpsc_ac97_dai); | 412 | snd_soc_unregister_dai(&pdev->dev); |
410 | out1: | 413 | out1: |
411 | release_mem_region(r->start, resource_size(r)); | 414 | release_mem_region(r->start, resource_size(r)); |
412 | out0: | 415 | out0: |
@@ -422,7 +425,7 @@ static int __devexit au1xpsc_ac97_drvremove(struct platform_device *pdev) | |||
422 | if (wd->dmapd) | 425 | if (wd->dmapd) |
423 | au1xpsc_pcm_destroy(wd->dmapd); | 426 | au1xpsc_pcm_destroy(wd->dmapd); |
424 | 427 | ||
425 | snd_soc_unregister_dai(&au1xpsc_ac97_dai); | 428 | snd_soc_unregister_dai(&pdev->dev); |
426 | 429 | ||
427 | /* disable PSC completely */ | 430 | /* disable PSC completely */ |
428 | au_writel(0, AC97_CFG(wd)); | 431 | au_writel(0, AC97_CFG(wd)); |
diff --git a/sound/soc/au1x/psc-i2s.c b/sound/soc/au1x/psc-i2s.c index 6083fe7799fa..fca091276320 100644 --- a/sound/soc/au1x/psc-i2s.c +++ b/sound/soc/au1x/psc-i2s.c | |||
@@ -10,9 +10,6 @@ | |||
10 | * | 10 | * |
11 | * Au1xxx-PSC I2S glue. | 11 | * Au1xxx-PSC I2S glue. |
12 | * | 12 | * |
13 | * NOTE: all of these drivers can only work with a SINGLE instance | ||
14 | * of a PSC. Multiple independent audio devices are impossible | ||
15 | * with ASoC v1. | ||
16 | * NOTE: so far only PSC slave mode (bit- and frameclock) is supported. | 13 | * NOTE: so far only PSC slave mode (bit- and frameclock) is supported. |
17 | */ | 14 | */ |
18 | 15 | ||
@@ -54,13 +51,10 @@ | |||
54 | ((stype) == PCM_TX ? PSC_I2SPCR_TC : PSC_I2SPCR_RC) | 51 | ((stype) == PCM_TX ? PSC_I2SPCR_TC : PSC_I2SPCR_RC) |
55 | 52 | ||
56 | 53 | ||
57 | /* instance data. There can be only one, MacLeod!!!! */ | ||
58 | static struct au1xpsc_audio_data *au1xpsc_i2s_workdata; | ||
59 | |||
60 | static int au1xpsc_i2s_set_fmt(struct snd_soc_dai *cpu_dai, | 54 | static int au1xpsc_i2s_set_fmt(struct snd_soc_dai *cpu_dai, |
61 | unsigned int fmt) | 55 | unsigned int fmt) |
62 | { | 56 | { |
63 | struct au1xpsc_audio_data *pscdata = au1xpsc_i2s_workdata; | 57 | struct au1xpsc_audio_data *pscdata = snd_soc_dai_get_drvdata(cpu_dai); |
64 | unsigned long ct; | 58 | unsigned long ct; |
65 | int ret; | 59 | int ret; |
66 | 60 | ||
@@ -120,7 +114,7 @@ static int au1xpsc_i2s_hw_params(struct snd_pcm_substream *substream, | |||
120 | struct snd_pcm_hw_params *params, | 114 | struct snd_pcm_hw_params *params, |
121 | struct snd_soc_dai *dai) | 115 | struct snd_soc_dai *dai) |
122 | { | 116 | { |
123 | struct au1xpsc_audio_data *pscdata = au1xpsc_i2s_workdata; | 117 | struct au1xpsc_audio_data *pscdata = snd_soc_dai_get_drvdata(dai); |
124 | 118 | ||
125 | int cfgbits; | 119 | int cfgbits; |
126 | unsigned long stat; | 120 | unsigned long stat; |
@@ -245,7 +239,7 @@ static int au1xpsc_i2s_stop(struct au1xpsc_audio_data *pscdata, int stype) | |||
245 | static int au1xpsc_i2s_trigger(struct snd_pcm_substream *substream, int cmd, | 239 | static int au1xpsc_i2s_trigger(struct snd_pcm_substream *substream, int cmd, |
246 | struct snd_soc_dai *dai) | 240 | struct snd_soc_dai *dai) |
247 | { | 241 | { |
248 | struct au1xpsc_audio_data *pscdata = au1xpsc_i2s_workdata; | 242 | struct au1xpsc_audio_data *pscdata = snd_soc_dai_get_drvdata(dai); |
249 | int ret, stype = SUBSTREAM_TYPE(substream); | 243 | int ret, stype = SUBSTREAM_TYPE(substream); |
250 | 244 | ||
251 | switch (cmd) { | 245 | switch (cmd) { |
@@ -263,27 +257,13 @@ static int au1xpsc_i2s_trigger(struct snd_pcm_substream *substream, int cmd, | |||
263 | return ret; | 257 | return ret; |
264 | } | 258 | } |
265 | 259 | ||
266 | static int au1xpsc_i2s_probe(struct platform_device *pdev, | ||
267 | struct snd_soc_dai *dai) | ||
268 | { | ||
269 | return au1xpsc_i2s_workdata ? 0 : -ENODEV; | ||
270 | } | ||
271 | |||
272 | static void au1xpsc_i2s_remove(struct platform_device *pdev, | ||
273 | struct snd_soc_dai *dai) | ||
274 | { | ||
275 | } | ||
276 | |||
277 | static struct snd_soc_dai_ops au1xpsc_i2s_dai_ops = { | 260 | static struct snd_soc_dai_ops au1xpsc_i2s_dai_ops = { |
278 | .trigger = au1xpsc_i2s_trigger, | 261 | .trigger = au1xpsc_i2s_trigger, |
279 | .hw_params = au1xpsc_i2s_hw_params, | 262 | .hw_params = au1xpsc_i2s_hw_params, |
280 | .set_fmt = au1xpsc_i2s_set_fmt, | 263 | .set_fmt = au1xpsc_i2s_set_fmt, |
281 | }; | 264 | }; |
282 | 265 | ||
283 | struct snd_soc_dai au1xpsc_i2s_dai = { | 266 | static const struct snd_soc_dai_driver au1xpsc_i2s_dai_template = { |
284 | .name = "au1xpsc_i2s", | ||
285 | .probe = au1xpsc_i2s_probe, | ||
286 | .remove = au1xpsc_i2s_remove, | ||
287 | .playback = { | 267 | .playback = { |
288 | .rates = AU1XPSC_I2S_RATES, | 268 | .rates = AU1XPSC_I2S_RATES, |
289 | .formats = AU1XPSC_I2S_FMTS, | 269 | .formats = AU1XPSC_I2S_FMTS, |
@@ -298,7 +278,6 @@ struct snd_soc_dai au1xpsc_i2s_dai = { | |||
298 | }, | 278 | }, |
299 | .ops = &au1xpsc_i2s_dai_ops, | 279 | .ops = &au1xpsc_i2s_dai_ops, |
300 | }; | 280 | }; |
301 | EXPORT_SYMBOL(au1xpsc_i2s_dai); | ||
302 | 281 | ||
303 | static int __devinit au1xpsc_i2s_drvprobe(struct platform_device *pdev) | 282 | static int __devinit au1xpsc_i2s_drvprobe(struct platform_device *pdev) |
304 | { | 283 | { |
@@ -307,9 +286,6 @@ static int __devinit au1xpsc_i2s_drvprobe(struct platform_device *pdev) | |||
307 | int ret; | 286 | int ret; |
308 | struct au1xpsc_audio_data *wd; | 287 | struct au1xpsc_audio_data *wd; |
309 | 288 | ||
310 | if (au1xpsc_i2s_workdata) | ||
311 | return -EBUSY; | ||
312 | |||
313 | wd = kzalloc(sizeof(struct au1xpsc_audio_data), GFP_KERNEL); | 289 | wd = kzalloc(sizeof(struct au1xpsc_audio_data), GFP_KERNEL); |
314 | if (!wd) | 290 | if (!wd) |
315 | return -ENOMEM; | 291 | return -ENOMEM; |
@@ -346,19 +322,23 @@ static int __devinit au1xpsc_i2s_drvprobe(struct platform_device *pdev) | |||
346 | * time out. | 322 | * time out. |
347 | */ | 323 | */ |
348 | 324 | ||
349 | ret = snd_soc_register_dai(&au1xpsc_i2s_dai); | 325 | /* name the DAI like this device instance ("au1xpsc-i2s.PSCINDEX") */ |
326 | memcpy(&wd->dai_drv, &au1xpsc_i2s_dai_template, | ||
327 | sizeof(struct snd_soc_dai_driver)); | ||
328 | wd->dai_drv.name = dev_name(&pdev->dev); | ||
329 | |||
330 | platform_set_drvdata(pdev, wd); | ||
331 | |||
332 | ret = snd_soc_register_dai(&pdev->dev, &wd->dai_drv); | ||
350 | if (ret) | 333 | if (ret) |
351 | goto out1; | 334 | goto out1; |
352 | 335 | ||
353 | /* finally add the DMA device for this PSC */ | 336 | /* finally add the DMA device for this PSC */ |
354 | wd->dmapd = au1xpsc_pcm_add(pdev); | 337 | wd->dmapd = au1xpsc_pcm_add(pdev); |
355 | if (wd->dmapd) { | 338 | if (wd->dmapd) |
356 | platform_set_drvdata(pdev, wd); | ||
357 | au1xpsc_i2s_workdata = wd; | ||
358 | return 0; | 339 | return 0; |
359 | } | ||
360 | 340 | ||
361 | snd_soc_unregister_dai(&au1xpsc_i2s_dai); | 341 | snd_soc_unregister_dai(&pdev->dev); |
362 | out1: | 342 | out1: |
363 | release_mem_region(r->start, resource_size(r)); | 343 | release_mem_region(r->start, resource_size(r)); |
364 | out0: | 344 | out0: |
@@ -374,7 +354,7 @@ static int __devexit au1xpsc_i2s_drvremove(struct platform_device *pdev) | |||
374 | if (wd->dmapd) | 354 | if (wd->dmapd) |
375 | au1xpsc_pcm_destroy(wd->dmapd); | 355 | au1xpsc_pcm_destroy(wd->dmapd); |
376 | 356 | ||
377 | snd_soc_unregister_dai(&au1xpsc_i2s_dai); | 357 | snd_soc_unregister_dai(&pdev->dev); |
378 | 358 | ||
379 | au_writel(0, I2S_CFG(wd)); | 359 | au_writel(0, I2S_CFG(wd)); |
380 | au_sync(); | 360 | au_sync(); |
@@ -385,8 +365,6 @@ static int __devexit au1xpsc_i2s_drvremove(struct platform_device *pdev) | |||
385 | release_mem_region(r->start, resource_size(r)); | 365 | release_mem_region(r->start, resource_size(r)); |
386 | kfree(wd); | 366 | kfree(wd); |
387 | 367 | ||
388 | au1xpsc_i2s_workdata = NULL; /* MDEV */ | ||
389 | |||
390 | return 0; | 368 | return 0; |
391 | } | 369 | } |
392 | 370 | ||
@@ -446,7 +424,6 @@ static struct platform_driver au1xpsc_i2s_driver = { | |||
446 | 424 | ||
447 | static int __init au1xpsc_i2s_load(void) | 425 | static int __init au1xpsc_i2s_load(void) |
448 | { | 426 | { |
449 | au1xpsc_i2s_workdata = NULL; | ||
450 | return platform_driver_register(&au1xpsc_i2s_driver); | 427 | return platform_driver_register(&au1xpsc_i2s_driver); |
451 | } | 428 | } |
452 | 429 | ||
diff --git a/sound/soc/au1x/psc.h b/sound/soc/au1x/psc.h index 093775d4dc3e..b30eadd422a7 100644 --- a/sound/soc/au1x/psc.h +++ b/sound/soc/au1x/psc.h | |||
@@ -8,19 +8,11 @@ | |||
8 | * it under the terms of the GNU General Public License version 2 as | 8 | * it under the terms of the GNU General Public License version 2 as |
9 | * published by the Free Software Foundation. | 9 | * published by the Free Software Foundation. |
10 | * | 10 | * |
11 | * NOTE: all of these drivers can only work with a SINGLE instance | ||
12 | * of a PSC. Multiple independent audio devices are impossible | ||
13 | * with ASoC v1. | ||
14 | */ | 11 | */ |
15 | 12 | ||
16 | #ifndef _AU1X_PCM_H | 13 | #ifndef _AU1X_PCM_H |
17 | #define _AU1X_PCM_H | 14 | #define _AU1X_PCM_H |
18 | 15 | ||
19 | extern struct snd_soc_dai au1xpsc_ac97_dai; | ||
20 | extern struct snd_soc_dai au1xpsc_i2s_dai; | ||
21 | extern struct snd_soc_platform au1xpsc_soc_platform; | ||
22 | extern struct snd_ac97_bus_ops soc_ac97_ops; | ||
23 | |||
24 | /* DBDMA helpers */ | 16 | /* DBDMA helpers */ |
25 | extern struct platform_device *au1xpsc_pcm_add(struct platform_device *pdev); | 17 | extern struct platform_device *au1xpsc_pcm_add(struct platform_device *pdev); |
26 | extern void au1xpsc_pcm_destroy(struct platform_device *dmapd); | 18 | extern void au1xpsc_pcm_destroy(struct platform_device *dmapd); |
@@ -31,6 +23,8 @@ struct au1xpsc_audio_data { | |||
31 | unsigned long cfg; | 23 | unsigned long cfg; |
32 | unsigned long rate; | 24 | unsigned long rate; |
33 | 25 | ||
26 | struct snd_soc_dai_driver dai_drv; | ||
27 | |||
34 | unsigned long pm[2]; | 28 | unsigned long pm[2]; |
35 | struct mutex lock; | 29 | struct mutex lock; |
36 | struct platform_device *dmapd; | 30 | struct platform_device *dmapd; |