diff options
Diffstat (limited to 'sound/soc/codecs/wm8961.c')
-rw-r--r-- | sound/soc/codecs/wm8961.c | 252 |
1 files changed, 80 insertions, 172 deletions
diff --git a/sound/soc/codecs/wm8961.c b/sound/soc/codecs/wm8961.c index 2549d3a297ab..cdee8103d09b 100644 --- a/sound/soc/codecs/wm8961.c +++ b/sound/soc/codecs/wm8961.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 | ||
@@ -288,12 +287,11 @@ static u16 wm8961_reg_defaults[] = { | |||
288 | }; | 287 | }; |
289 | 288 | ||
290 | struct wm8961_priv { | 289 | struct wm8961_priv { |
291 | struct snd_soc_codec codec; | 290 | enum snd_soc_control_type control_type; |
292 | int sysclk; | 291 | int sysclk; |
293 | u16 reg_cache[WM8961_MAX_REGISTER]; | ||
294 | }; | 292 | }; |
295 | 293 | ||
296 | static int wm8961_volatile_register(unsigned int reg) | 294 | static int wm8961_volatile_register(struct snd_soc_codec *codec, unsigned int reg) |
297 | { | 295 | { |
298 | switch (reg) { | 296 | switch (reg) { |
299 | case WM8961_SOFTWARE_RESET: | 297 | case WM8961_SOFTWARE_RESET: |
@@ -711,7 +709,7 @@ static int wm8961_hw_params(struct snd_pcm_substream *substream, | |||
711 | if (fs <= 24000) | 709 | if (fs <= 24000) |
712 | reg |= WM8961_DACSLOPE; | 710 | reg |= WM8961_DACSLOPE; |
713 | else | 711 | else |
714 | reg &= WM8961_DACSLOPE; | 712 | reg &= ~WM8961_DACSLOPE; |
715 | snd_soc_write(codec, WM8961_ADC_DAC_CONTROL_2, reg); | 713 | snd_soc_write(codec, WM8961_ADC_DAC_CONTROL_2, reg); |
716 | 714 | ||
717 | return 0; | 715 | return 0; |
@@ -736,7 +734,7 @@ static int wm8961_set_sysclk(struct snd_soc_dai *dai, int clk_id, | |||
736 | freq /= 2; | 734 | freq /= 2; |
737 | } else { | 735 | } else { |
738 | dev_dbg(codec->dev, "Using MCLK/1 for %dHz MCLK\n", freq); | 736 | dev_dbg(codec->dev, "Using MCLK/1 for %dHz MCLK\n", freq); |
739 | reg &= WM8961_MCLKDIV; | 737 | reg &= ~WM8961_MCLKDIV; |
740 | } | 738 | } |
741 | 739 | ||
742 | snd_soc_write(codec, WM8961_CLOCKING1, reg); | 740 | snd_soc_write(codec, WM8961_CLOCKING1, reg); |
@@ -882,7 +880,7 @@ static int wm8961_set_bias_level(struct snd_soc_codec *codec, | |||
882 | break; | 880 | break; |
883 | 881 | ||
884 | case SND_SOC_BIAS_PREPARE: | 882 | case SND_SOC_BIAS_PREPARE: |
885 | if (codec->bias_level == SND_SOC_BIAS_STANDBY) { | 883 | if (codec->dapm.bias_level == SND_SOC_BIAS_STANDBY) { |
886 | /* Enable bias generation */ | 884 | /* Enable bias generation */ |
887 | reg = snd_soc_read(codec, WM8961_ANTI_POP); | 885 | reg = snd_soc_read(codec, WM8961_ANTI_POP); |
888 | reg |= WM8961_BUFIOEN | WM8961_BUFDCOPEN; | 886 | reg |= WM8961_BUFIOEN | WM8961_BUFDCOPEN; |
@@ -897,7 +895,7 @@ static int wm8961_set_bias_level(struct snd_soc_codec *codec, | |||
897 | break; | 895 | break; |
898 | 896 | ||
899 | case SND_SOC_BIAS_STANDBY: | 897 | case SND_SOC_BIAS_STANDBY: |
900 | if (codec->bias_level == SND_SOC_BIAS_PREPARE) { | 898 | if (codec->dapm.bias_level == SND_SOC_BIAS_PREPARE) { |
901 | /* VREF off */ | 899 | /* VREF off */ |
902 | reg = snd_soc_read(codec, WM8961_PWR_MGMT_1); | 900 | reg = snd_soc_read(codec, WM8961_PWR_MGMT_1); |
903 | reg &= ~WM8961_VREF; | 901 | reg &= ~WM8961_VREF; |
@@ -919,7 +917,7 @@ static int wm8961_set_bias_level(struct snd_soc_codec *codec, | |||
919 | break; | 917 | break; |
920 | } | 918 | } |
921 | 919 | ||
922 | codec->bias_level = level; | 920 | codec->dapm.bias_level = level; |
923 | 921 | ||
924 | return 0; | 922 | return 0; |
925 | } | 923 | } |
@@ -940,8 +938,8 @@ static struct snd_soc_dai_ops wm8961_dai_ops = { | |||
940 | .set_clkdiv = wm8961_set_clkdiv, | 938 | .set_clkdiv = wm8961_set_clkdiv, |
941 | }; | 939 | }; |
942 | 940 | ||
943 | struct snd_soc_dai wm8961_dai = { | 941 | static struct snd_soc_dai_driver wm8961_dai = { |
944 | .name = "WM8961", | 942 | .name = "wm8961-hifi", |
945 | .playback = { | 943 | .playback = { |
946 | .stream_name = "HiFi Playback", | 944 | .stream_name = "HiFi Playback", |
947 | .channels_min = 1, | 945 | .channels_min = 1, |
@@ -956,140 +954,23 @@ struct snd_soc_dai wm8961_dai = { | |||
956 | .formats = WM8961_FORMATS,}, | 954 | .formats = WM8961_FORMATS,}, |
957 | .ops = &wm8961_dai_ops, | 955 | .ops = &wm8961_dai_ops, |
958 | }; | 956 | }; |
959 | EXPORT_SYMBOL_GPL(wm8961_dai); | ||
960 | 957 | ||
961 | 958 | static int wm8961_probe(struct snd_soc_codec *codec) | |
962 | static struct snd_soc_codec *wm8961_codec; | ||
963 | |||
964 | static int wm8961_probe(struct platform_device *pdev) | ||
965 | { | 959 | { |
966 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | 960 | struct snd_soc_dapm_context *dapm = &codec->dapm; |
967 | struct snd_soc_codec *codec; | ||
968 | int ret = 0; | 961 | int ret = 0; |
969 | |||
970 | if (wm8961_codec == NULL) { | ||
971 | dev_err(&pdev->dev, "Codec device not registered\n"); | ||
972 | return -ENODEV; | ||
973 | } | ||
974 | |||
975 | socdev->card->codec = wm8961_codec; | ||
976 | codec = wm8961_codec; | ||
977 | |||
978 | /* register pcms */ | ||
979 | ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); | ||
980 | if (ret < 0) { | ||
981 | dev_err(codec->dev, "failed to create pcms: %d\n", ret); | ||
982 | goto pcm_err; | ||
983 | } | ||
984 | |||
985 | snd_soc_add_controls(codec, wm8961_snd_controls, | ||
986 | ARRAY_SIZE(wm8961_snd_controls)); | ||
987 | snd_soc_dapm_new_controls(codec, wm8961_dapm_widgets, | ||
988 | ARRAY_SIZE(wm8961_dapm_widgets)); | ||
989 | snd_soc_dapm_add_routes(codec, audio_paths, ARRAY_SIZE(audio_paths)); | ||
990 | |||
991 | return ret; | ||
992 | |||
993 | pcm_err: | ||
994 | return ret; | ||
995 | } | ||
996 | |||
997 | static int wm8961_remove(struct platform_device *pdev) | ||
998 | { | ||
999 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
1000 | |||
1001 | snd_soc_free_pcms(socdev); | ||
1002 | snd_soc_dapm_free(socdev); | ||
1003 | |||
1004 | return 0; | ||
1005 | } | ||
1006 | |||
1007 | #ifdef CONFIG_PM | ||
1008 | static int wm8961_suspend(struct platform_device *pdev, pm_message_t state) | ||
1009 | { | ||
1010 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
1011 | struct snd_soc_codec *codec = socdev->card->codec; | ||
1012 | |||
1013 | wm8961_set_bias_level(codec, SND_SOC_BIAS_OFF); | ||
1014 | |||
1015 | return 0; | ||
1016 | } | ||
1017 | |||
1018 | static int wm8961_resume(struct platform_device *pdev) | ||
1019 | { | ||
1020 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
1021 | struct snd_soc_codec *codec = socdev->card->codec; | ||
1022 | u16 *reg_cache = codec->reg_cache; | ||
1023 | int i; | ||
1024 | |||
1025 | for (i = 0; i < codec->reg_cache_size; i++) { | ||
1026 | if (reg_cache[i] == wm8961_reg_defaults[i]) | ||
1027 | continue; | ||
1028 | |||
1029 | if (i == WM8961_SOFTWARE_RESET) | ||
1030 | continue; | ||
1031 | |||
1032 | snd_soc_write(codec, i, reg_cache[i]); | ||
1033 | } | ||
1034 | |||
1035 | wm8961_set_bias_level(codec, SND_SOC_BIAS_STANDBY); | ||
1036 | |||
1037 | return 0; | ||
1038 | } | ||
1039 | #else | ||
1040 | #define wm8961_suspend NULL | ||
1041 | #define wm8961_resume NULL | ||
1042 | #endif | ||
1043 | |||
1044 | struct snd_soc_codec_device soc_codec_dev_wm8961 = { | ||
1045 | .probe = wm8961_probe, | ||
1046 | .remove = wm8961_remove, | ||
1047 | .suspend = wm8961_suspend, | ||
1048 | .resume = wm8961_resume, | ||
1049 | }; | ||
1050 | EXPORT_SYMBOL_GPL(soc_codec_dev_wm8961); | ||
1051 | |||
1052 | static int wm8961_register(struct wm8961_priv *wm8961) | ||
1053 | { | ||
1054 | struct snd_soc_codec *codec = &wm8961->codec; | ||
1055 | int ret; | ||
1056 | u16 reg; | 962 | u16 reg; |
1057 | 963 | ||
1058 | if (wm8961_codec) { | ||
1059 | dev_err(codec->dev, "Another WM8961 is registered\n"); | ||
1060 | ret = -EINVAL; | ||
1061 | goto err; | ||
1062 | } | ||
1063 | |||
1064 | mutex_init(&codec->mutex); | ||
1065 | INIT_LIST_HEAD(&codec->dapm_widgets); | ||
1066 | INIT_LIST_HEAD(&codec->dapm_paths); | ||
1067 | |||
1068 | snd_soc_codec_set_drvdata(codec, wm8961); | ||
1069 | codec->name = "WM8961"; | ||
1070 | codec->owner = THIS_MODULE; | ||
1071 | codec->dai = &wm8961_dai; | ||
1072 | codec->num_dai = 1; | ||
1073 | codec->reg_cache_size = ARRAY_SIZE(wm8961->reg_cache); | ||
1074 | codec->reg_cache = &wm8961->reg_cache; | ||
1075 | codec->bias_level = SND_SOC_BIAS_OFF; | ||
1076 | codec->set_bias_level = wm8961_set_bias_level; | ||
1077 | codec->volatile_register = wm8961_volatile_register; | ||
1078 | |||
1079 | memcpy(codec->reg_cache, wm8961_reg_defaults, | ||
1080 | sizeof(wm8961_reg_defaults)); | ||
1081 | |||
1082 | ret = snd_soc_codec_set_cache_io(codec, 8, 16, SND_SOC_I2C); | 964 | ret = snd_soc_codec_set_cache_io(codec, 8, 16, SND_SOC_I2C); |
1083 | if (ret != 0) { | 965 | if (ret != 0) { |
1084 | dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret); | 966 | dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret); |
1085 | goto err; | 967 | return ret; |
1086 | } | 968 | } |
1087 | 969 | ||
1088 | reg = snd_soc_read(codec, WM8961_SOFTWARE_RESET); | 970 | reg = snd_soc_read(codec, WM8961_SOFTWARE_RESET); |
1089 | if (reg != 0x1801) { | 971 | if (reg != 0x1801) { |
1090 | dev_err(codec->dev, "Device is not a WM8961: ID=0x%x\n", reg); | 972 | dev_err(codec->dev, "Device is not a WM8961: ID=0x%x\n", reg); |
1091 | ret = -EINVAL; | 973 | return -EINVAL; |
1092 | goto err; | ||
1093 | } | 974 | } |
1094 | 975 | ||
1095 | /* This isn't volatile - readback doesn't correspond to write */ | 976 | /* This isn't volatile - readback doesn't correspond to write */ |
@@ -1102,7 +983,7 @@ static int wm8961_register(struct wm8961_priv *wm8961) | |||
1102 | ret = wm8961_reset(codec); | 983 | ret = wm8961_reset(codec); |
1103 | if (ret < 0) { | 984 | if (ret < 0) { |
1104 | dev_err(codec->dev, "Failed to issue reset\n"); | 985 | dev_err(codec->dev, "Failed to issue reset\n"); |
1105 | goto err; | 986 | return ret; |
1106 | } | 987 | } |
1107 | 988 | ||
1108 | /* Enable class W */ | 989 | /* Enable class W */ |
@@ -1140,64 +1021,89 @@ static int wm8961_register(struct wm8961_priv *wm8961) | |||
1140 | 1021 | ||
1141 | wm8961_set_bias_level(codec, SND_SOC_BIAS_STANDBY); | 1022 | wm8961_set_bias_level(codec, SND_SOC_BIAS_STANDBY); |
1142 | 1023 | ||
1143 | wm8961_dai.dev = codec->dev; | 1024 | snd_soc_add_controls(codec, wm8961_snd_controls, |
1025 | ARRAY_SIZE(wm8961_snd_controls)); | ||
1026 | snd_soc_dapm_new_controls(dapm, wm8961_dapm_widgets, | ||
1027 | ARRAY_SIZE(wm8961_dapm_widgets)); | ||
1028 | snd_soc_dapm_add_routes(dapm, audio_paths, ARRAY_SIZE(audio_paths)); | ||
1144 | 1029 | ||
1145 | wm8961_codec = codec; | 1030 | return 0; |
1031 | } | ||
1146 | 1032 | ||
1147 | ret = snd_soc_register_codec(codec); | 1033 | static int wm8961_remove(struct snd_soc_codec *codec) |
1148 | if (ret != 0) { | 1034 | { |
1149 | dev_err(codec->dev, "Failed to register codec: %d\n", ret); | 1035 | wm8961_set_bias_level(codec, SND_SOC_BIAS_OFF); |
1150 | goto err; | 1036 | return 0; |
1151 | } | 1037 | } |
1152 | 1038 | ||
1153 | ret = snd_soc_register_dai(&wm8961_dai); | 1039 | #ifdef CONFIG_PM |
1154 | if (ret != 0) { | 1040 | static int wm8961_suspend(struct snd_soc_codec *codec, pm_message_t state) |
1155 | dev_err(codec->dev, "Failed to register DAI: %d\n", ret); | 1041 | { |
1156 | goto err_codec; | 1042 | wm8961_set_bias_level(codec, SND_SOC_BIAS_OFF); |
1157 | } | ||
1158 | 1043 | ||
1159 | return 0; | 1044 | return 0; |
1160 | |||
1161 | err_codec: | ||
1162 | snd_soc_unregister_codec(codec); | ||
1163 | err: | ||
1164 | kfree(wm8961); | ||
1165 | return ret; | ||
1166 | } | 1045 | } |
1167 | 1046 | ||
1168 | static void wm8961_unregister(struct wm8961_priv *wm8961) | 1047 | static int wm8961_resume(struct snd_soc_codec *codec) |
1169 | { | 1048 | { |
1170 | wm8961_set_bias_level(&wm8961->codec, SND_SOC_BIAS_OFF); | 1049 | u16 *reg_cache = codec->reg_cache; |
1171 | snd_soc_unregister_dai(&wm8961_dai); | 1050 | int i; |
1172 | snd_soc_unregister_codec(&wm8961->codec); | 1051 | |
1173 | kfree(wm8961); | 1052 | for (i = 0; i < codec->driver->reg_cache_size; i++) { |
1174 | wm8961_codec = NULL; | 1053 | if (reg_cache[i] == wm8961_reg_defaults[i]) |
1054 | continue; | ||
1055 | |||
1056 | if (i == WM8961_SOFTWARE_RESET) | ||
1057 | continue; | ||
1058 | |||
1059 | snd_soc_write(codec, i, reg_cache[i]); | ||
1060 | } | ||
1061 | |||
1062 | wm8961_set_bias_level(codec, SND_SOC_BIAS_STANDBY); | ||
1063 | |||
1064 | return 0; | ||
1175 | } | 1065 | } |
1066 | #else | ||
1067 | #define wm8961_suspend NULL | ||
1068 | #define wm8961_resume NULL | ||
1069 | #endif | ||
1176 | 1070 | ||
1071 | static struct snd_soc_codec_driver soc_codec_dev_wm8961 = { | ||
1072 | .probe = wm8961_probe, | ||
1073 | .remove = wm8961_remove, | ||
1074 | .suspend = wm8961_suspend, | ||
1075 | .resume = wm8961_resume, | ||
1076 | .set_bias_level = wm8961_set_bias_level, | ||
1077 | .reg_cache_size = ARRAY_SIZE(wm8961_reg_defaults), | ||
1078 | .reg_word_size = sizeof(u16), | ||
1079 | .reg_cache_default = wm8961_reg_defaults, | ||
1080 | .volatile_register = wm8961_volatile_register, | ||
1081 | }; | ||
1082 | |||
1083 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | ||
1177 | static __devinit int wm8961_i2c_probe(struct i2c_client *i2c, | 1084 | static __devinit int wm8961_i2c_probe(struct i2c_client *i2c, |
1178 | const struct i2c_device_id *id) | 1085 | const struct i2c_device_id *id) |
1179 | { | 1086 | { |
1180 | struct wm8961_priv *wm8961; | 1087 | struct wm8961_priv *wm8961; |
1181 | struct snd_soc_codec *codec; | 1088 | int ret; |
1182 | 1089 | ||
1183 | wm8961 = kzalloc(sizeof(struct wm8961_priv), GFP_KERNEL); | 1090 | wm8961 = kzalloc(sizeof(struct wm8961_priv), GFP_KERNEL); |
1184 | if (wm8961 == NULL) | 1091 | if (wm8961 == NULL) |
1185 | return -ENOMEM; | 1092 | return -ENOMEM; |
1186 | 1093 | ||
1187 | codec = &wm8961->codec; | ||
1188 | |||
1189 | i2c_set_clientdata(i2c, wm8961); | 1094 | i2c_set_clientdata(i2c, wm8961); |
1190 | codec->control_data = i2c; | ||
1191 | 1095 | ||
1192 | codec->dev = &i2c->dev; | 1096 | ret = snd_soc_register_codec(&i2c->dev, |
1193 | 1097 | &soc_codec_dev_wm8961, &wm8961_dai, 1); | |
1194 | return wm8961_register(wm8961); | 1098 | if (ret < 0) |
1099 | kfree(wm8961); | ||
1100 | return ret; | ||
1195 | } | 1101 | } |
1196 | 1102 | ||
1197 | static __devexit int wm8961_i2c_remove(struct i2c_client *client) | 1103 | static __devexit int wm8961_i2c_remove(struct i2c_client *client) |
1198 | { | 1104 | { |
1199 | struct wm8961_priv *wm8961 = i2c_get_clientdata(client); | 1105 | snd_soc_unregister_codec(&client->dev); |
1200 | wm8961_unregister(wm8961); | 1106 | kfree(i2c_get_clientdata(client)); |
1201 | return 0; | 1107 | return 0; |
1202 | } | 1108 | } |
1203 | 1109 | ||
@@ -1209,35 +1115,37 @@ MODULE_DEVICE_TABLE(i2c, wm8961_i2c_id); | |||
1209 | 1115 | ||
1210 | static struct i2c_driver wm8961_i2c_driver = { | 1116 | static struct i2c_driver wm8961_i2c_driver = { |
1211 | .driver = { | 1117 | .driver = { |
1212 | .name = "wm8961", | 1118 | .name = "wm8961-codec", |
1213 | .owner = THIS_MODULE, | 1119 | .owner = THIS_MODULE, |
1214 | }, | 1120 | }, |
1215 | .probe = wm8961_i2c_probe, | 1121 | .probe = wm8961_i2c_probe, |
1216 | .remove = __devexit_p(wm8961_i2c_remove), | 1122 | .remove = __devexit_p(wm8961_i2c_remove), |
1217 | .id_table = wm8961_i2c_id, | 1123 | .id_table = wm8961_i2c_id, |
1218 | }; | 1124 | }; |
1125 | #endif | ||
1219 | 1126 | ||
1220 | static int __init wm8961_modinit(void) | 1127 | static int __init wm8961_modinit(void) |
1221 | { | 1128 | { |
1222 | int ret; | 1129 | int ret = 0; |
1223 | 1130 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | |
1224 | ret = i2c_add_driver(&wm8961_i2c_driver); | 1131 | ret = i2c_add_driver(&wm8961_i2c_driver); |
1225 | if (ret != 0) { | 1132 | if (ret != 0) { |
1226 | printk(KERN_ERR "Failed to register WM8961 I2C driver: %d\n", | 1133 | printk(KERN_ERR "Failed to register wm8961 I2C driver: %d\n", |
1227 | ret); | 1134 | ret); |
1228 | } | 1135 | } |
1229 | 1136 | #endif | |
1230 | return ret; | 1137 | return ret; |
1231 | } | 1138 | } |
1232 | module_init(wm8961_modinit); | 1139 | module_init(wm8961_modinit); |
1233 | 1140 | ||
1234 | static void __exit wm8961_exit(void) | 1141 | static void __exit wm8961_exit(void) |
1235 | { | 1142 | { |
1143 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | ||
1236 | i2c_del_driver(&wm8961_i2c_driver); | 1144 | i2c_del_driver(&wm8961_i2c_driver); |
1145 | #endif | ||
1237 | } | 1146 | } |
1238 | module_exit(wm8961_exit); | 1147 | module_exit(wm8961_exit); |
1239 | 1148 | ||
1240 | |||
1241 | MODULE_DESCRIPTION("ASoC WM8961 driver"); | 1149 | MODULE_DESCRIPTION("ASoC WM8961 driver"); |
1242 | MODULE_AUTHOR("Mark Brown <broonie@opensource.wolfsonmicro.com>"); | 1150 | MODULE_AUTHOR("Mark Brown <broonie@opensource.wolfsonmicro.com>"); |
1243 | MODULE_LICENSE("GPL"); | 1151 | MODULE_LICENSE("GPL"); |