aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm8900.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm8900.c')
-rw-r--r--sound/soc/codecs/wm8900.c265
1 files changed, 123 insertions, 142 deletions
diff --git a/sound/soc/codecs/wm8900.c b/sound/soc/codecs/wm8900.c
index 5da17a704e5a..449ea09a193d 100644
--- a/sound/soc/codecs/wm8900.c
+++ b/sound/soc/codecs/wm8900.c
@@ -23,13 +23,13 @@
23#include <linux/delay.h> 23#include <linux/delay.h>
24#include <linux/pm.h> 24#include <linux/pm.h>
25#include <linux/i2c.h> 25#include <linux/i2c.h>
26#include <linux/spi/spi.h>
26#include <linux/platform_device.h> 27#include <linux/platform_device.h>
27#include <linux/slab.h> 28#include <linux/slab.h>
28#include <sound/core.h> 29#include <sound/core.h>
29#include <sound/pcm.h> 30#include <sound/pcm.h>
30#include <sound/pcm_params.h> 31#include <sound/pcm_params.h>
31#include <sound/soc.h> 32#include <sound/soc.h>
32#include <sound/soc-dapm.h>
33#include <sound/initval.h> 33#include <sound/initval.h>
34#include <sound/tlv.h> 34#include <sound/tlv.h>
35 35
@@ -137,12 +137,8 @@
137 137
138#define WM8900_LRC_MASK 0xfc00 138#define WM8900_LRC_MASK 0xfc00
139 139
140struct snd_soc_codec_device soc_codec_dev_wm8900;
141
142struct wm8900_priv { 140struct wm8900_priv {
143 struct snd_soc_codec codec; 141 enum snd_soc_control_type control_type;
144
145 u16 reg_cache[WM8900_MAXREG];
146 142
147 u32 fll_in; /* FLL input frequency */ 143 u32 fll_in; /* FLL input frequency */
148 u32 fll_out; /* FLL output frequency */ 144 u32 fll_out; /* FLL output frequency */
@@ -184,11 +180,10 @@ static const u16 wm8900_reg_defaults[WM8900_MAXREG] = {
184 /* Remaining registers all zero */ 180 /* Remaining registers all zero */
185}; 181};
186 182
187static int wm8900_volatile_register(unsigned int reg) 183static int wm8900_volatile_register(struct snd_soc_codec *codec, unsigned int reg)
188{ 184{
189 switch (reg) { 185 switch (reg) {
190 case WM8900_REG_ID: 186 case WM8900_REG_ID:
191 case WM8900_REG_POWER1:
192 return 1; 187 return 1;
193 default: 188 default:
194 return 0; 189 return 0;
@@ -614,10 +609,11 @@ static const struct snd_soc_dapm_route audio_map[] = {
614 609
615static int wm8900_add_widgets(struct snd_soc_codec *codec) 610static int wm8900_add_widgets(struct snd_soc_codec *codec)
616{ 611{
617 snd_soc_dapm_new_controls(codec, wm8900_dapm_widgets, 612 struct snd_soc_dapm_context *dapm = &codec->dapm;
618 ARRAY_SIZE(wm8900_dapm_widgets));
619 613
620 snd_soc_dapm_add_routes(codec, audio_map, ARRAY_SIZE(audio_map)); 614 snd_soc_dapm_new_controls(dapm, wm8900_dapm_widgets,
615 ARRAY_SIZE(wm8900_dapm_widgets));
616 snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map));
621 617
622 return 0; 618 return 0;
623} 619}
@@ -627,8 +623,7 @@ static int wm8900_hw_params(struct snd_pcm_substream *substream,
627 struct snd_soc_dai *dai) 623 struct snd_soc_dai *dai)
628{ 624{
629 struct snd_soc_pcm_runtime *rtd = substream->private_data; 625 struct snd_soc_pcm_runtime *rtd = substream->private_data;
630 struct snd_soc_device *socdev = rtd->socdev; 626 struct snd_soc_codec *codec = rtd->codec;
631 struct snd_soc_codec *codec = socdev->card->codec;
632 u16 reg; 627 u16 reg;
633 628
634 reg = snd_soc_read(codec, WM8900_REG_AUDIO1) & ~0x60; 629 reg = snd_soc_read(codec, WM8900_REG_AUDIO1) & ~0x60;
@@ -1015,8 +1010,8 @@ static struct snd_soc_dai_ops wm8900_dai_ops = {
1015 .digital_mute = wm8900_digital_mute, 1010 .digital_mute = wm8900_digital_mute,
1016}; 1011};
1017 1012
1018struct snd_soc_dai wm8900_dai = { 1013static struct snd_soc_dai_driver wm8900_dai = {
1019 .name = "WM8900 HiFi", 1014 .name = "wm8900-hifi",
1020 .playback = { 1015 .playback = {
1021 .stream_name = "HiFi Playback", 1016 .stream_name = "HiFi Playback",
1022 .channels_min = 1, 1017 .channels_min = 1,
@@ -1033,7 +1028,6 @@ struct snd_soc_dai wm8900_dai = {
1033 }, 1028 },
1034 .ops = &wm8900_dai_ops, 1029 .ops = &wm8900_dai_ops,
1035}; 1030};
1036EXPORT_SYMBOL_GPL(wm8900_dai);
1037 1031
1038static int wm8900_set_bias_level(struct snd_soc_codec *codec, 1032static int wm8900_set_bias_level(struct snd_soc_codec *codec,
1039 enum snd_soc_bias_level level) 1033 enum snd_soc_bias_level level)
@@ -1056,7 +1050,7 @@ static int wm8900_set_bias_level(struct snd_soc_codec *codec,
1056 1050
1057 case SND_SOC_BIAS_STANDBY: 1051 case SND_SOC_BIAS_STANDBY:
1058 /* Charge capacitors if initial power up */ 1052 /* Charge capacitors if initial power up */
1059 if (codec->bias_level == SND_SOC_BIAS_OFF) { 1053 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
1060 /* STARTUP_BIAS_ENA on */ 1054 /* STARTUP_BIAS_ENA on */
1061 snd_soc_write(codec, WM8900_REG_POWER1, 1055 snd_soc_write(codec, WM8900_REG_POWER1,
1062 WM8900_REG_POWER1_STARTUP_BIAS_ENA); 1056 WM8900_REG_POWER1_STARTUP_BIAS_ENA);
@@ -1124,14 +1118,12 @@ static int wm8900_set_bias_level(struct snd_soc_codec *codec,
1124 WM8900_REG_POWER2_SYSCLK_ENA); 1118 WM8900_REG_POWER2_SYSCLK_ENA);
1125 break; 1119 break;
1126 } 1120 }
1127 codec->bias_level = level; 1121 codec->dapm.bias_level = level;
1128 return 0; 1122 return 0;
1129} 1123}
1130 1124
1131static int wm8900_suspend(struct platform_device *pdev, pm_message_t state) 1125static int wm8900_suspend(struct snd_soc_codec *codec, pm_message_t state)
1132{ 1126{
1133 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1134 struct snd_soc_codec *codec = socdev->card->codec;
1135 struct wm8900_priv *wm8900 = snd_soc_codec_get_drvdata(codec); 1127 struct wm8900_priv *wm8900 = snd_soc_codec_get_drvdata(codec);
1136 int fll_out = wm8900->fll_out; 1128 int fll_out = wm8900->fll_out;
1137 int fll_in = wm8900->fll_in; 1129 int fll_in = wm8900->fll_in;
@@ -1140,7 +1132,7 @@ static int wm8900_suspend(struct platform_device *pdev, pm_message_t state)
1140 /* Stop the FLL in an orderly fashion */ 1132 /* Stop the FLL in an orderly fashion */
1141 ret = wm8900_set_fll(codec, 0, 0, 0); 1133 ret = wm8900_set_fll(codec, 0, 0, 0);
1142 if (ret != 0) { 1134 if (ret != 0) {
1143 dev_err(&pdev->dev, "Failed to stop FLL\n"); 1135 dev_err(codec->dev, "Failed to stop FLL\n");
1144 return ret; 1136 return ret;
1145 } 1137 }
1146 1138
@@ -1152,10 +1144,8 @@ static int wm8900_suspend(struct platform_device *pdev, pm_message_t state)
1152 return 0; 1144 return 0;
1153} 1145}
1154 1146
1155static int wm8900_resume(struct platform_device *pdev) 1147static int wm8900_resume(struct snd_soc_codec *codec)
1156{ 1148{
1157 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1158 struct snd_soc_codec *codec = socdev->card->codec;
1159 struct wm8900_priv *wm8900 = snd_soc_codec_get_drvdata(codec); 1149 struct wm8900_priv *wm8900 = snd_soc_codec_get_drvdata(codec);
1160 u16 *cache; 1150 u16 *cache;
1161 int i, ret; 1151 int i, ret;
@@ -1176,7 +1166,7 @@ static int wm8900_resume(struct platform_device *pdev)
1176 1166
1177 ret = wm8900_set_fll(codec, 0, fll_in, fll_out); 1167 ret = wm8900_set_fll(codec, 0, fll_in, fll_out);
1178 if (ret != 0) { 1168 if (ret != 0) {
1179 dev_err(&pdev->dev, "Failed to restart FLL\n"); 1169 dev_err(codec->dev, "Failed to restart FLL\n");
1180 return ret; 1170 return ret;
1181 } 1171 }
1182 } 1172 }
@@ -1186,61 +1176,28 @@ static int wm8900_resume(struct platform_device *pdev)
1186 snd_soc_write(codec, i, cache[i]); 1176 snd_soc_write(codec, i, cache[i]);
1187 kfree(cache); 1177 kfree(cache);
1188 } else 1178 } else
1189 dev_err(&pdev->dev, "Unable to allocate register cache\n"); 1179 dev_err(codec->dev, "Unable to allocate register cache\n");
1190 1180
1191 return 0; 1181 return 0;
1192} 1182}
1193 1183
1194static struct snd_soc_codec *wm8900_codec; 1184static int wm8900_probe(struct snd_soc_codec *codec)
1195
1196static __devinit int wm8900_i2c_probe(struct i2c_client *i2c,
1197 const struct i2c_device_id *id)
1198{ 1185{
1199 struct wm8900_priv *wm8900; 1186 struct wm8900_priv *wm8900 = snd_soc_codec_get_drvdata(codec);
1200 struct snd_soc_codec *codec; 1187 int ret = 0, reg;
1201 unsigned int reg;
1202 int ret;
1203
1204 wm8900 = kzalloc(sizeof(struct wm8900_priv), GFP_KERNEL);
1205 if (wm8900 == NULL)
1206 return -ENOMEM;
1207 1188
1208 codec = &wm8900->codec; 1189 ret = snd_soc_codec_set_cache_io(codec, 8, 16, wm8900->control_type);
1209 snd_soc_codec_set_drvdata(codec, wm8900);
1210 codec->reg_cache = &wm8900->reg_cache[0];
1211 codec->reg_cache_size = WM8900_MAXREG;
1212
1213 mutex_init(&codec->mutex);
1214 INIT_LIST_HEAD(&codec->dapm_widgets);
1215 INIT_LIST_HEAD(&codec->dapm_paths);
1216
1217 codec->name = "WM8900";
1218 codec->owner = THIS_MODULE;
1219 codec->dai = &wm8900_dai;
1220 codec->num_dai = 1;
1221 codec->control_data = i2c;
1222 codec->set_bias_level = wm8900_set_bias_level;
1223 codec->volatile_register = wm8900_volatile_register;
1224 codec->dev = &i2c->dev;
1225
1226 ret = snd_soc_codec_set_cache_io(codec, 8, 16, SND_SOC_I2C);
1227 if (ret != 0) { 1190 if (ret != 0) {
1228 dev_err(&i2c->dev, "Failed to set cache I/O: %d\n", ret); 1191 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
1229 goto err; 1192 return ret;
1230 } 1193 }
1231 1194
1232 reg = snd_soc_read(codec, WM8900_REG_ID); 1195 reg = snd_soc_read(codec, WM8900_REG_ID);
1233 if (reg != 0x8900) { 1196 if (reg != 0x8900) {
1234 dev_err(&i2c->dev, "Device is not a WM8900 - ID %x\n", reg); 1197 dev_err(codec->dev, "Device is not a WM8900 - ID %x\n", reg);
1235 ret = -ENODEV; 1198 return -ENODEV;
1236 goto err;
1237 } 1199 }
1238 1200
1239 /* Read back from the chip */
1240 reg = snd_soc_read(codec, WM8900_REG_POWER1);
1241 reg = (reg >> 12) & 0xf;
1242 dev_info(&i2c->dev, "WM8900 revision %d\n", reg);
1243
1244 wm8900_reset(codec); 1201 wm8900_reset(codec);
1245 1202
1246 /* Turn the chip on */ 1203 /* Turn the chip on */
@@ -1271,43 +1228,94 @@ static __devinit int wm8900_i2c_probe(struct i2c_client *i2c,
1271 /* Set the DAC and mixer output bias */ 1228 /* Set the DAC and mixer output bias */
1272 snd_soc_write(codec, WM8900_REG_OUTBIASCTL, 0x81); 1229 snd_soc_write(codec, WM8900_REG_OUTBIASCTL, 0x81);
1273 1230
1274 wm8900_dai.dev = &i2c->dev; 1231 snd_soc_add_controls(codec, wm8900_snd_controls,
1232 ARRAY_SIZE(wm8900_snd_controls));
1233 wm8900_add_widgets(codec);
1234
1235 return 0;
1236}
1275 1237
1276 wm8900_codec = codec; 1238/* power down chip */
1239static int wm8900_remove(struct snd_soc_codec *codec)
1240{
1241 wm8900_set_bias_level(codec, SND_SOC_BIAS_OFF);
1242 return 0;
1243}
1277 1244
1278 ret = snd_soc_register_codec(codec); 1245static struct snd_soc_codec_driver soc_codec_dev_wm8900 = {
1279 if (ret != 0) { 1246 .probe = wm8900_probe,
1280 dev_err(&i2c->dev, "Failed to register codec: %d\n", ret); 1247 .remove = wm8900_remove,
1281 goto err; 1248 .suspend = wm8900_suspend,
1282 } 1249 .resume = wm8900_resume,
1250 .set_bias_level = wm8900_set_bias_level,
1251 .volatile_register = wm8900_volatile_register,
1252 .reg_cache_size = ARRAY_SIZE(wm8900_reg_defaults),
1253 .reg_word_size = sizeof(u16),
1254 .reg_cache_default = wm8900_reg_defaults,
1255};
1283 1256
1284 ret = snd_soc_register_dai(&wm8900_dai); 1257#if defined(CONFIG_SPI_MASTER)
1285 if (ret != 0) { 1258static int __devinit wm8900_spi_probe(struct spi_device *spi)
1286 dev_err(&i2c->dev, "Failed to register DAI: %d\n", ret); 1259{
1287 goto err_codec; 1260 struct wm8900_priv *wm8900;
1288 } 1261 int ret;
1289 1262
1290 return ret; 1263 wm8900 = kzalloc(sizeof(struct wm8900_priv), GFP_KERNEL);
1264 if (wm8900 == NULL)
1265 return -ENOMEM;
1266
1267 wm8900->control_type = SND_SOC_SPI;
1268 spi_set_drvdata(spi, wm8900);
1291 1269
1292err_codec: 1270 ret = snd_soc_register_codec(&spi->dev,
1293 snd_soc_unregister_codec(codec); 1271 &soc_codec_dev_wm8900, &wm8900_dai, 1);
1294err: 1272 if (ret < 0)
1295 kfree(wm8900); 1273 kfree(wm8900);
1296 wm8900_codec = NULL;
1297 return ret; 1274 return ret;
1298} 1275}
1299 1276
1300static __devexit int wm8900_i2c_remove(struct i2c_client *client) 1277static int __devexit wm8900_spi_remove(struct spi_device *spi)
1301{ 1278{
1302 snd_soc_unregister_dai(&wm8900_dai); 1279 snd_soc_unregister_codec(&spi->dev);
1303 snd_soc_unregister_codec(wm8900_codec); 1280 kfree(spi_get_drvdata(spi));
1281 return 0;
1282}
1304 1283
1305 wm8900_set_bias_level(wm8900_codec, SND_SOC_BIAS_OFF); 1284static struct spi_driver wm8900_spi_driver = {
1285 .driver = {
1286 .name = "wm8900-codec",
1287 .owner = THIS_MODULE,
1288 },
1289 .probe = wm8900_spi_probe,
1290 .remove = __devexit_p(wm8900_spi_remove),
1291};
1292#endif /* CONFIG_SPI_MASTER */
1306 1293
1307 wm8900_dai.dev = NULL; 1294#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1308 kfree(snd_soc_codec_get_drvdata(wm8900_codec)); 1295static __devinit int wm8900_i2c_probe(struct i2c_client *i2c,
1309 wm8900_codec = NULL; 1296 const struct i2c_device_id *id)
1297{
1298 struct wm8900_priv *wm8900;
1299 int ret;
1310 1300
1301 wm8900 = kzalloc(sizeof(struct wm8900_priv), GFP_KERNEL);
1302 if (wm8900 == NULL)
1303 return -ENOMEM;
1304
1305 i2c_set_clientdata(i2c, wm8900);
1306 wm8900->control_type = SND_SOC_I2C;
1307
1308 ret = snd_soc_register_codec(&i2c->dev,
1309 &soc_codec_dev_wm8900, &wm8900_dai, 1);
1310 if (ret < 0)
1311 kfree(wm8900);
1312 return ret;
1313}
1314
1315static __devexit int wm8900_i2c_remove(struct i2c_client *client)
1316{
1317 snd_soc_unregister_codec(&client->dev);
1318 kfree(i2c_get_clientdata(client));
1311 return 0; 1319 return 0;
1312} 1320}
1313 1321
@@ -1319,71 +1327,44 @@ MODULE_DEVICE_TABLE(i2c, wm8900_i2c_id);
1319 1327
1320static struct i2c_driver wm8900_i2c_driver = { 1328static struct i2c_driver wm8900_i2c_driver = {
1321 .driver = { 1329 .driver = {
1322 .name = "WM8900", 1330 .name = "wm8900-codec",
1323 .owner = THIS_MODULE, 1331 .owner = THIS_MODULE,
1324 }, 1332 },
1325 .probe = wm8900_i2c_probe, 1333 .probe = wm8900_i2c_probe,
1326 .remove = __devexit_p(wm8900_i2c_remove), 1334 .remove = __devexit_p(wm8900_i2c_remove),
1327 .id_table = wm8900_i2c_id, 1335 .id_table = wm8900_i2c_id,
1328}; 1336};
1337#endif
1329 1338
1330static int wm8900_probe(struct platform_device *pdev) 1339static int __init wm8900_modinit(void)
1331{ 1340{
1332 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1333 struct snd_soc_codec *codec;
1334 int ret = 0; 1341 int ret = 0;
1335 1342#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1336 if (!wm8900_codec) { 1343 ret = i2c_add_driver(&wm8900_i2c_driver);
1337 dev_err(&pdev->dev, "I2C client not yet instantiated\n"); 1344 if (ret != 0) {
1338 return -ENODEV; 1345 printk(KERN_ERR "Failed to register wm8900 I2C driver: %d\n",
1346 ret);
1339 } 1347 }
1340 1348#endif
1341 codec = wm8900_codec; 1349#if defined(CONFIG_SPI_MASTER)
1342 socdev->card->codec = codec; 1350 ret = spi_register_driver(&wm8900_spi_driver);
1343 1351 if (ret != 0) {
1344 /* Register pcms */ 1352 printk(KERN_ERR "Failed to register wm8900 SPI driver: %d\n",
1345 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); 1353 ret);
1346 if (ret < 0) {
1347 dev_err(&pdev->dev, "Failed to register new PCMs\n");
1348 goto pcm_err;
1349 } 1354 }
1350 1355#endif
1351 snd_soc_add_controls(codec, wm8900_snd_controls,
1352 ARRAY_SIZE(wm8900_snd_controls));
1353 wm8900_add_widgets(codec);
1354
1355pcm_err:
1356 return ret; 1356 return ret;
1357} 1357}
1358
1359/* power down chip */
1360static int wm8900_remove(struct platform_device *pdev)
1361{
1362 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1363
1364 snd_soc_free_pcms(socdev);
1365 snd_soc_dapm_free(socdev);
1366
1367 return 0;
1368}
1369
1370struct snd_soc_codec_device soc_codec_dev_wm8900 = {
1371 .probe = wm8900_probe,
1372 .remove = wm8900_remove,
1373 .suspend = wm8900_suspend,
1374 .resume = wm8900_resume,
1375};
1376EXPORT_SYMBOL_GPL(soc_codec_dev_wm8900);
1377
1378static int __init wm8900_modinit(void)
1379{
1380 return i2c_add_driver(&wm8900_i2c_driver);
1381}
1382module_init(wm8900_modinit); 1358module_init(wm8900_modinit);
1383 1359
1384static void __exit wm8900_exit(void) 1360static void __exit wm8900_exit(void)
1385{ 1361{
1362#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1386 i2c_del_driver(&wm8900_i2c_driver); 1363 i2c_del_driver(&wm8900_i2c_driver);
1364#endif
1365#if defined(CONFIG_SPI_MASTER)
1366 spi_unregister_driver(&wm8900_spi_driver);
1367#endif
1387} 1368}
1388module_exit(wm8900_exit); 1369module_exit(wm8900_exit);
1389 1370