aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm8961.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm8961.c')
-rw-r--r--sound/soc/codecs/wm8961.c252
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
290struct wm8961_priv { 289struct 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
296static int wm8961_volatile_register(unsigned int reg) 294static 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
943struct snd_soc_dai wm8961_dai = { 941static 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};
959EXPORT_SYMBOL_GPL(wm8961_dai);
960 957
961 958static int wm8961_probe(struct snd_soc_codec *codec)
962static struct snd_soc_codec *wm8961_codec;
963
964static 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
993pcm_err:
994 return ret;
995}
996
997static 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
1008static 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
1018static 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
1044struct 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};
1050EXPORT_SYMBOL_GPL(soc_codec_dev_wm8961);
1051
1052static 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); 1033static 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) { 1040static 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
1161err_codec:
1162 snd_soc_unregister_codec(codec);
1163err:
1164 kfree(wm8961);
1165 return ret;
1166} 1045}
1167 1046
1168static void wm8961_unregister(struct wm8961_priv *wm8961) 1047static 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
1071static 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)
1177static __devinit int wm8961_i2c_probe(struct i2c_client *i2c, 1084static __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
1197static __devexit int wm8961_i2c_remove(struct i2c_client *client) 1103static __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
1210static struct i2c_driver wm8961_i2c_driver = { 1116static 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
1220static int __init wm8961_modinit(void) 1127static 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}
1232module_init(wm8961_modinit); 1139module_init(wm8961_modinit);
1233 1140
1234static void __exit wm8961_exit(void) 1141static 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}
1238module_exit(wm8961_exit); 1147module_exit(wm8961_exit);
1239 1148
1240
1241MODULE_DESCRIPTION("ASoC WM8961 driver"); 1149MODULE_DESCRIPTION("ASoC WM8961 driver");
1242MODULE_AUTHOR("Mark Brown <broonie@opensource.wolfsonmicro.com>"); 1150MODULE_AUTHOR("Mark Brown <broonie@opensource.wolfsonmicro.com>");
1243MODULE_LICENSE("GPL"); 1151MODULE_LICENSE("GPL");