aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm8400.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm8400.c')
-rw-r--r--sound/soc/codecs/wm8400.c194
1 files changed, 56 insertions, 138 deletions
diff --git a/sound/soc/codecs/wm8400.c b/sound/soc/codecs/wm8400.c
index 8f294066b0ed..fbee556cbf35 100644
--- a/sound/soc/codecs/wm8400.c
+++ b/sound/soc/codecs/wm8400.c
@@ -22,11 +22,11 @@
22#include <linux/regulator/consumer.h> 22#include <linux/regulator/consumer.h>
23#include <linux/mfd/wm8400-audio.h> 23#include <linux/mfd/wm8400-audio.h>
24#include <linux/mfd/wm8400-private.h> 24#include <linux/mfd/wm8400-private.h>
25#include <linux/mfd/core.h>
25#include <sound/core.h> 26#include <sound/core.h>
26#include <sound/pcm.h> 27#include <sound/pcm.h>
27#include <sound/pcm_params.h> 28#include <sound/pcm_params.h>
28#include <sound/soc.h> 29#include <sound/soc.h>
29#include <sound/soc-dapm.h>
30#include <sound/initval.h> 30#include <sound/initval.h>
31#include <sound/tlv.h> 31#include <sound/tlv.h>
32 32
@@ -65,7 +65,7 @@ static struct regulator_bulk_data power[] = {
65 65
66/* codec private data */ 66/* codec private data */
67struct wm8400_priv { 67struct wm8400_priv {
68 struct snd_soc_codec codec; 68 struct snd_soc_codec *codec;
69 struct wm8400 *wm8400; 69 struct wm8400 *wm8400;
70 u16 fake_register; 70 u16 fake_register;
71 unsigned int sysclk; 71 unsigned int sysclk;
@@ -911,10 +911,11 @@ static const struct snd_soc_dapm_route audio_map[] = {
911 911
912static int wm8400_add_widgets(struct snd_soc_codec *codec) 912static int wm8400_add_widgets(struct snd_soc_codec *codec)
913{ 913{
914 snd_soc_dapm_new_controls(codec, wm8400_dapm_widgets, 914 struct snd_soc_dapm_context *dapm = &codec->dapm;
915 ARRAY_SIZE(wm8400_dapm_widgets));
916 915
917 snd_soc_dapm_add_routes(codec, audio_map, ARRAY_SIZE(audio_map)); 916 snd_soc_dapm_new_controls(dapm, wm8400_dapm_widgets,
917 ARRAY_SIZE(wm8400_dapm_widgets));
918 snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map));
918 919
919 return 0; 920 return 0;
920} 921}
@@ -1163,8 +1164,7 @@ static int wm8400_hw_params(struct snd_pcm_substream *substream,
1163 struct snd_soc_dai *dai) 1164 struct snd_soc_dai *dai)
1164{ 1165{
1165 struct snd_soc_pcm_runtime *rtd = substream->private_data; 1166 struct snd_soc_pcm_runtime *rtd = substream->private_data;
1166 struct snd_soc_device *socdev = rtd->socdev; 1167 struct snd_soc_codec *codec = rtd->codec;
1167 struct snd_soc_codec *codec = socdev->card->codec;
1168 u16 audio1 = wm8400_read(codec, WM8400_AUDIO_INTERFACE_1); 1168 u16 audio1 = wm8400_read(codec, WM8400_AUDIO_INTERFACE_1);
1169 1169
1170 audio1 &= ~WM8400_AIF_WL_MASK; 1170 audio1 &= ~WM8400_AIF_WL_MASK;
@@ -1220,7 +1220,7 @@ static int wm8400_set_bias_level(struct snd_soc_codec *codec,
1220 break; 1220 break;
1221 1221
1222 case SND_SOC_BIAS_STANDBY: 1222 case SND_SOC_BIAS_STANDBY:
1223 if (codec->bias_level == SND_SOC_BIAS_OFF) { 1223 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
1224 ret = regulator_bulk_enable(ARRAY_SIZE(power), 1224 ret = regulator_bulk_enable(ARRAY_SIZE(power),
1225 &power[0]); 1225 &power[0]);
1226 if (ret != 0) { 1226 if (ret != 0) {
@@ -1307,7 +1307,7 @@ static int wm8400_set_bias_level(struct snd_soc_codec *codec,
1307 break; 1307 break;
1308 } 1308 }
1309 1309
1310 codec->bias_level = level; 1310 codec->dapm.bias_level = level;
1311 return 0; 1311 return 0;
1312} 1312}
1313 1313
@@ -1332,10 +1332,9 @@ static struct snd_soc_dai_ops wm8400_dai_ops = {
1332 * 1. ADC/DAC on Primary Interface 1332 * 1. ADC/DAC on Primary Interface
1333 * 2. ADC on Primary Interface/DAC on secondary 1333 * 2. ADC on Primary Interface/DAC on secondary
1334 */ 1334 */
1335struct snd_soc_dai wm8400_dai = { 1335static struct snd_soc_dai_driver wm8400_dai = {
1336/* ADC/DAC on primary */ 1336/* ADC/DAC on primary */
1337 .name = "WM8400 ADC/DAC Primary", 1337 .name = "wm8400-hifi",
1338 .id = 1,
1339 .playback = { 1338 .playback = {
1340 .stream_name = "Playback", 1339 .stream_name = "Playback",
1341 .channels_min = 1, 1340 .channels_min = 1,
@@ -1352,147 +1351,53 @@ struct snd_soc_dai wm8400_dai = {
1352 }, 1351 },
1353 .ops = &wm8400_dai_ops, 1352 .ops = &wm8400_dai_ops,
1354}; 1353};
1355EXPORT_SYMBOL_GPL(wm8400_dai);
1356 1354
1357static int wm8400_suspend(struct platform_device *pdev, pm_message_t state) 1355static int wm8400_suspend(struct snd_soc_codec *codec, pm_message_t state)
1358{ 1356{
1359 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1360 struct snd_soc_codec *codec = socdev->card->codec;
1361
1362 wm8400_set_bias_level(codec, SND_SOC_BIAS_OFF); 1357 wm8400_set_bias_level(codec, SND_SOC_BIAS_OFF);
1363 1358
1364 return 0; 1359 return 0;
1365} 1360}
1366 1361
1367static int wm8400_resume(struct platform_device *pdev) 1362static int wm8400_resume(struct snd_soc_codec *codec)
1368{ 1363{
1369 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1370 struct snd_soc_codec *codec = socdev->card->codec;
1371
1372 wm8400_set_bias_level(codec, SND_SOC_BIAS_STANDBY); 1364 wm8400_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
1373 1365
1374 return 0; 1366 return 0;
1375} 1367}
1376 1368
1377static struct snd_soc_codec *wm8400_codec;
1378
1379static int wm8400_probe(struct platform_device *pdev)
1380{
1381 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1382 struct snd_soc_codec *codec;
1383 int ret;
1384
1385 if (!wm8400_codec) {
1386 dev_err(&pdev->dev, "wm8400 not yet discovered\n");
1387 return -ENODEV;
1388 }
1389 codec = wm8400_codec;
1390
1391 socdev->card->codec = codec;
1392
1393 /* register pcms */
1394 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1);
1395 if (ret < 0) {
1396 dev_err(&pdev->dev, "failed to create pcms\n");
1397 goto pcm_err;
1398 }
1399
1400 wm8400_add_controls(codec);
1401 wm8400_add_widgets(codec);
1402
1403pcm_err:
1404 return ret;
1405}
1406
1407/* power down chip */
1408static int wm8400_remove(struct platform_device *pdev)
1409{
1410 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1411
1412 snd_soc_free_pcms(socdev);
1413 snd_soc_dapm_free(socdev);
1414
1415 return 0;
1416}
1417
1418struct snd_soc_codec_device soc_codec_dev_wm8400 = {
1419 .probe = wm8400_probe,
1420 .remove = wm8400_remove,
1421 .suspend = wm8400_suspend,
1422 .resume = wm8400_resume,
1423};
1424
1425static void wm8400_probe_deferred(struct work_struct *work) 1369static void wm8400_probe_deferred(struct work_struct *work)
1426{ 1370{
1427 struct wm8400_priv *priv = container_of(work, struct wm8400_priv, 1371 struct wm8400_priv *priv = container_of(work, struct wm8400_priv,
1428 work); 1372 work);
1429 struct snd_soc_codec *codec = &priv->codec; 1373 struct snd_soc_codec *codec = priv->codec;
1430 int ret;
1431 1374
1432 /* charge output caps */ 1375 /* charge output caps */
1433 wm8400_set_bias_level(codec, SND_SOC_BIAS_STANDBY); 1376 wm8400_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
1434
1435 /* We're done, tell the subsystem. */
1436 ret = snd_soc_register_codec(codec);
1437 if (ret != 0) {
1438 dev_err(priv->wm8400->dev,
1439 "Failed to register codec: %d\n", ret);
1440 goto err;
1441 }
1442
1443 ret = snd_soc_register_dai(&wm8400_dai);
1444 if (ret != 0) {
1445 dev_err(priv->wm8400->dev,
1446 "Failed to register DAI: %d\n", ret);
1447 goto err_codec;
1448 }
1449
1450 return;
1451
1452err_codec:
1453 snd_soc_unregister_codec(codec);
1454err:
1455 wm8400_set_bias_level(codec, SND_SOC_BIAS_OFF);
1456} 1377}
1457 1378
1458static int wm8400_codec_probe(struct platform_device *dev) 1379static int wm8400_codec_probe(struct snd_soc_codec *codec)
1459{ 1380{
1381 struct wm8400 *wm8400 = dev_get_platdata(codec->dev);
1460 struct wm8400_priv *priv; 1382 struct wm8400_priv *priv;
1461 int ret; 1383 int ret;
1462 u16 reg; 1384 u16 reg;
1463 struct snd_soc_codec *codec;
1464 1385
1465 priv = kzalloc(sizeof(struct wm8400_priv), GFP_KERNEL); 1386 priv = kzalloc(sizeof(struct wm8400_priv), GFP_KERNEL);
1466 if (priv == NULL) 1387 if (priv == NULL)
1467 return -ENOMEM; 1388 return -ENOMEM;
1468 1389
1469 codec = &priv->codec;
1470 snd_soc_codec_set_drvdata(codec, priv); 1390 snd_soc_codec_set_drvdata(codec, priv);
1471 codec->control_data = dev_get_drvdata(&dev->dev); 1391 codec->control_data = priv->wm8400 = wm8400;
1472 priv->wm8400 = dev_get_drvdata(&dev->dev); 1392 priv->codec = codec;
1473 1393
1474 ret = regulator_bulk_get(priv->wm8400->dev, 1394 ret = regulator_bulk_get(wm8400->dev,
1475 ARRAY_SIZE(power), &power[0]); 1395 ARRAY_SIZE(power), &power[0]);
1476 if (ret != 0) { 1396 if (ret != 0) {
1477 dev_err(&dev->dev, "Failed to get regulators: %d\n", ret); 1397 dev_err(codec->dev, "Failed to get regulators: %d\n", ret);
1478 goto err; 1398 goto err;
1479 } 1399 }
1480 1400
1481 codec->dev = &dev->dev;
1482 wm8400_dai.dev = &dev->dev;
1483
1484 codec->name = "WM8400";
1485 codec->owner = THIS_MODULE;
1486 codec->read = wm8400_read;
1487 codec->write = wm8400_write;
1488 codec->bias_level = SND_SOC_BIAS_OFF;
1489 codec->set_bias_level = wm8400_set_bias_level;
1490 codec->dai = &wm8400_dai;
1491 codec->num_dai = 1;
1492 codec->reg_cache_size = WM8400_REGISTER_COUNT;
1493 mutex_init(&codec->mutex);
1494 INIT_LIST_HEAD(&codec->dapm_widgets);
1495 INIT_LIST_HEAD(&codec->dapm_paths);
1496 INIT_WORK(&priv->work, wm8400_probe_deferred); 1401 INIT_WORK(&priv->work, wm8400_probe_deferred);
1497 1402
1498 wm8400_codec_reset(codec); 1403 wm8400_codec_reset(codec);
@@ -1511,65 +1416,78 @@ static int wm8400_codec_probe(struct platform_device *dev)
1511 wm8400_write(codec, WM8400_LEFT_OUTPUT_VOLUME, 0x50 | (1<<8)); 1416 wm8400_write(codec, WM8400_LEFT_OUTPUT_VOLUME, 0x50 | (1<<8));
1512 wm8400_write(codec, WM8400_RIGHT_OUTPUT_VOLUME, 0x50 | (1<<8)); 1417 wm8400_write(codec, WM8400_RIGHT_OUTPUT_VOLUME, 0x50 | (1<<8));
1513 1418
1514 wm8400_codec = codec;
1515
1516 if (!schedule_work(&priv->work)) { 1419 if (!schedule_work(&priv->work)) {
1517 ret = -EINVAL; 1420 ret = -EINVAL;
1518 goto err_regulator; 1421 goto err_regulator;
1519 } 1422 }
1520 1423 wm8400_add_controls(codec);
1424 wm8400_add_widgets(codec);
1521 return 0; 1425 return 0;
1522 1426
1523err_regulator: 1427err_regulator:
1524 wm8400_codec = NULL;
1525 regulator_bulk_free(ARRAY_SIZE(power), power); 1428 regulator_bulk_free(ARRAY_SIZE(power), power);
1526err: 1429err:
1527 kfree(priv); 1430 kfree(priv);
1528 return ret; 1431 return ret;
1529} 1432}
1530 1433
1531static int __exit wm8400_codec_remove(struct platform_device *dev) 1434static int wm8400_codec_remove(struct snd_soc_codec *codec)
1532{ 1435{
1533 struct wm8400_priv *priv = snd_soc_codec_get_drvdata(wm8400_codec); 1436 struct wm8400_priv *priv = snd_soc_codec_get_drvdata(codec);
1534 u16 reg; 1437 u16 reg;
1535 1438
1536 snd_soc_unregister_dai(&wm8400_dai); 1439 reg = wm8400_read(codec, WM8400_POWER_MANAGEMENT_1);
1537 snd_soc_unregister_codec(wm8400_codec); 1440 wm8400_write(codec, WM8400_POWER_MANAGEMENT_1,
1538
1539 reg = wm8400_read(wm8400_codec, WM8400_POWER_MANAGEMENT_1);
1540 wm8400_write(wm8400_codec, WM8400_POWER_MANAGEMENT_1,
1541 reg & (~WM8400_CODEC_ENA)); 1441 reg & (~WM8400_CODEC_ENA));
1542 1442
1543 regulator_bulk_free(ARRAY_SIZE(power), power); 1443 regulator_bulk_free(ARRAY_SIZE(power), power);
1544 kfree(priv); 1444 kfree(priv);
1545 1445
1546 wm8400_codec = NULL; 1446 return 0;
1447}
1547 1448
1449static struct snd_soc_codec_driver soc_codec_dev_wm8400 = {
1450 .probe = wm8400_codec_probe,
1451 .remove = wm8400_codec_remove,
1452 .suspend = wm8400_suspend,
1453 .resume = wm8400_resume,
1454 .read = wm8400_read,
1455 .write = wm8400_write,
1456 .set_bias_level = wm8400_set_bias_level,
1457};
1458
1459static int __devinit wm8400_probe(struct platform_device *pdev)
1460{
1461 return snd_soc_register_codec(&pdev->dev, &soc_codec_dev_wm8400,
1462 &wm8400_dai, 1);
1463}
1464
1465static int __devexit wm8400_remove(struct platform_device *pdev)
1466{
1467 snd_soc_unregister_codec(&pdev->dev);
1548 return 0; 1468 return 0;
1549} 1469}
1550 1470
1551static struct platform_driver wm8400_codec_driver = { 1471static struct platform_driver wm8400_codec_driver = {
1552 .driver = { 1472 .driver = {
1553 .name = "wm8400-codec", 1473 .name = "wm8400-codec",
1554 .owner = THIS_MODULE, 1474 .owner = THIS_MODULE,
1555 }, 1475 },
1556 .probe = wm8400_codec_probe, 1476 .probe = wm8400_probe,
1557 .remove = __exit_p(wm8400_codec_remove), 1477 .remove = __devexit_p(wm8400_remove),
1558}; 1478};
1559 1479
1560static int __init wm8400_codec_init(void) 1480static __init int wm8400_init(void)
1561{ 1481{
1562 return platform_driver_register(&wm8400_codec_driver); 1482 return platform_driver_register(&wm8400_codec_driver);
1563} 1483}
1564module_init(wm8400_codec_init); 1484module_init(wm8400_init);
1565 1485
1566static void __exit wm8400_codec_exit(void) 1486static __exit void wm8400_exit(void)
1567{ 1487{
1568 platform_driver_unregister(&wm8400_codec_driver); 1488 platform_driver_unregister(&wm8400_codec_driver);
1569} 1489}
1570module_exit(wm8400_codec_exit); 1490module_exit(wm8400_exit);
1571
1572EXPORT_SYMBOL_GPL(soc_codec_dev_wm8400);
1573 1491
1574MODULE_DESCRIPTION("ASoC WM8400 driver"); 1492MODULE_DESCRIPTION("ASoC WM8400 driver");
1575MODULE_AUTHOR("Mark Brown"); 1493MODULE_AUTHOR("Mark Brown");