aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm8750.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm8750.c')
-rw-r--r--sound/soc/codecs/wm8750.c264
1 files changed, 79 insertions, 185 deletions
diff --git a/sound/soc/codecs/wm8750.c b/sound/soc/codecs/wm8750.c
index e2c05e3e323a..6c924cd2cfd4 100644
--- a/sound/soc/codecs/wm8750.c
+++ b/sound/soc/codecs/wm8750.c
@@ -52,7 +52,7 @@ static const u16 wm8750_reg[] = {
52/* codec private data */ 52/* codec private data */
53struct wm8750_priv { 53struct wm8750_priv {
54 unsigned int sysclk; 54 unsigned int sysclk;
55 struct snd_soc_codec codec; 55 enum snd_soc_control_type control_type;
56 u16 reg_cache[ARRAY_SIZE(wm8750_reg)]; 56 u16 reg_cache[ARRAY_SIZE(wm8750_reg)];
57}; 57};
58 58
@@ -560,8 +560,7 @@ static int wm8750_pcm_hw_params(struct snd_pcm_substream *substream,
560 struct snd_soc_dai *dai) 560 struct snd_soc_dai *dai)
561{ 561{
562 struct snd_soc_pcm_runtime *rtd = substream->private_data; 562 struct snd_soc_pcm_runtime *rtd = substream->private_data;
563 struct snd_soc_device *socdev = rtd->socdev; 563 struct snd_soc_codec *codec = rtd->codec;
564 struct snd_soc_codec *codec = socdev->card->codec;
565 struct wm8750_priv *wm8750 = snd_soc_codec_get_drvdata(codec); 564 struct wm8750_priv *wm8750 = snd_soc_codec_get_drvdata(codec);
566 u16 iface = snd_soc_read(codec, WM8750_IFACE) & 0x1f3; 565 u16 iface = snd_soc_read(codec, WM8750_IFACE) & 0x1f3;
567 u16 srate = snd_soc_read(codec, WM8750_SRATE) & 0x1c0; 566 u16 srate = snd_soc_read(codec, WM8750_SRATE) & 0x1c0;
@@ -649,8 +648,8 @@ static struct snd_soc_dai_ops wm8750_dai_ops = {
649 .set_sysclk = wm8750_set_dai_sysclk, 648 .set_sysclk = wm8750_set_dai_sysclk,
650}; 649};
651 650
652struct snd_soc_dai wm8750_dai = { 651static struct snd_soc_dai_driver wm8750_dai = {
653 .name = "WM8750", 652 .name = "wm8750-hifi",
654 .playback = { 653 .playback = {
655 .stream_name = "Playback", 654 .stream_name = "Playback",
656 .channels_min = 1, 655 .channels_min = 1,
@@ -665,21 +664,15 @@ struct snd_soc_dai wm8750_dai = {
665 .formats = WM8750_FORMATS,}, 664 .formats = WM8750_FORMATS,},
666 .ops = &wm8750_dai_ops, 665 .ops = &wm8750_dai_ops,
667}; 666};
668EXPORT_SYMBOL_GPL(wm8750_dai);
669 667
670static int wm8750_suspend(struct platform_device *pdev, pm_message_t state) 668static int wm8750_suspend(struct snd_soc_codec *codec, pm_message_t state)
671{ 669{
672 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
673 struct snd_soc_codec *codec = socdev->card->codec;
674
675 wm8750_set_bias_level(codec, SND_SOC_BIAS_OFF); 670 wm8750_set_bias_level(codec, SND_SOC_BIAS_OFF);
676 return 0; 671 return 0;
677} 672}
678 673
679static int wm8750_resume(struct platform_device *pdev) 674static int wm8750_resume(struct snd_soc_codec *codec)
680{ 675{
681 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
682 struct snd_soc_codec *codec = socdev->card->codec;
683 int i; 676 int i;
684 u8 data[2]; 677 u8 data[2];
685 u16 *cache = codec->reg_cache; 678 u16 *cache = codec->reg_cache;
@@ -698,100 +691,21 @@ static int wm8750_resume(struct platform_device *pdev)
698 return 0; 691 return 0;
699} 692}
700 693
701static struct snd_soc_codec *wm8750_codec; 694static int wm8750_probe(struct snd_soc_codec *codec)
702
703static int wm8750_probe(struct platform_device *pdev)
704{
705 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
706 struct snd_soc_codec *codec;
707 int ret = 0;
708
709 if (!wm8750_codec) {
710 dev_err(&pdev->dev, "WM8750 codec not yet registered\n");
711 return -EINVAL;
712 }
713
714 socdev->card->codec = wm8750_codec;
715 codec = wm8750_codec;
716
717 /* register pcms */
718 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1);
719 if (ret < 0) {
720 printk(KERN_ERR "wm8750: failed to create pcms\n");
721 goto err;
722 }
723
724 snd_soc_add_controls(codec, wm8750_snd_controls,
725 ARRAY_SIZE(wm8750_snd_controls));
726 wm8750_add_widgets(codec);
727
728 return 0;
729
730err:
731 return ret;
732}
733
734/* power down chip */
735static int wm8750_remove(struct platform_device *pdev)
736{
737 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
738
739 snd_soc_free_pcms(socdev);
740 snd_soc_dapm_free(socdev);
741
742 return 0;
743}
744
745struct snd_soc_codec_device soc_codec_dev_wm8750 = {
746 .probe = wm8750_probe,
747 .remove = wm8750_remove,
748 .suspend = wm8750_suspend,
749 .resume = wm8750_resume,
750};
751EXPORT_SYMBOL_GPL(soc_codec_dev_wm8750);
752
753/*
754 * initialise the WM8750 driver
755 * register the mixer and dsp interfaces with the kernel
756 */
757static int wm8750_register(struct wm8750_priv *wm8750,
758 enum snd_soc_control_type control)
759{ 695{
760 struct snd_soc_codec *codec = &wm8750->codec; 696 struct wm8750_priv *wm8750 = snd_soc_codec_get_drvdata(codec);
761 int reg, ret = 0; 697 int reg, ret;
762
763 if (wm8750_codec) {
764 dev_err(codec->dev, "Multiple WM8750 devices not supported\n");
765 ret = -EINVAL;
766 goto err;
767 }
768
769 mutex_init(&codec->mutex);
770 INIT_LIST_HEAD(&codec->dapm_widgets);
771 INIT_LIST_HEAD(&codec->dapm_paths);
772
773 codec->name = "WM8750";
774 codec->owner = THIS_MODULE;
775 codec->bias_level = SND_SOC_BIAS_STANDBY;
776 codec->set_bias_level = wm8750_set_bias_level;
777 codec->dai = &wm8750_dai;
778 codec->num_dai = 1;
779 codec->reg_cache_size = ARRAY_SIZE(wm8750->reg_cache) + 1;
780 codec->reg_cache = &wm8750->reg_cache;
781 snd_soc_codec_set_drvdata(codec, wm8750);
782
783 memcpy(codec->reg_cache, wm8750_reg, sizeof(wm8750->reg_cache));
784 698
785 ret = snd_soc_codec_set_cache_io(codec, 7, 9, control); 699 ret = snd_soc_codec_set_cache_io(codec, 7, 9, wm8750->control_type);
786 if (ret < 0) { 700 if (ret < 0) {
787 printk(KERN_ERR "wm8750: failed to set cache I/O: %d\n", ret); 701 printk(KERN_ERR "wm8750: failed to set cache I/O: %d\n", ret);
788 goto err; 702 return ret;
789 } 703 }
790 704
791 ret = wm8750_reset(codec); 705 ret = wm8750_reset(codec);
792 if (ret < 0) { 706 if (ret < 0) {
793 printk(KERN_ERR "wm8750: failed to reset: %d\n", ret); 707 printk(KERN_ERR "wm8750: failed to reset: %d\n", ret);
794 goto err; 708 return ret;
795 } 709 }
796 710
797 /* charge output caps */ 711 /* charge output caps */
@@ -815,150 +729,130 @@ static int wm8750_register(struct wm8750_priv *wm8750,
815 reg = snd_soc_read(codec, WM8750_RINVOL); 729 reg = snd_soc_read(codec, WM8750_RINVOL);
816 snd_soc_write(codec, WM8750_RINVOL, reg | 0x0100); 730 snd_soc_write(codec, WM8750_RINVOL, reg | 0x0100);
817 731
818 wm8750_codec = codec; 732 snd_soc_add_controls(codec, wm8750_snd_controls,
819 733 ARRAY_SIZE(wm8750_snd_controls));
820 ret = snd_soc_register_codec(codec); 734 wm8750_add_widgets(codec);
821 if (ret != 0) {
822 dev_err(codec->dev, "Failed to register codec: %d\n", ret);
823 goto err;
824 }
825
826 ret = snd_soc_register_dais(&wm8750_dai, 1);
827 if (ret != 0) {
828 dev_err(codec->dev, "Failed to register DAIs: %d\n", ret);
829 goto err_codec;
830 }
831
832 return 0;
833
834err_codec:
835 snd_soc_unregister_codec(codec);
836err:
837 kfree(wm8750);
838 return ret; 735 return ret;
839} 736}
840 737
841static void wm8750_unregister(struct wm8750_priv *wm8750) 738static int wm8750_remove(struct snd_soc_codec *codec)
842{ 739{
843 wm8750_set_bias_level(&wm8750->codec, SND_SOC_BIAS_OFF); 740 wm8750_set_bias_level(codec, SND_SOC_BIAS_OFF);
844 snd_soc_unregister_dais(&wm8750_dai, 1); 741 return 0;
845 snd_soc_unregister_codec(&wm8750->codec);
846 kfree(wm8750);
847 wm8750_codec = NULL;
848} 742}
849 743
850#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) 744static struct snd_soc_codec_driver soc_codec_dev_wm8750 = {
851 745 .probe = wm8750_probe,
852/* 746 .remove = wm8750_remove,
853 * WM8750 2 wire address is determined by GPIO5 747 .suspend = wm8750_suspend,
854 * state during powerup. 748 .resume = wm8750_resume,
855 * low = 0x1a 749 .set_bias_level = wm8750_set_bias_level,
856 * high = 0x1b 750 .reg_cache_size = ARRAY_SIZE(wm8750_reg),
857 */ 751 .reg_word_size = sizeof(u16),
752 .reg_cache_default = wm8750_reg,
753};
858 754
859static int wm8750_i2c_probe(struct i2c_client *i2c, 755#if defined(CONFIG_SPI_MASTER)
860 const struct i2c_device_id *id) 756static int __devinit wm8750_spi_probe(struct spi_device *spi)
861{ 757{
862 struct snd_soc_codec *codec;
863 struct wm8750_priv *wm8750; 758 struct wm8750_priv *wm8750;
759 int ret;
864 760
865 wm8750 = kzalloc(sizeof(struct wm8750_priv), GFP_KERNEL); 761 wm8750 = kzalloc(sizeof(struct wm8750_priv), GFP_KERNEL);
866 if (wm8750 == NULL) 762 if (wm8750 == NULL)
867 return -ENOMEM; 763 return -ENOMEM;
868 764
869 codec = &wm8750->codec; 765 wm8750->control_type = SND_SOC_SPI;
870 codec->control_data = i2c; 766 spi_set_drvdata(spi, wm8750);
871 i2c_set_clientdata(i2c, wm8750);
872
873 codec->dev = &i2c->dev;
874 767
875 return wm8750_register(wm8750, SND_SOC_I2C); 768 ret = snd_soc_register_codec(&spi->dev,
769 &soc_codec_dev_wm8750, &wm8750_dai, 1);
770 if (ret < 0)
771 kfree(wm8750);
772 return ret;
876} 773}
877 774
878static int wm8750_i2c_remove(struct i2c_client *client) 775static int __devexit wm8750_spi_remove(struct spi_device *spi)
879{ 776{
880 struct wm8750_priv *wm8750 = i2c_get_clientdata(client); 777 snd_soc_unregister_codec(&spi->dev);
881 wm8750_unregister(wm8750); 778 kfree(spi_get_drvdata(spi));
882 return 0; 779 return 0;
883} 780}
884 781
885static const struct i2c_device_id wm8750_i2c_id[] = { 782static struct spi_driver wm8750_spi_driver = {
886 { "wm8750", 0 },
887 { "wm8987", 0 }, /* WM8987 is register compatible with WM8750 */
888 { }
889};
890MODULE_DEVICE_TABLE(i2c, wm8750_i2c_id);
891
892static struct i2c_driver wm8750_i2c_driver = {
893 .driver = { 783 .driver = {
894 .name = "WM8750 I2C Codec", 784 .name = "wm8750-codec",
895 .owner = THIS_MODULE, 785 .owner = THIS_MODULE,
896 }, 786 },
897 .probe = wm8750_i2c_probe, 787 .probe = wm8750_spi_probe,
898 .remove = wm8750_i2c_remove, 788 .remove = __devexit_p(wm8750_spi_remove),
899 .id_table = wm8750_i2c_id,
900}; 789};
901#endif 790#endif /* CONFIG_SPI_MASTER */
902 791
903#if defined(CONFIG_SPI_MASTER) 792#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
904static int __devinit wm8750_spi_probe(struct spi_device *spi) 793static __devinit int wm8750_i2c_probe(struct i2c_client *i2c,
794 const struct i2c_device_id *id)
905{ 795{
906 struct snd_soc_codec *codec;
907 struct wm8750_priv *wm8750; 796 struct wm8750_priv *wm8750;
797 int ret;
908 798
909 wm8750 = kzalloc(sizeof(struct wm8750_priv), GFP_KERNEL); 799 wm8750 = kzalloc(sizeof(struct wm8750_priv), GFP_KERNEL);
910 if (wm8750 == NULL) 800 if (wm8750 == NULL)
911 return -ENOMEM; 801 return -ENOMEM;
912 802
913 codec = &wm8750->codec; 803 i2c_set_clientdata(i2c, wm8750);
914 codec->control_data = spi; 804 wm8750->control_type = SND_SOC_I2C;
915 codec->dev = &spi->dev;
916
917 dev_set_drvdata(&spi->dev, wm8750);
918 805
919 return wm8750_register(wm8750, SND_SOC_SPI); 806 ret = snd_soc_register_codec(&i2c->dev,
807 &soc_codec_dev_wm8750, &wm8750_dai, 1);
808 if (ret < 0)
809 kfree(wm8750);
810 return ret;
920} 811}
921 812
922static int __devexit wm8750_spi_remove(struct spi_device *spi) 813static __devexit int wm8750_i2c_remove(struct i2c_client *client)
923{ 814{
924 struct wm8750_priv *wm8750 = dev_get_drvdata(&spi->dev); 815 snd_soc_unregister_codec(&client->dev);
925 wm8750_unregister(wm8750); 816 kfree(i2c_get_clientdata(client));
926 return 0; 817 return 0;
927} 818}
928 819
929static const struct spi_device_id wm8750_spi_id[] = { 820static const struct i2c_device_id wm8750_i2c_id[] = {
930 { "wm8750", 0 }, 821 { "wm8750", 0 },
931 { "wm8987", 0 }, 822 { "wm8987", 0 },
932 { } 823 { }
933}; 824};
934MODULE_DEVICE_TABLE(spi, wm8750_spi_id); 825MODULE_DEVICE_TABLE(i2c, wm8750_i2c_id);
935 826
936static struct spi_driver wm8750_spi_driver = { 827static struct i2c_driver wm8750_i2c_driver = {
937 .driver = { 828 .driver = {
938 .name = "WM8750 SPI Codec", 829 .name = "wm8750-codec",
939 .bus = &spi_bus_type, 830 .owner = THIS_MODULE,
940 .owner = THIS_MODULE,
941 }, 831 },
942 .probe = wm8750_spi_probe, 832 .probe = wm8750_i2c_probe,
943 .remove = __devexit_p(wm8750_spi_remove), 833 .remove = __devexit_p(wm8750_i2c_remove),
944 .id_table = wm8750_spi_id, 834 .id_table = wm8750_i2c_id,
945}; 835};
946#endif 836#endif
947 837
948static int __init wm8750_modinit(void) 838static int __init wm8750_modinit(void)
949{ 839{
950 int ret; 840 int ret = 0;
951#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) 841#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
952 ret = i2c_add_driver(&wm8750_i2c_driver); 842 ret = i2c_add_driver(&wm8750_i2c_driver);
953 if (ret != 0) 843 if (ret != 0) {
954 pr_err("Failed to register WM8750 I2C driver: %d\n", ret); 844 printk(KERN_ERR "Failed to register wm8750 I2C driver: %d\n",
845 ret);
846 }
955#endif 847#endif
956#if defined(CONFIG_SPI_MASTER) 848#if defined(CONFIG_SPI_MASTER)
957 ret = spi_register_driver(&wm8750_spi_driver); 849 ret = spi_register_driver(&wm8750_spi_driver);
958 if (ret != 0) 850 if (ret != 0) {
959 pr_err("Failed to register WM8750 SPI driver: %d\n", ret); 851 printk(KERN_ERR "Failed to register wm8750 SPI driver: %d\n",
852 ret);
853 }
960#endif 854#endif
961 return 0; 855 return ret;
962} 856}
963module_init(wm8750_modinit); 857module_init(wm8750_modinit);
964 858