aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm8974.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm8974.c')
-rw-r--r--sound/soc/codecs/wm8974.c180
1 files changed, 49 insertions, 131 deletions
diff --git a/sound/soc/codecs/wm8974.c b/sound/soc/codecs/wm8974.c
index 1468fe10cbbe..ca646a822444 100644
--- a/sound/soc/codecs/wm8974.c
+++ b/sound/soc/codecs/wm8974.c
@@ -23,7 +23,6 @@
23#include <sound/pcm.h> 23#include <sound/pcm.h>
24#include <sound/pcm_params.h> 24#include <sound/pcm_params.h>
25#include <sound/soc.h> 25#include <sound/soc.h>
26#include <sound/soc-dapm.h>
27#include <sound/initval.h> 26#include <sound/initval.h>
28#include <sound/tlv.h> 27#include <sound/tlv.h>
29 28
@@ -51,12 +50,9 @@ static const u16 wm8974_reg[WM8974_CACHEREGNUM] = {
51#define WM8974_POWER1_BUFIOEN 0x04 50#define WM8974_POWER1_BUFIOEN 0x04
52 51
53struct wm8974_priv { 52struct wm8974_priv {
54 struct snd_soc_codec codec; 53 enum snd_soc_control_type control_type;
55 u16 reg_cache[WM8974_CACHEREGNUM];
56}; 54};
57 55
58static struct snd_soc_codec *wm8974_codec;
59
60#define wm8974_reset(c) snd_soc_write(c, WM8974_RESET, 0) 56#define wm8974_reset(c) snd_soc_write(c, WM8974_RESET, 0)
61 57
62static const char *wm8974_companding[] = {"Off", "NC", "u-law", "A-law" }; 58static const char *wm8974_companding[] = {"Off", "NC", "u-law", "A-law" };
@@ -276,10 +272,11 @@ static const struct snd_soc_dapm_route audio_map[] = {
276 272
277static int wm8974_add_widgets(struct snd_soc_codec *codec) 273static int wm8974_add_widgets(struct snd_soc_codec *codec)
278{ 274{
279 snd_soc_dapm_new_controls(codec, wm8974_dapm_widgets, 275 struct snd_soc_dapm_context *dapm = &codec->dapm;
280 ARRAY_SIZE(wm8974_dapm_widgets));
281 276
282 snd_soc_dapm_add_routes(codec, audio_map, ARRAY_SIZE(audio_map)); 277 snd_soc_dapm_new_controls(dapm, wm8974_dapm_widgets,
278 ARRAY_SIZE(wm8974_dapm_widgets));
279 snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map));
283 280
284 return 0; 281 return 0;
285} 282}
@@ -532,7 +529,7 @@ static int wm8974_set_bias_level(struct snd_soc_codec *codec,
532 case SND_SOC_BIAS_STANDBY: 529 case SND_SOC_BIAS_STANDBY:
533 power1 |= WM8974_POWER1_BIASEN | WM8974_POWER1_BUFIOEN; 530 power1 |= WM8974_POWER1_BIASEN | WM8974_POWER1_BUFIOEN;
534 531
535 if (codec->bias_level == SND_SOC_BIAS_OFF) { 532 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
536 /* Initial cap charge at VMID 5k */ 533 /* Initial cap charge at VMID 5k */
537 snd_soc_write(codec, WM8974_POWER1, power1 | 0x3); 534 snd_soc_write(codec, WM8974_POWER1, power1 | 0x3);
538 mdelay(100); 535 mdelay(100);
@@ -549,7 +546,7 @@ static int wm8974_set_bias_level(struct snd_soc_codec *codec,
549 break; 546 break;
550 } 547 }
551 548
552 codec->bias_level = level; 549 codec->dapm.bias_level = level;
553 return 0; 550 return 0;
554} 551}
555 552
@@ -566,8 +563,8 @@ static struct snd_soc_dai_ops wm8974_ops = {
566 .set_pll = wm8974_set_dai_pll, 563 .set_pll = wm8974_set_dai_pll,
567}; 564};
568 565
569struct snd_soc_dai wm8974_dai = { 566static struct snd_soc_dai_driver wm8974_dai = {
570 .name = "WM8974 HiFi", 567 .name = "wm8974-hifi",
571 .playback = { 568 .playback = {
572 .stream_name = "Playback", 569 .stream_name = "Playback",
573 .channels_min = 1, 570 .channels_min = 1,
@@ -583,21 +580,15 @@ struct snd_soc_dai wm8974_dai = {
583 .ops = &wm8974_ops, 580 .ops = &wm8974_ops,
584 .symmetric_rates = 1, 581 .symmetric_rates = 1,
585}; 582};
586EXPORT_SYMBOL_GPL(wm8974_dai);
587 583
588static int wm8974_suspend(struct platform_device *pdev, pm_message_t state) 584static int wm8974_suspend(struct snd_soc_codec *codec, pm_message_t state)
589{ 585{
590 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
591 struct snd_soc_codec *codec = socdev->card->codec;
592
593 wm8974_set_bias_level(codec, SND_SOC_BIAS_OFF); 586 wm8974_set_bias_level(codec, SND_SOC_BIAS_OFF);
594 return 0; 587 return 0;
595} 588}
596 589
597static int wm8974_resume(struct platform_device *pdev) 590static int wm8974_resume(struct snd_soc_codec *codec)
598{ 591{
599 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
600 struct snd_soc_codec *codec = socdev->card->codec;
601 int i; 592 int i;
602 u8 data[2]; 593 u8 data[2];
603 u16 *cache = codec->reg_cache; 594 u16 *cache = codec->reg_cache;
@@ -613,156 +604,72 @@ static int wm8974_resume(struct platform_device *pdev)
613 return 0; 604 return 0;
614} 605}
615 606
616static int wm8974_probe(struct platform_device *pdev) 607static int wm8974_probe(struct snd_soc_codec *codec)
617{ 608{
618 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
619 struct snd_soc_codec *codec;
620 int ret = 0; 609 int ret = 0;
621 610
622 if (wm8974_codec == NULL) { 611 ret = snd_soc_codec_set_cache_io(codec, 7, 9, SND_SOC_I2C);
623 dev_err(&pdev->dev, "Codec device not registered\n"); 612 if (ret < 0) {
624 return -ENODEV; 613 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
614 return ret;
625 } 615 }
626 616
627 socdev->card->codec = wm8974_codec; 617 ret = wm8974_reset(codec);
628 codec = wm8974_codec;
629
630 /* register pcms */
631 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1);
632 if (ret < 0) { 618 if (ret < 0) {
633 dev_err(codec->dev, "failed to create pcms: %d\n", ret); 619 dev_err(codec->dev, "Failed to issue reset\n");
634 goto pcm_err; 620 return ret;
635 } 621 }
636 622
623 wm8974_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
637 snd_soc_add_controls(codec, wm8974_snd_controls, 624 snd_soc_add_controls(codec, wm8974_snd_controls,
638 ARRAY_SIZE(wm8974_snd_controls)); 625 ARRAY_SIZE(wm8974_snd_controls));
639 wm8974_add_widgets(codec); 626 wm8974_add_widgets(codec);
640 627
641 return ret; 628 return ret;
642
643pcm_err:
644 return ret;
645} 629}
646 630
647/* power down chip */ 631/* power down chip */
648static int wm8974_remove(struct platform_device *pdev) 632static int wm8974_remove(struct snd_soc_codec *codec)
649{ 633{
650 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 634 wm8974_set_bias_level(codec, SND_SOC_BIAS_OFF);
651
652 snd_soc_free_pcms(socdev);
653 snd_soc_dapm_free(socdev);
654
655 return 0; 635 return 0;
656} 636}
657 637
658struct snd_soc_codec_device soc_codec_dev_wm8974 = { 638static struct snd_soc_codec_driver soc_codec_dev_wm8974 = {
659 .probe = wm8974_probe, 639 .probe = wm8974_probe,
660 .remove = wm8974_remove, 640 .remove = wm8974_remove,
661 .suspend = wm8974_suspend, 641 .suspend = wm8974_suspend,
662 .resume = wm8974_resume, 642 .resume = wm8974_resume,
643 .set_bias_level = wm8974_set_bias_level,
644 .reg_cache_size = ARRAY_SIZE(wm8974_reg),
645 .reg_word_size = sizeof(u16),
646 .reg_cache_default = wm8974_reg,
663}; 647};
664EXPORT_SYMBOL_GPL(soc_codec_dev_wm8974);
665
666static __devinit int wm8974_register(struct wm8974_priv *wm8974)
667{
668 int ret;
669 struct snd_soc_codec *codec = &wm8974->codec;
670
671 if (wm8974_codec) {
672 dev_err(codec->dev, "Another WM8974 is registered\n");
673 ret = -EINVAL;
674 goto err;
675 }
676
677 mutex_init(&codec->mutex);
678 INIT_LIST_HEAD(&codec->dapm_widgets);
679 INIT_LIST_HEAD(&codec->dapm_paths);
680
681 snd_soc_codec_set_drvdata(codec, wm8974);
682 codec->name = "WM8974";
683 codec->owner = THIS_MODULE;
684 codec->bias_level = SND_SOC_BIAS_OFF;
685 codec->set_bias_level = wm8974_set_bias_level;
686 codec->dai = &wm8974_dai;
687 codec->num_dai = 1;
688 codec->reg_cache_size = WM8974_CACHEREGNUM;
689 codec->reg_cache = &wm8974->reg_cache;
690
691 ret = snd_soc_codec_set_cache_io(codec, 7, 9, SND_SOC_I2C);
692 if (ret < 0) {
693 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
694 goto err;
695 }
696
697 memcpy(codec->reg_cache, wm8974_reg, sizeof(wm8974_reg));
698
699 ret = wm8974_reset(codec);
700 if (ret < 0) {
701 dev_err(codec->dev, "Failed to issue reset\n");
702 goto err;
703 }
704
705 wm8974_dai.dev = codec->dev;
706
707 wm8974_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
708
709 wm8974_codec = codec;
710
711 ret = snd_soc_register_codec(codec);
712 if (ret != 0) {
713 dev_err(codec->dev, "Failed to register codec: %d\n", ret);
714 goto err;
715 }
716
717 ret = snd_soc_register_dai(&wm8974_dai);
718 if (ret != 0) {
719 dev_err(codec->dev, "Failed to register DAI: %d\n", ret);
720 goto err_codec;
721 }
722
723 return 0;
724
725err_codec:
726 snd_soc_unregister_codec(codec);
727err:
728 kfree(wm8974);
729 return ret;
730}
731
732static __devexit void wm8974_unregister(struct wm8974_priv *wm8974)
733{
734 wm8974_set_bias_level(&wm8974->codec, SND_SOC_BIAS_OFF);
735 snd_soc_unregister_dai(&wm8974_dai);
736 snd_soc_unregister_codec(&wm8974->codec);
737 kfree(wm8974);
738 wm8974_codec = NULL;
739}
740 648
649#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
741static __devinit int wm8974_i2c_probe(struct i2c_client *i2c, 650static __devinit int wm8974_i2c_probe(struct i2c_client *i2c,
742 const struct i2c_device_id *id) 651 const struct i2c_device_id *id)
743{ 652{
744 struct wm8974_priv *wm8974; 653 struct wm8974_priv *wm8974;
745 struct snd_soc_codec *codec; 654 int ret;
746 655
747 wm8974 = kzalloc(sizeof(struct wm8974_priv), GFP_KERNEL); 656 wm8974 = kzalloc(sizeof(struct wm8974_priv), GFP_KERNEL);
748 if (wm8974 == NULL) 657 if (wm8974 == NULL)
749 return -ENOMEM; 658 return -ENOMEM;
750 659
751 codec = &wm8974->codec;
752 codec->hw_write = (hw_write_t)i2c_master_send;
753
754 i2c_set_clientdata(i2c, wm8974); 660 i2c_set_clientdata(i2c, wm8974);
755 codec->control_data = i2c;
756
757 codec->dev = &i2c->dev;
758 661
759 return wm8974_register(wm8974); 662 ret = snd_soc_register_codec(&i2c->dev,
663 &soc_codec_dev_wm8974, &wm8974_dai, 1);
664 if (ret < 0)
665 kfree(wm8974);
666 return ret;
760} 667}
761 668
762static __devexit int wm8974_i2c_remove(struct i2c_client *client) 669static __devexit int wm8974_i2c_remove(struct i2c_client *client)
763{ 670{
764 struct wm8974_priv *wm8974 = i2c_get_clientdata(client); 671 snd_soc_unregister_codec(&client->dev);
765 wm8974_unregister(wm8974); 672 kfree(i2c_get_clientdata(client));
766 return 0; 673 return 0;
767} 674}
768 675
@@ -774,23 +681,34 @@ MODULE_DEVICE_TABLE(i2c, wm8974_i2c_id);
774 681
775static struct i2c_driver wm8974_i2c_driver = { 682static struct i2c_driver wm8974_i2c_driver = {
776 .driver = { 683 .driver = {
777 .name = "WM8974", 684 .name = "wm8974-codec",
778 .owner = THIS_MODULE, 685 .owner = THIS_MODULE,
779 }, 686 },
780 .probe = wm8974_i2c_probe, 687 .probe = wm8974_i2c_probe,
781 .remove = __devexit_p(wm8974_i2c_remove), 688 .remove = __devexit_p(wm8974_i2c_remove),
782 .id_table = wm8974_i2c_id, 689 .id_table = wm8974_i2c_id,
783}; 690};
691#endif
784 692
785static int __init wm8974_modinit(void) 693static int __init wm8974_modinit(void)
786{ 694{
787 return i2c_add_driver(&wm8974_i2c_driver); 695 int ret = 0;
696#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
697 ret = i2c_add_driver(&wm8974_i2c_driver);
698 if (ret != 0) {
699 printk(KERN_ERR "Failed to register wm8974 I2C driver: %d\n",
700 ret);
701 }
702#endif
703 return ret;
788} 704}
789module_init(wm8974_modinit); 705module_init(wm8974_modinit);
790 706
791static void __exit wm8974_exit(void) 707static void __exit wm8974_exit(void)
792{ 708{
709#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
793 i2c_del_driver(&wm8974_i2c_driver); 710 i2c_del_driver(&wm8974_i2c_driver);
711#endif
794} 712}
795module_exit(wm8974_exit); 713module_exit(wm8974_exit);
796 714