aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/au1x
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/au1x')
-rw-r--r--sound/soc/au1x/db1200.c40
-rw-r--r--sound/soc/au1x/dbdma2.c95
-rw-r--r--sound/soc/au1x/psc-ac97.c71
-rw-r--r--sound/soc/au1x/psc-i2s.c53
-rw-r--r--sound/soc/au1x/psc.h10
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 @@
28static struct snd_soc_dai_link db1200_ac97_dai = { 26static 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
35static struct snd_soc_card db1200_ac97_machine = { 35static 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
42static 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 = {
49static int db1200_i2s_startup(struct snd_pcm_substream *substream) 43static 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 = {
80static struct snd_soc_dai_link db1200_i2s_dai = { 74static 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
95static 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!!!! */
65static 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
196static 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
202static int au1xpsc_pcm_hw_params(struct snd_pcm_substream *substream, 204static 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
250static int au1xpsc_pcm_prepare(struct snd_pcm_substream *substream) 252static 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
268static int au1xpsc_pcm_trigger(struct snd_pcm_substream *substream, int cmd) 269static 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)
287static snd_pcm_uframes_t 288static snd_pcm_uframes_t
288au1xpsc_pcm_pointer(struct snd_pcm_substream *substream) 289au1xpsc_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
294static int au1xpsc_pcm_open(struct snd_pcm_substream *substream) 294static 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
300static int au1xpsc_pcm_close(struct snd_pcm_substream *substream) 300static 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
332static 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
340static int au1xpsc_pcm_remove(struct platform_device *pdev)
341{
342 return 0;
343}
344
345/* au1xpsc audio platform */ 332/* au1xpsc audio platform */
346struct snd_soc_platform au1xpsc_soc_platform = { 333struct 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};
354EXPORT_SYMBOL_GPL(au1xpsc_soc_platform);
355 338
356static int __devinit au1xpsc_pcm_drvprobe(struct platform_device *pdev) 339static 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
394out2:
395 kfree(au1xpsc_audio_pcmdma[PCM_RX]);
396 au1xpsc_audio_pcmdma[PCM_RX] = NULL;
397out1: 370out1:
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
403static int __devexit au1xpsc_pcm_drvremove(struct platform_device *pdev) 375static 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
429static int __init au1xpsc_audio_dbdma_load(void) 394static 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!!!! */
57static struct au1xpsc_audio_data *au1xpsc_ac97_workdata; 54static 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 */
62static 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 */
60static unsigned short au1xpsc_ac97_read(struct snd_ac97 *ac97, 75static 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,
102static void au1xpsc_ac97_write(struct snd_ac97 *ac97, unsigned short reg, 116static 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 */
135static void au1xpsc_ac97_warm_reset(struct snd_ac97 *ac97) 148static 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
147static void au1xpsc_ac97_cold_reset(struct snd_ac97 *ac97) 159static 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:
283static int au1xpsc_ac97_trigger(struct snd_pcm_substream *substream, 293static 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
318static int au1xpsc_ac97_probe(struct platform_device *pdev, 327static 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
324static void au1xpsc_ac97_remove(struct platform_device *pdev,
325 struct snd_soc_dai *dai)
326{
327}
328
329static struct snd_soc_dai_ops au1xpsc_ac97_dai_ops = { 332static 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
334struct snd_soc_dai au1xpsc_ac97_dai = { 337static 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};
353EXPORT_SYMBOL_GPL(au1xpsc_ac97_dai);
354 354
355static int __devinit au1xpsc_ac97_drvprobe(struct platform_device *pdev) 355static 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);
410out1: 413out1:
411 release_mem_region(r->start, resource_size(r)); 414 release_mem_region(r->start, resource_size(r));
412out0: 415out0:
@@ -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!!!! */
58static struct au1xpsc_audio_data *au1xpsc_i2s_workdata;
59
60static int au1xpsc_i2s_set_fmt(struct snd_soc_dai *cpu_dai, 54static 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)
245static int au1xpsc_i2s_trigger(struct snd_pcm_substream *substream, int cmd, 239static 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
266static int au1xpsc_i2s_probe(struct platform_device *pdev,
267 struct snd_soc_dai *dai)
268{
269 return au1xpsc_i2s_workdata ? 0 : -ENODEV;
270}
271
272static void au1xpsc_i2s_remove(struct platform_device *pdev,
273 struct snd_soc_dai *dai)
274{
275}
276
277static struct snd_soc_dai_ops au1xpsc_i2s_dai_ops = { 260static 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
283struct snd_soc_dai au1xpsc_i2s_dai = { 266static 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};
301EXPORT_SYMBOL(au1xpsc_i2s_dai);
302 281
303static int __devinit au1xpsc_i2s_drvprobe(struct platform_device *pdev) 282static 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);
362out1: 342out1:
363 release_mem_region(r->start, resource_size(r)); 343 release_mem_region(r->start, resource_size(r));
364out0: 344out0:
@@ -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
447static int __init au1xpsc_i2s_load(void) 425static 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
19extern struct snd_soc_dai au1xpsc_ac97_dai;
20extern struct snd_soc_dai au1xpsc_i2s_dai;
21extern struct snd_soc_platform au1xpsc_soc_platform;
22extern struct snd_ac97_bus_ops soc_ac97_ops;
23
24/* DBDMA helpers */ 16/* DBDMA helpers */
25extern struct platform_device *au1xpsc_pcm_add(struct platform_device *pdev); 17extern struct platform_device *au1xpsc_pcm_add(struct platform_device *pdev);
26extern void au1xpsc_pcm_destroy(struct platform_device *dmapd); 18extern 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;