diff options
Diffstat (limited to 'sound/soc/codecs/wm8750.c')
-rw-r--r-- | sound/soc/codecs/wm8750.c | 264 |
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 */ |
53 | struct wm8750_priv { | 53 | struct 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 | ||
652 | struct snd_soc_dai wm8750_dai = { | 651 | static 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 | }; |
668 | EXPORT_SYMBOL_GPL(wm8750_dai); | ||
669 | 667 | ||
670 | static int wm8750_suspend(struct platform_device *pdev, pm_message_t state) | 668 | static 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 | ||
679 | static int wm8750_resume(struct platform_device *pdev) | 674 | static 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 | ||
701 | static struct snd_soc_codec *wm8750_codec; | 694 | static int wm8750_probe(struct snd_soc_codec *codec) |
702 | |||
703 | static 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 | |||
730 | err: | ||
731 | return ret; | ||
732 | } | ||
733 | |||
734 | /* power down chip */ | ||
735 | static 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 | |||
745 | struct 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 | }; | ||
751 | EXPORT_SYMBOL_GPL(soc_codec_dev_wm8750); | ||
752 | |||
753 | /* | ||
754 | * initialise the WM8750 driver | ||
755 | * register the mixer and dsp interfaces with the kernel | ||
756 | */ | ||
757 | static 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 | |||
834 | err_codec: | ||
835 | snd_soc_unregister_codec(codec); | ||
836 | err: | ||
837 | kfree(wm8750); | ||
838 | return ret; | 735 | return ret; |
839 | } | 736 | } |
840 | 737 | ||
841 | static void wm8750_unregister(struct wm8750_priv *wm8750) | 738 | static 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) | 744 | static 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 | ||
859 | static int wm8750_i2c_probe(struct i2c_client *i2c, | 755 | #if defined(CONFIG_SPI_MASTER) |
860 | const struct i2c_device_id *id) | 756 | static 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 | ||
878 | static int wm8750_i2c_remove(struct i2c_client *client) | 775 | static 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 | ||
885 | static const struct i2c_device_id wm8750_i2c_id[] = { | 782 | static struct spi_driver wm8750_spi_driver = { |
886 | { "wm8750", 0 }, | ||
887 | { "wm8987", 0 }, /* WM8987 is register compatible with WM8750 */ | ||
888 | { } | ||
889 | }; | ||
890 | MODULE_DEVICE_TABLE(i2c, wm8750_i2c_id); | ||
891 | |||
892 | static 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) |
904 | static int __devinit wm8750_spi_probe(struct spi_device *spi) | 793 | static __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 | ||
922 | static int __devexit wm8750_spi_remove(struct spi_device *spi) | 813 | static __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 | ||
929 | static const struct spi_device_id wm8750_spi_id[] = { | 820 | static const struct i2c_device_id wm8750_i2c_id[] = { |
930 | { "wm8750", 0 }, | 821 | { "wm8750", 0 }, |
931 | { "wm8987", 0 }, | 822 | { "wm8987", 0 }, |
932 | { } | 823 | { } |
933 | }; | 824 | }; |
934 | MODULE_DEVICE_TABLE(spi, wm8750_spi_id); | 825 | MODULE_DEVICE_TABLE(i2c, wm8750_i2c_id); |
935 | 826 | ||
936 | static struct spi_driver wm8750_spi_driver = { | 827 | static 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 | ||
948 | static int __init wm8750_modinit(void) | 838 | static 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 | } |
963 | module_init(wm8750_modinit); | 857 | module_init(wm8750_modinit); |
964 | 858 | ||