diff options
Diffstat (limited to 'sound/soc/au1x/psc-i2s.c')
-rw-r--r-- | sound/soc/au1x/psc-i2s.c | 189 |
1 files changed, 120 insertions, 69 deletions
diff --git a/sound/soc/au1x/psc-i2s.c b/sound/soc/au1x/psc-i2s.c index bb589327ee32..0cf2ca61c776 100644 --- a/sound/soc/au1x/psc-i2s.c +++ b/sound/soc/au1x/psc-i2s.c | |||
@@ -2,7 +2,7 @@ | |||
2 | * Au12x0/Au1550 PSC ALSA ASoC audio support. | 2 | * Au12x0/Au1550 PSC ALSA ASoC audio support. |
3 | * | 3 | * |
4 | * (c) 2007-2008 MSC Vertriebsges.m.b.H., | 4 | * (c) 2007-2008 MSC Vertriebsges.m.b.H., |
5 | * Manuel Lauss <mano@roarinelk.homelinux.net> | 5 | * Manuel Lauss <manuel.lauss@gmail.com> |
6 | * | 6 | * |
7 | * This program is free software; you can redistribute it and/or modify | 7 | * This program is free software; you can redistribute it and/or modify |
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 |
@@ -265,16 +265,52 @@ static int au1xpsc_i2s_trigger(struct snd_pcm_substream *substream, int cmd, | |||
265 | static int au1xpsc_i2s_probe(struct platform_device *pdev, | 265 | static int au1xpsc_i2s_probe(struct platform_device *pdev, |
266 | struct snd_soc_dai *dai) | 266 | struct snd_soc_dai *dai) |
267 | { | 267 | { |
268 | return au1xpsc_i2s_workdata ? 0 : -ENODEV; | ||
269 | } | ||
270 | |||
271 | static void au1xpsc_i2s_remove(struct platform_device *pdev, | ||
272 | struct snd_soc_dai *dai) | ||
273 | { | ||
274 | } | ||
275 | |||
276 | static struct snd_soc_dai_ops au1xpsc_i2s_dai_ops = { | ||
277 | .trigger = au1xpsc_i2s_trigger, | ||
278 | .hw_params = au1xpsc_i2s_hw_params, | ||
279 | .set_fmt = au1xpsc_i2s_set_fmt, | ||
280 | }; | ||
281 | |||
282 | struct snd_soc_dai au1xpsc_i2s_dai = { | ||
283 | .name = "au1xpsc_i2s", | ||
284 | .probe = au1xpsc_i2s_probe, | ||
285 | .remove = au1xpsc_i2s_remove, | ||
286 | .playback = { | ||
287 | .rates = AU1XPSC_I2S_RATES, | ||
288 | .formats = AU1XPSC_I2S_FMTS, | ||
289 | .channels_min = 2, | ||
290 | .channels_max = 8, /* 2 without external help */ | ||
291 | }, | ||
292 | .capture = { | ||
293 | .rates = AU1XPSC_I2S_RATES, | ||
294 | .formats = AU1XPSC_I2S_FMTS, | ||
295 | .channels_min = 2, | ||
296 | .channels_max = 8, /* 2 without external help */ | ||
297 | }, | ||
298 | .ops = &au1xpsc_i2s_dai_ops, | ||
299 | }; | ||
300 | EXPORT_SYMBOL(au1xpsc_i2s_dai); | ||
301 | |||
302 | static int __init au1xpsc_i2s_drvprobe(struct platform_device *pdev) | ||
303 | { | ||
268 | struct resource *r; | 304 | struct resource *r; |
269 | unsigned long sel; | 305 | unsigned long sel; |
270 | int ret; | 306 | int ret; |
307 | struct au1xpsc_audio_data *wd; | ||
271 | 308 | ||
272 | if (au1xpsc_i2s_workdata) | 309 | if (au1xpsc_i2s_workdata) |
273 | return -EBUSY; | 310 | return -EBUSY; |
274 | 311 | ||
275 | au1xpsc_i2s_workdata = | 312 | wd = kzalloc(sizeof(struct au1xpsc_audio_data), GFP_KERNEL); |
276 | kzalloc(sizeof(struct au1xpsc_audio_data), GFP_KERNEL); | 313 | if (!wd) |
277 | if (!au1xpsc_i2s_workdata) | ||
278 | return -ENOMEM; | 314 | return -ENOMEM; |
279 | 315 | ||
280 | r = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 316 | r = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
@@ -284,131 +320,146 @@ static int au1xpsc_i2s_probe(struct platform_device *pdev, | |||
284 | } | 320 | } |
285 | 321 | ||
286 | ret = -EBUSY; | 322 | ret = -EBUSY; |
287 | au1xpsc_i2s_workdata->ioarea = | 323 | wd->ioarea = request_mem_region(r->start, r->end - r->start + 1, |
288 | request_mem_region(r->start, r->end - r->start + 1, | ||
289 | "au1xpsc_i2s"); | 324 | "au1xpsc_i2s"); |
290 | if (!au1xpsc_i2s_workdata->ioarea) | 325 | if (!wd->ioarea) |
291 | goto out0; | 326 | goto out0; |
292 | 327 | ||
293 | au1xpsc_i2s_workdata->mmio = ioremap(r->start, 0xffff); | 328 | wd->mmio = ioremap(r->start, 0xffff); |
294 | if (!au1xpsc_i2s_workdata->mmio) | 329 | if (!wd->mmio) |
295 | goto out1; | 330 | goto out1; |
296 | 331 | ||
297 | /* preserve PSC clock source set up by platform (dev.platform_data | 332 | /* preserve PSC clock source set up by platform (dev.platform_data |
298 | * is already occupied by soc layer) | 333 | * is already occupied by soc layer) |
299 | */ | 334 | */ |
300 | sel = au_readl(PSC_SEL(au1xpsc_i2s_workdata)) & PSC_SEL_CLK_MASK; | 335 | sel = au_readl(PSC_SEL(wd)) & PSC_SEL_CLK_MASK; |
301 | au_writel(PSC_CTRL_DISABLE, PSC_CTRL(au1xpsc_i2s_workdata)); | 336 | au_writel(PSC_CTRL_DISABLE, PSC_CTRL(wd)); |
302 | au_sync(); | 337 | au_sync(); |
303 | au_writel(PSC_SEL_PS_I2SMODE | sel, PSC_SEL(au1xpsc_i2s_workdata)); | 338 | au_writel(PSC_SEL_PS_I2SMODE | sel, PSC_SEL(wd)); |
304 | au_writel(0, I2S_CFG(au1xpsc_i2s_workdata)); | 339 | au_writel(0, I2S_CFG(wd)); |
305 | au_sync(); | 340 | au_sync(); |
306 | 341 | ||
307 | /* preconfigure: set max rx/tx fifo depths */ | 342 | /* preconfigure: set max rx/tx fifo depths */ |
308 | au1xpsc_i2s_workdata->cfg |= | 343 | wd->cfg |= PSC_I2SCFG_RT_FIFO8 | PSC_I2SCFG_TT_FIFO8; |
309 | PSC_I2SCFG_RT_FIFO8 | PSC_I2SCFG_TT_FIFO8; | ||
310 | 344 | ||
311 | /* don't wait for I2S core to become ready now; clocks may not | 345 | /* don't wait for I2S core to become ready now; clocks may not |
312 | * be running yet; depending on clock input for PSC a wait might | 346 | * be running yet; depending on clock input for PSC a wait might |
313 | * time out. | 347 | * time out. |
314 | */ | 348 | */ |
315 | 349 | ||
316 | return 0; | 350 | ret = snd_soc_register_dai(&au1xpsc_i2s_dai); |
351 | if (ret) | ||
352 | goto out1; | ||
317 | 353 | ||
354 | /* finally add the DMA device for this PSC */ | ||
355 | wd->dmapd = au1xpsc_pcm_add(pdev); | ||
356 | if (wd->dmapd) { | ||
357 | platform_set_drvdata(pdev, wd); | ||
358 | au1xpsc_i2s_workdata = wd; | ||
359 | return 0; | ||
360 | } | ||
361 | |||
362 | snd_soc_unregister_dai(&au1xpsc_i2s_dai); | ||
318 | out1: | 363 | out1: |
319 | release_resource(au1xpsc_i2s_workdata->ioarea); | 364 | release_resource(wd->ioarea); |
320 | kfree(au1xpsc_i2s_workdata->ioarea); | 365 | kfree(wd->ioarea); |
321 | out0: | 366 | out0: |
322 | kfree(au1xpsc_i2s_workdata); | 367 | kfree(wd); |
323 | au1xpsc_i2s_workdata = NULL; | ||
324 | return ret; | 368 | return ret; |
325 | } | 369 | } |
326 | 370 | ||
327 | static void au1xpsc_i2s_remove(struct platform_device *pdev, | 371 | static int __devexit au1xpsc_i2s_drvremove(struct platform_device *pdev) |
328 | struct snd_soc_dai *dai) | ||
329 | { | 372 | { |
330 | au_writel(0, I2S_CFG(au1xpsc_i2s_workdata)); | 373 | struct au1xpsc_audio_data *wd = platform_get_drvdata(pdev); |
374 | |||
375 | if (wd->dmapd) | ||
376 | au1xpsc_pcm_destroy(wd->dmapd); | ||
377 | |||
378 | snd_soc_unregister_dai(&au1xpsc_i2s_dai); | ||
379 | |||
380 | au_writel(0, I2S_CFG(wd)); | ||
331 | au_sync(); | 381 | au_sync(); |
332 | au_writel(PSC_CTRL_DISABLE, PSC_CTRL(au1xpsc_i2s_workdata)); | 382 | au_writel(PSC_CTRL_DISABLE, PSC_CTRL(wd)); |
333 | au_sync(); | 383 | au_sync(); |
334 | 384 | ||
335 | iounmap(au1xpsc_i2s_workdata->mmio); | 385 | iounmap(wd->mmio); |
336 | release_resource(au1xpsc_i2s_workdata->ioarea); | 386 | release_resource(wd->ioarea); |
337 | kfree(au1xpsc_i2s_workdata->ioarea); | 387 | kfree(wd->ioarea); |
338 | kfree(au1xpsc_i2s_workdata); | 388 | kfree(wd); |
339 | au1xpsc_i2s_workdata = NULL; | 389 | |
390 | au1xpsc_i2s_workdata = NULL; /* MDEV */ | ||
391 | |||
392 | return 0; | ||
340 | } | 393 | } |
341 | 394 | ||
342 | static int au1xpsc_i2s_suspend(struct snd_soc_dai *cpu_dai) | 395 | #ifdef CONFIG_PM |
396 | static int au1xpsc_i2s_drvsuspend(struct device *dev) | ||
343 | { | 397 | { |
398 | struct au1xpsc_audio_data *wd = dev_get_drvdata(dev); | ||
399 | |||
344 | /* save interesting register and disable PSC */ | 400 | /* save interesting register and disable PSC */ |
345 | au1xpsc_i2s_workdata->pm[0] = | 401 | wd->pm[0] = au_readl(PSC_SEL(wd)); |
346 | au_readl(PSC_SEL(au1xpsc_i2s_workdata)); | ||
347 | 402 | ||
348 | au_writel(0, I2S_CFG(au1xpsc_i2s_workdata)); | 403 | au_writel(0, I2S_CFG(wd)); |
349 | au_sync(); | 404 | au_sync(); |
350 | au_writel(PSC_CTRL_DISABLE, PSC_CTRL(au1xpsc_i2s_workdata)); | 405 | au_writel(PSC_CTRL_DISABLE, PSC_CTRL(wd)); |
351 | au_sync(); | 406 | au_sync(); |
352 | 407 | ||
353 | return 0; | 408 | return 0; |
354 | } | 409 | } |
355 | 410 | ||
356 | static int au1xpsc_i2s_resume(struct snd_soc_dai *cpu_dai) | 411 | static int au1xpsc_i2s_drvresume(struct device *dev) |
357 | { | 412 | { |
413 | struct au1xpsc_audio_data *wd = dev_get_drvdata(dev); | ||
414 | |||
358 | /* select I2S mode and PSC clock */ | 415 | /* select I2S mode and PSC clock */ |
359 | au_writel(PSC_CTRL_DISABLE, PSC_CTRL(au1xpsc_i2s_workdata)); | 416 | au_writel(PSC_CTRL_DISABLE, PSC_CTRL(wd)); |
360 | au_sync(); | 417 | au_sync(); |
361 | au_writel(0, PSC_SEL(au1xpsc_i2s_workdata)); | 418 | au_writel(0, PSC_SEL(wd)); |
362 | au_sync(); | 419 | au_sync(); |
363 | au_writel(au1xpsc_i2s_workdata->pm[0], | 420 | au_writel(wd->pm[0], PSC_SEL(wd)); |
364 | PSC_SEL(au1xpsc_i2s_workdata)); | ||
365 | au_sync(); | 421 | au_sync(); |
366 | 422 | ||
367 | return 0; | 423 | return 0; |
368 | } | 424 | } |
369 | 425 | ||
370 | static struct snd_soc_dai_ops au1xpsc_i2s_dai_ops = { | 426 | static struct dev_pm_ops au1xpsci2s_pmops = { |
371 | .trigger = au1xpsc_i2s_trigger, | 427 | .suspend = au1xpsc_i2s_drvsuspend, |
372 | .hw_params = au1xpsc_i2s_hw_params, | 428 | .resume = au1xpsc_i2s_drvresume, |
373 | .set_fmt = au1xpsc_i2s_set_fmt, | ||
374 | }; | 429 | }; |
375 | 430 | ||
376 | struct snd_soc_dai au1xpsc_i2s_dai = { | 431 | #define AU1XPSCI2S_PMOPS &au1xpsci2s_pmops |
377 | .name = "au1xpsc_i2s", | 432 | |
378 | .probe = au1xpsc_i2s_probe, | 433 | #else |
379 | .remove = au1xpsc_i2s_remove, | 434 | |
380 | .suspend = au1xpsc_i2s_suspend, | 435 | #define AU1XPSCI2S_PMOPS NULL |
381 | .resume = au1xpsc_i2s_resume, | 436 | |
382 | .playback = { | 437 | #endif |
383 | .rates = AU1XPSC_I2S_RATES, | 438 | |
384 | .formats = AU1XPSC_I2S_FMTS, | 439 | static struct platform_driver au1xpsc_i2s_driver = { |
385 | .channels_min = 2, | 440 | .driver = { |
386 | .channels_max = 8, /* 2 without external help */ | 441 | .name = "au1xpsc_i2s", |
387 | }, | 442 | .owner = THIS_MODULE, |
388 | .capture = { | 443 | .pm = AU1XPSCI2S_PMOPS, |
389 | .rates = AU1XPSC_I2S_RATES, | ||
390 | .formats = AU1XPSC_I2S_FMTS, | ||
391 | .channels_min = 2, | ||
392 | .channels_max = 8, /* 2 without external help */ | ||
393 | }, | 444 | }, |
394 | .ops = &au1xpsc_i2s_dai_ops, | 445 | .probe = au1xpsc_i2s_drvprobe, |
446 | .remove = __devexit_p(au1xpsc_i2s_drvremove), | ||
395 | }; | 447 | }; |
396 | EXPORT_SYMBOL(au1xpsc_i2s_dai); | ||
397 | 448 | ||
398 | static int __init au1xpsc_i2s_init(void) | 449 | static int __init au1xpsc_i2s_load(void) |
399 | { | 450 | { |
400 | au1xpsc_i2s_workdata = NULL; | 451 | au1xpsc_i2s_workdata = NULL; |
401 | return snd_soc_register_dai(&au1xpsc_i2s_dai); | 452 | return platform_driver_register(&au1xpsc_i2s_driver); |
402 | } | 453 | } |
403 | 454 | ||
404 | static void __exit au1xpsc_i2s_exit(void) | 455 | static void __exit au1xpsc_i2s_unload(void) |
405 | { | 456 | { |
406 | snd_soc_unregister_dai(&au1xpsc_i2s_dai); | 457 | platform_driver_unregister(&au1xpsc_i2s_driver); |
407 | } | 458 | } |
408 | 459 | ||
409 | module_init(au1xpsc_i2s_init); | 460 | module_init(au1xpsc_i2s_load); |
410 | module_exit(au1xpsc_i2s_exit); | 461 | module_exit(au1xpsc_i2s_unload); |
411 | 462 | ||
412 | MODULE_LICENSE("GPL"); | 463 | MODULE_LICENSE("GPL"); |
413 | MODULE_DESCRIPTION("Au12x0/Au1550 PSC I2S ALSA ASoC audio driver"); | 464 | MODULE_DESCRIPTION("Au12x0/Au1550 PSC I2S ALSA ASoC audio driver"); |
414 | MODULE_AUTHOR("Manuel Lauss <mano@roarinelk.homelinux.net>"); | 465 | MODULE_AUTHOR("Manuel Lauss"); |