aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm8955.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm8955.c')
-rw-r--r--sound/soc/codecs/wm8955.c238
1 files changed, 79 insertions, 159 deletions
diff --git a/sound/soc/codecs/wm8955.c b/sound/soc/codecs/wm8955.c
index 5f025593d84d..3c7198779c31 100644
--- a/sound/soc/codecs/wm8955.c
+++ b/sound/soc/codecs/wm8955.c
@@ -23,16 +23,12 @@
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#include <sound/wm8955.h> 28#include <sound/wm8955.h>
30 29
31#include "wm8955.h" 30#include "wm8955.h"
32 31
33static struct snd_soc_codec *wm8955_codec;
34struct snd_soc_codec_device soc_codec_dev_wm8955;
35
36#define WM8955_NUM_SUPPLIES 4 32#define WM8955_NUM_SUPPLIES 4
37static const char *wm8955_supply_names[WM8955_NUM_SUPPLIES] = { 33static const char *wm8955_supply_names[WM8955_NUM_SUPPLIES] = {
38 "DCVDD", 34 "DCVDD",
@@ -43,8 +39,7 @@ static const char *wm8955_supply_names[WM8955_NUM_SUPPLIES] = {
43 39
44/* codec private data */ 40/* codec private data */
45struct wm8955_priv { 41struct wm8955_priv {
46 struct snd_soc_codec codec; 42 enum snd_soc_control_type control_type;
47 u16 reg_cache[WM8955_MAX_REGISTER + 1];
48 43
49 unsigned int mclk_rate; 44 unsigned int mclk_rate;
50 45
@@ -52,8 +47,6 @@ struct wm8955_priv {
52 int fs; 47 int fs;
53 48
54 struct regulator_bulk_data supplies[WM8955_NUM_SUPPLIES]; 49 struct regulator_bulk_data supplies[WM8955_NUM_SUPPLIES];
55
56 struct wm8955_pdata *pdata;
57}; 50};
58 51
59static const u16 wm8955_reg[WM8955_MAX_REGISTER + 1] = { 52static const u16 wm8955_reg[WM8955_MAX_REGISTER + 1] = {
@@ -183,7 +176,7 @@ static int wm8995_pll_factors(struct device *dev,
183 return 0; 176 return 0;
184} 177}
185 178
186/* Lookup table specifiying SRATE (table 25 in datasheet); some of the 179/* Lookup table specifying SRATE (table 25 in datasheet); some of the
187 * output frequencies have been rounded to the standard frequencies 180 * output frequencies have been rounded to the standard frequencies
188 * they are intended to match where the error is slight. */ 181 * they are intended to match where the error is slight. */
189static struct { 182static struct {
@@ -384,7 +377,8 @@ static int wm8955_get_deemph(struct snd_kcontrol *kcontrol,
384 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); 377 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
385 struct wm8955_priv *wm8955 = snd_soc_codec_get_drvdata(codec); 378 struct wm8955_priv *wm8955 = snd_soc_codec_get_drvdata(codec);
386 379
387 return wm8955->deemph; 380 ucontrol->value.enumerated.item[0] = wm8955->deemph;
381 return 0;
388} 382}
389 383
390static int wm8955_put_deemph(struct snd_kcontrol *kcontrol, 384static int wm8955_put_deemph(struct snd_kcontrol *kcontrol,
@@ -581,13 +575,14 @@ static const struct snd_soc_dapm_route wm8955_intercon[] = {
581 575
582static int wm8955_add_widgets(struct snd_soc_codec *codec) 576static int wm8955_add_widgets(struct snd_soc_codec *codec)
583{ 577{
578 struct snd_soc_dapm_context *dapm = &codec->dapm;
579
584 snd_soc_add_controls(codec, wm8955_snd_controls, 580 snd_soc_add_controls(codec, wm8955_snd_controls,
585 ARRAY_SIZE(wm8955_snd_controls)); 581 ARRAY_SIZE(wm8955_snd_controls));
586 582
587 snd_soc_dapm_new_controls(codec, wm8955_dapm_widgets, 583 snd_soc_dapm_new_controls(dapm, wm8955_dapm_widgets,
588 ARRAY_SIZE(wm8955_dapm_widgets)); 584 ARRAY_SIZE(wm8955_dapm_widgets));
589 585 snd_soc_dapm_add_routes(dapm, wm8955_intercon,
590 snd_soc_dapm_add_routes(codec, wm8955_intercon,
591 ARRAY_SIZE(wm8955_intercon)); 586 ARRAY_SIZE(wm8955_intercon));
592 587
593 return 0; 588 return 0;
@@ -771,6 +766,7 @@ static int wm8955_set_bias_level(struct snd_soc_codec *codec,
771 enum snd_soc_bias_level level) 766 enum snd_soc_bias_level level)
772{ 767{
773 struct wm8955_priv *wm8955 = snd_soc_codec_get_drvdata(codec); 768 struct wm8955_priv *wm8955 = snd_soc_codec_get_drvdata(codec);
769 u16 *reg_cache = codec->reg_cache;
774 int ret, i; 770 int ret, i;
775 771
776 switch (level) { 772 switch (level) {
@@ -790,7 +786,7 @@ static int wm8955_set_bias_level(struct snd_soc_codec *codec,
790 break; 786 break;
791 787
792 case SND_SOC_BIAS_STANDBY: 788 case SND_SOC_BIAS_STANDBY:
793 if (codec->bias_level == SND_SOC_BIAS_OFF) { 789 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
794 ret = regulator_bulk_enable(ARRAY_SIZE(wm8955->supplies), 790 ret = regulator_bulk_enable(ARRAY_SIZE(wm8955->supplies),
795 wm8955->supplies); 791 wm8955->supplies);
796 if (ret != 0) { 792 if (ret != 0) {
@@ -803,14 +799,14 @@ static int wm8955_set_bias_level(struct snd_soc_codec *codec,
803 /* Sync back cached values if they're 799 /* Sync back cached values if they're
804 * different from the hardware default. 800 * different from the hardware default.
805 */ 801 */
806 for (i = 0; i < ARRAY_SIZE(wm8955->reg_cache); i++) { 802 for (i = 0; i < codec->driver->reg_cache_size; i++) {
807 if (i == WM8955_RESET) 803 if (i == WM8955_RESET)
808 continue; 804 continue;
809 805
810 if (wm8955->reg_cache[i] == wm8955_reg[i]) 806 if (reg_cache[i] == wm8955_reg[i])
811 continue; 807 continue;
812 808
813 snd_soc_write(codec, i, wm8955->reg_cache[i]); 809 snd_soc_write(codec, i, reg_cache[i]);
814 } 810 }
815 811
816 /* Enable VREF and VMID */ 812 /* Enable VREF and VMID */
@@ -854,7 +850,7 @@ static int wm8955_set_bias_level(struct snd_soc_codec *codec,
854 wm8955->supplies); 850 wm8955->supplies);
855 break; 851 break;
856 } 852 }
857 codec->bias_level = level; 853 codec->dapm.bias_level = level;
858 return 0; 854 return 0;
859} 855}
860 856
@@ -870,8 +866,8 @@ static struct snd_soc_dai_ops wm8955_dai_ops = {
870 .digital_mute = wm8955_digital_mute, 866 .digital_mute = wm8955_digital_mute,
871}; 867};
872 868
873struct snd_soc_dai wm8955_dai = { 869static struct snd_soc_dai_driver wm8955_dai = {
874 .name = "WM8955", 870 .name = "wm8955-hifi",
875 .playback = { 871 .playback = {
876 .stream_name = "Playback", 872 .stream_name = "Playback",
877 .channels_min = 2, 873 .channels_min = 2,
@@ -881,24 +877,17 @@ struct snd_soc_dai wm8955_dai = {
881 }, 877 },
882 .ops = &wm8955_dai_ops, 878 .ops = &wm8955_dai_ops,
883}; 879};
884EXPORT_SYMBOL_GPL(wm8955_dai);
885 880
886#ifdef CONFIG_PM 881#ifdef CONFIG_PM
887static int wm8955_suspend(struct platform_device *pdev, pm_message_t state) 882static int wm8955_suspend(struct snd_soc_codec *codec, pm_message_t state)
888{ 883{
889 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
890 struct snd_soc_codec *codec = socdev->card->codec;
891
892 wm8955_set_bias_level(codec, SND_SOC_BIAS_OFF); 884 wm8955_set_bias_level(codec, SND_SOC_BIAS_OFF);
893 885
894 return 0; 886 return 0;
895} 887}
896 888
897static int wm8955_resume(struct platform_device *pdev) 889static int wm8955_resume(struct snd_soc_codec *codec)
898{ 890{
899 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
900 struct snd_soc_codec *codec = socdev->card->codec;
901
902 wm8955_set_bias_level(codec, SND_SOC_BIAS_STANDBY); 891 wm8955_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
903 892
904 return 0; 893 return 0;
@@ -908,86 +897,17 @@ static int wm8955_resume(struct platform_device *pdev)
908#define wm8955_resume NULL 897#define wm8955_resume NULL
909#endif 898#endif
910 899
911static int wm8955_probe(struct platform_device *pdev) 900static int wm8955_probe(struct snd_soc_codec *codec)
912{ 901{
913 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 902 struct wm8955_priv *wm8955 = snd_soc_codec_get_drvdata(codec);
914 struct snd_soc_codec *codec; 903 struct wm8955_pdata *pdata = dev_get_platdata(codec->dev);
915 int ret = 0; 904 u16 *reg_cache = codec->reg_cache;
916 905 int ret, i;
917 if (wm8955_codec == NULL) {
918 dev_err(&pdev->dev, "Codec device not registered\n");
919 return -ENODEV;
920 }
921
922 socdev->card->codec = wm8955_codec;
923 codec = wm8955_codec;
924
925 /* register pcms */
926 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1);
927 if (ret < 0) {
928 dev_err(codec->dev, "failed to create pcms: %d\n", ret);
929 goto pcm_err;
930 }
931
932 wm8955_add_widgets(codec);
933
934 return ret;
935
936pcm_err:
937 return ret;
938}
939
940static int wm8955_remove(struct platform_device *pdev)
941{
942 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
943
944 snd_soc_free_pcms(socdev);
945 snd_soc_dapm_free(socdev);
946
947 return 0;
948}
949
950struct snd_soc_codec_device soc_codec_dev_wm8955 = {
951 .probe = wm8955_probe,
952 .remove = wm8955_remove,
953 .suspend = wm8955_suspend,
954 .resume = wm8955_resume,
955};
956EXPORT_SYMBOL_GPL(soc_codec_dev_wm8955);
957
958static int wm8955_register(struct wm8955_priv *wm8955,
959 enum snd_soc_control_type control)
960{
961 int ret;
962 struct snd_soc_codec *codec = &wm8955->codec;
963 int i;
964
965 if (wm8955_codec) {
966 dev_err(codec->dev, "Another WM8955 is registered\n");
967 ret = -EINVAL;
968 goto err;
969 }
970
971 mutex_init(&codec->mutex);
972 INIT_LIST_HEAD(&codec->dapm_widgets);
973 INIT_LIST_HEAD(&codec->dapm_paths);
974
975 snd_soc_codec_set_drvdata(codec, wm8955);
976 codec->name = "WM8955";
977 codec->owner = THIS_MODULE;
978 codec->bias_level = SND_SOC_BIAS_OFF;
979 codec->set_bias_level = wm8955_set_bias_level;
980 codec->dai = &wm8955_dai;
981 codec->num_dai = 1;
982 codec->reg_cache_size = WM8955_MAX_REGISTER;
983 codec->reg_cache = &wm8955->reg_cache;
984
985 memcpy(codec->reg_cache, wm8955_reg, sizeof(wm8955_reg));
986 906
987 ret = snd_soc_codec_set_cache_io(codec, 7, 9, control); 907 ret = snd_soc_codec_set_cache_io(codec, 7, 9, wm8955->control_type);
988 if (ret != 0) { 908 if (ret != 0) {
989 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret); 909 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
990 goto err; 910 return ret;
991 } 911 }
992 912
993 for (i = 0; i < ARRAY_SIZE(wm8955->supplies); i++) 913 for (i = 0; i < ARRAY_SIZE(wm8955->supplies); i++)
@@ -997,7 +917,7 @@ static int wm8955_register(struct wm8955_priv *wm8955,
997 wm8955->supplies); 917 wm8955->supplies);
998 if (ret != 0) { 918 if (ret != 0) {
999 dev_err(codec->dev, "Failed to request supplies: %d\n", ret); 919 dev_err(codec->dev, "Failed to request supplies: %d\n", ret);
1000 goto err; 920 return ret;
1001 } 921 }
1002 922
1003 ret = regulator_bulk_enable(ARRAY_SIZE(wm8955->supplies), 923 ret = regulator_bulk_enable(ARRAY_SIZE(wm8955->supplies),
@@ -1013,28 +933,37 @@ static int wm8955_register(struct wm8955_priv *wm8955,
1013 goto err_enable; 933 goto err_enable;
1014 } 934 }
1015 935
1016 wm8955_dai.dev = codec->dev;
1017
1018 /* Change some default settings - latch VU and enable ZC */ 936 /* Change some default settings - latch VU and enable ZC */
1019 wm8955->reg_cache[WM8955_LEFT_DAC_VOLUME] |= WM8955_LDVU; 937 snd_soc_update_bits(codec, WM8955_LEFT_DAC_VOLUME,
1020 wm8955->reg_cache[WM8955_RIGHT_DAC_VOLUME] |= WM8955_RDVU; 938 WM8955_LDVU, WM8955_LDVU);
1021 wm8955->reg_cache[WM8955_LOUT1_VOLUME] |= WM8955_LO1VU | WM8955_LO1ZC; 939 snd_soc_update_bits(codec, WM8955_RIGHT_DAC_VOLUME,
1022 wm8955->reg_cache[WM8955_ROUT1_VOLUME] |= WM8955_RO1VU | WM8955_RO1ZC; 940 WM8955_RDVU, WM8955_RDVU);
1023 wm8955->reg_cache[WM8955_LOUT2_VOLUME] |= WM8955_LO2VU | WM8955_LO2ZC; 941 snd_soc_update_bits(codec, WM8955_LOUT1_VOLUME,
1024 wm8955->reg_cache[WM8955_ROUT2_VOLUME] |= WM8955_RO2VU | WM8955_RO2ZC; 942 WM8955_LO1VU | WM8955_LO1ZC,
1025 wm8955->reg_cache[WM8955_MONOOUT_VOLUME] |= WM8955_MOZC; 943 WM8955_LO1VU | WM8955_LO1ZC);
944 snd_soc_update_bits(codec, WM8955_ROUT1_VOLUME,
945 WM8955_RO1VU | WM8955_RO1ZC,
946 WM8955_RO1VU | WM8955_RO1ZC);
947 snd_soc_update_bits(codec, WM8955_LOUT2_VOLUME,
948 WM8955_LO2VU | WM8955_LO2ZC,
949 WM8955_LO2VU | WM8955_LO2ZC);
950 snd_soc_update_bits(codec, WM8955_ROUT2_VOLUME,
951 WM8955_RO2VU | WM8955_RO2ZC,
952 WM8955_RO2VU | WM8955_RO2ZC);
953 snd_soc_update_bits(codec, WM8955_MONOOUT_VOLUME,
954 WM8955_MOZC, WM8955_MOZC);
1026 955
1027 /* Also enable adaptive bass boost by default */ 956 /* Also enable adaptive bass boost by default */
1028 wm8955->reg_cache[WM8955_BASS_CONTROL] |= WM8955_BB; 957 snd_soc_update_bits(codec, WM8955_BASS_CONTROL, WM8955_BB, WM8955_BB);
1029 958
1030 /* Set platform data values */ 959 /* Set platform data values */
1031 if (wm8955->pdata) { 960 if (pdata) {
1032 if (wm8955->pdata->out2_speaker) 961 if (pdata->out2_speaker)
1033 wm8955->reg_cache[WM8955_ADDITIONAL_CONTROL_2] 962 reg_cache[WM8955_ADDITIONAL_CONTROL_2]
1034 |= WM8955_ROUT2INV; 963 |= WM8955_ROUT2INV;
1035 964
1036 if (wm8955->pdata->monoin_diff) 965 if (pdata->monoin_diff)
1037 wm8955->reg_cache[WM8955_MONO_OUT_MIX_1] 966 reg_cache[WM8955_MONO_OUT_MIX_1]
1038 |= WM8955_DMEN; 967 |= WM8955_DMEN;
1039 } 968 }
1040 969
@@ -1043,70 +972,61 @@ static int wm8955_register(struct wm8955_priv *wm8955,
1043 /* Bias level configuration will have done an extra enable */ 972 /* Bias level configuration will have done an extra enable */
1044 regulator_bulk_disable(ARRAY_SIZE(wm8955->supplies), wm8955->supplies); 973 regulator_bulk_disable(ARRAY_SIZE(wm8955->supplies), wm8955->supplies);
1045 974
1046 wm8955_codec = codec; 975 wm8955_add_widgets(codec);
1047
1048 ret = snd_soc_register_codec(codec);
1049 if (ret != 0) {
1050 dev_err(codec->dev, "Failed to register codec: %d\n", ret);
1051 goto err_enable;
1052 }
1053
1054 ret = snd_soc_register_dai(&wm8955_dai);
1055 if (ret != 0) {
1056 dev_err(codec->dev, "Failed to register DAI: %d\n", ret);
1057 goto err_codec;
1058 }
1059
1060 return 0; 976 return 0;
1061 977
1062err_codec:
1063 snd_soc_unregister_codec(codec);
1064err_enable: 978err_enable:
1065 regulator_bulk_disable(ARRAY_SIZE(wm8955->supplies), wm8955->supplies); 979 regulator_bulk_disable(ARRAY_SIZE(wm8955->supplies), wm8955->supplies);
1066err_get: 980err_get:
1067 regulator_bulk_free(ARRAY_SIZE(wm8955->supplies), wm8955->supplies); 981 regulator_bulk_free(ARRAY_SIZE(wm8955->supplies), wm8955->supplies);
1068err:
1069 kfree(wm8955);
1070 return ret; 982 return ret;
1071} 983}
1072 984
1073static void wm8955_unregister(struct wm8955_priv *wm8955) 985static int wm8955_remove(struct snd_soc_codec *codec)
1074{ 986{
1075 wm8955_set_bias_level(&wm8955->codec, SND_SOC_BIAS_OFF); 987 struct wm8955_priv *wm8955 = snd_soc_codec_get_drvdata(codec);
988
989 wm8955_set_bias_level(codec, SND_SOC_BIAS_OFF);
1076 regulator_bulk_free(ARRAY_SIZE(wm8955->supplies), wm8955->supplies); 990 regulator_bulk_free(ARRAY_SIZE(wm8955->supplies), wm8955->supplies);
1077 snd_soc_unregister_dai(&wm8955_dai); 991 return 0;
1078 snd_soc_unregister_codec(&wm8955->codec);
1079 kfree(wm8955);
1080 wm8955_codec = NULL;
1081} 992}
1082 993
994static struct snd_soc_codec_driver soc_codec_dev_wm8955 = {
995 .probe = wm8955_probe,
996 .remove = wm8955_remove,
997 .suspend = wm8955_suspend,
998 .resume = wm8955_resume,
999 .set_bias_level = wm8955_set_bias_level,
1000 .reg_cache_size = ARRAY_SIZE(wm8955_reg),
1001 .reg_word_size = sizeof(u16),
1002 .reg_cache_default = wm8955_reg,
1003};
1004
1083#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) 1005#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1084static __devinit int wm8955_i2c_probe(struct i2c_client *i2c, 1006static __devinit int wm8955_i2c_probe(struct i2c_client *i2c,
1085 const struct i2c_device_id *id) 1007 const struct i2c_device_id *id)
1086{ 1008{
1087 struct wm8955_priv *wm8955; 1009 struct wm8955_priv *wm8955;
1088 struct snd_soc_codec *codec; 1010 int ret;
1089 1011
1090 wm8955 = kzalloc(sizeof(struct wm8955_priv), GFP_KERNEL); 1012 wm8955 = kzalloc(sizeof(struct wm8955_priv), GFP_KERNEL);
1091 if (wm8955 == NULL) 1013 if (wm8955 == NULL)
1092 return -ENOMEM; 1014 return -ENOMEM;
1093 1015
1094 codec = &wm8955->codec;
1095 codec->hw_write = (hw_write_t)i2c_master_send;
1096
1097 i2c_set_clientdata(i2c, wm8955); 1016 i2c_set_clientdata(i2c, wm8955);
1098 codec->control_data = i2c; 1017 wm8955->control_type = SND_SOC_I2C;
1099 wm8955->pdata = i2c->dev.platform_data;
1100
1101 codec->dev = &i2c->dev;
1102 1018
1103 return wm8955_register(wm8955, SND_SOC_I2C); 1019 ret = snd_soc_register_codec(&i2c->dev,
1020 &soc_codec_dev_wm8955, &wm8955_dai, 1);
1021 if (ret < 0)
1022 kfree(wm8955);
1023 return ret;
1104} 1024}
1105 1025
1106static __devexit int wm8955_i2c_remove(struct i2c_client *client) 1026static __devexit int wm8955_i2c_remove(struct i2c_client *client)
1107{ 1027{
1108 struct wm8955_priv *wm8955 = i2c_get_clientdata(client); 1028 snd_soc_unregister_codec(&client->dev);
1109 wm8955_unregister(wm8955); 1029 kfree(i2c_get_clientdata(client));
1110 return 0; 1030 return 0;
1111} 1031}
1112 1032
@@ -1118,7 +1038,7 @@ MODULE_DEVICE_TABLE(i2c, wm8955_i2c_id);
1118 1038
1119static struct i2c_driver wm8955_i2c_driver = { 1039static struct i2c_driver wm8955_i2c_driver = {
1120 .driver = { 1040 .driver = {
1121 .name = "wm8955", 1041 .name = "wm8955-codec",
1122 .owner = THIS_MODULE, 1042 .owner = THIS_MODULE,
1123 }, 1043 },
1124 .probe = wm8955_i2c_probe, 1044 .probe = wm8955_i2c_probe,
@@ -1129,7 +1049,7 @@ static struct i2c_driver wm8955_i2c_driver = {
1129 1049
1130static int __init wm8955_modinit(void) 1050static int __init wm8955_modinit(void)
1131{ 1051{
1132 int ret; 1052 int ret = 0;
1133#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) 1053#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1134 ret = i2c_add_driver(&wm8955_i2c_driver); 1054 ret = i2c_add_driver(&wm8955_i2c_driver);
1135 if (ret != 0) { 1055 if (ret != 0) {
@@ -1137,7 +1057,7 @@ static int __init wm8955_modinit(void)
1137 ret); 1057 ret);
1138 } 1058 }
1139#endif 1059#endif
1140 return 0; 1060 return ret;
1141} 1061}
1142module_init(wm8955_modinit); 1062module_init(wm8955_modinit);
1143 1063