aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm9081.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm9081.c')
-rw-r--r--sound/soc/codecs/wm9081.c297
1 files changed, 103 insertions, 194 deletions
diff --git a/sound/soc/codecs/wm9081.c b/sound/soc/codecs/wm9081.c
index 76b37ff6c264..91c6b39de50c 100644
--- a/sound/soc/codecs/wm9081.c
+++ b/sound/soc/codecs/wm9081.c
@@ -15,6 +15,7 @@
15#include <linux/moduleparam.h> 15#include <linux/moduleparam.h>
16#include <linux/init.h> 16#include <linux/init.h>
17#include <linux/delay.h> 17#include <linux/delay.h>
18#include <linux/device.h>
18#include <linux/pm.h> 19#include <linux/pm.h>
19#include <linux/i2c.h> 20#include <linux/i2c.h>
20#include <linux/platform_device.h> 21#include <linux/platform_device.h>
@@ -23,7 +24,6 @@
23#include <sound/pcm.h> 24#include <sound/pcm.h>
24#include <sound/pcm_params.h> 25#include <sound/pcm_params.h>
25#include <sound/soc.h> 26#include <sound/soc.h>
26#include <sound/soc-dapm.h>
27#include <sound/initval.h> 27#include <sound/initval.h>
28#include <sound/tlv.h> 28#include <sound/tlv.h>
29 29
@@ -156,8 +156,8 @@ static struct {
156}; 156};
157 157
158struct wm9081_priv { 158struct wm9081_priv {
159 struct snd_soc_codec codec; 159 enum snd_soc_control_type control_type;
160 u16 reg_cache[WM9081_MAX_REGISTER + 1]; 160 void *control_data;
161 int sysclk_source; 161 int sysclk_source;
162 int mclk_rate; 162 int mclk_rate;
163 int sysclk_rate; 163 int sysclk_rate;
@@ -167,10 +167,10 @@ struct wm9081_priv {
167 int fll_fref; 167 int fll_fref;
168 int fll_fout; 168 int fll_fout;
169 int tdm_width; 169 int tdm_width;
170 struct wm9081_retune_mobile_config *retune; 170 struct wm9081_pdata pdata;
171}; 171};
172 172
173static int wm9081_volatile_register(unsigned int reg) 173static int wm9081_volatile_register(struct snd_soc_codec *codec, unsigned int reg)
174{ 174{
175 switch (reg) { 175 switch (reg) {
176 case WM9081_SOFTWARE_RESET: 176 case WM9081_SOFTWARE_RESET:
@@ -305,7 +305,7 @@ static int speaker_mode_get(struct snd_kcontrol *kcontrol,
305/* 305/*
306 * Stop any attempts to change speaker mode while the speaker is enabled. 306 * Stop any attempts to change speaker mode while the speaker is enabled.
307 * 307 *
308 * We also have some special anti-pop controls dependant on speaker 308 * We also have some special anti-pop controls dependent on speaker
309 * mode which must be changed along with the mode. 309 * mode which must be changed along with the mode.
310 */ 310 */
311static int speaker_mode_put(struct snd_kcontrol *kcontrol, 311static int speaker_mode_put(struct snd_kcontrol *kcontrol,
@@ -389,27 +389,6 @@ SOC_DAPM_SINGLE("IN2 Switch", WM9081_ANALOGUE_MIXER, 2, 1, 0),
389SOC_DAPM_SINGLE("Playback Switch", WM9081_ANALOGUE_MIXER, 4, 1, 0), 389SOC_DAPM_SINGLE("Playback Switch", WM9081_ANALOGUE_MIXER, 4, 1, 0),
390}; 390};
391 391
392static int speaker_event(struct snd_soc_dapm_widget *w,
393 struct snd_kcontrol *kcontrol, int event)
394{
395 struct snd_soc_codec *codec = w->codec;
396 unsigned int reg = snd_soc_read(codec, WM9081_POWER_MANAGEMENT);
397
398 switch (event) {
399 case SND_SOC_DAPM_POST_PMU:
400 reg |= WM9081_SPK_ENA;
401 break;
402
403 case SND_SOC_DAPM_PRE_PMD:
404 reg &= ~WM9081_SPK_ENA;
405 break;
406 }
407
408 snd_soc_write(codec, WM9081_POWER_MANAGEMENT, reg);
409
410 return 0;
411}
412
413struct _fll_div { 392struct _fll_div {
414 u16 fll_fratio; 393 u16 fll_fratio;
415 u16 fll_outdiv; 394 u16 fll_outdiv;
@@ -477,7 +456,7 @@ static int fll_factors(struct _fll_div *fll_div, unsigned int Fref,
477 456
478 pr_debug("Fvco=%dHz\n", target); 457 pr_debug("Fvco=%dHz\n", target);
479 458
480 /* Find an appropraite FLL_FRATIO and factor it out of the target */ 459 /* Find an appropriate FLL_FRATIO and factor it out of the target */
481 for (i = 0; i < ARRAY_SIZE(fll_fratios); i++) { 460 for (i = 0; i < ARRAY_SIZE(fll_fratios); i++) {
482 if (fll_fratios[i].min <= Fref && Fref <= fll_fratios[i].max) { 461 if (fll_fratios[i].min <= Fref && Fref <= fll_fratios[i].max) {
483 fll_div->fll_fratio = fll_fratios[i].fll_fratio; 462 fll_div->fll_fratio = fll_fratios[i].fll_fratio;
@@ -590,6 +569,10 @@ static int wm9081_set_fll(struct snd_soc_codec *codec, int fll_id,
590 reg5 |= fll_div.fll_clk_ref_div << WM9081_FLL_CLK_REF_DIV_SHIFT; 569 reg5 |= fll_div.fll_clk_ref_div << WM9081_FLL_CLK_REF_DIV_SHIFT;
591 snd_soc_write(codec, WM9081_FLL_CONTROL_5, reg5); 570 snd_soc_write(codec, WM9081_FLL_CONTROL_5, reg5);
592 571
572 /* Set gain to the recommended value */
573 snd_soc_update_bits(codec, WM9081_FLL_CONTROL_4,
574 WM9081_FLL_GAIN_MASK, 0);
575
593 /* Enable the FLL */ 576 /* Enable the FLL */
594 snd_soc_write(codec, WM9081_FLL_CONTROL_1, reg1 | WM9081_FLL_ENA); 577 snd_soc_write(codec, WM9081_FLL_CONTROL_1, reg1 | WM9081_FLL_ENA);
595 578
@@ -743,9 +726,8 @@ SND_SOC_DAPM_MIXER_NAMED_CTL("Mixer", SND_SOC_NOPM, 0, 0,
743 726
744SND_SOC_DAPM_PGA("LINEOUT PGA", WM9081_POWER_MANAGEMENT, 4, 0, NULL, 0), 727SND_SOC_DAPM_PGA("LINEOUT PGA", WM9081_POWER_MANAGEMENT, 4, 0, NULL, 0),
745 728
746SND_SOC_DAPM_PGA_E("Speaker PGA", WM9081_POWER_MANAGEMENT, 2, 0, NULL, 0, 729SND_SOC_DAPM_PGA("Speaker PGA", WM9081_POWER_MANAGEMENT, 2, 0, NULL, 0),
747 speaker_event, 730SND_SOC_DAPM_PGA("Speaker", WM9081_POWER_MANAGEMENT, 1, 0, NULL, 0),
748 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
749 731
750SND_SOC_DAPM_OUTPUT("LINEOUT"), 732SND_SOC_DAPM_OUTPUT("LINEOUT"),
751SND_SOC_DAPM_OUTPUT("SPKN"), 733SND_SOC_DAPM_OUTPUT("SPKN"),
@@ -758,7 +740,7 @@ SND_SOC_DAPM_SUPPLY("TOCLK", WM9081_CLOCK_CONTROL_3, 2, 0, NULL, 0),
758}; 740};
759 741
760 742
761static const struct snd_soc_dapm_route audio_paths[] = { 743static const struct snd_soc_dapm_route wm9081_audio_paths[] = {
762 { "DAC", NULL, "CLK_SYS" }, 744 { "DAC", NULL, "CLK_SYS" },
763 { "DAC", NULL, "CLK_DSP" }, 745 { "DAC", NULL, "CLK_DSP" },
764 746
@@ -776,8 +758,10 @@ static const struct snd_soc_dapm_route audio_paths[] = {
776 { "Speaker PGA", NULL, "TOCLK" }, 758 { "Speaker PGA", NULL, "TOCLK" },
777 { "Speaker PGA", NULL, "CLK_SYS" }, 759 { "Speaker PGA", NULL, "CLK_SYS" },
778 760
779 { "SPKN", NULL, "Speaker PGA" }, 761 { "Speaker", NULL, "Speaker PGA" },
780 { "SPKP", NULL, "Speaker PGA" }, 762
763 { "SPKN", NULL, "Speaker" },
764 { "SPKP", NULL, "Speaker" },
781}; 765};
782 766
783static int wm9081_set_bias_level(struct snd_soc_codec *codec, 767static int wm9081_set_bias_level(struct snd_soc_codec *codec,
@@ -804,7 +788,7 @@ static int wm9081_set_bias_level(struct snd_soc_codec *codec,
804 788
805 case SND_SOC_BIAS_STANDBY: 789 case SND_SOC_BIAS_STANDBY:
806 /* Initial cold start */ 790 /* Initial cold start */
807 if (codec->bias_level == SND_SOC_BIAS_OFF) { 791 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
808 /* Disable LINEOUT discharge */ 792 /* Disable LINEOUT discharge */
809 reg = snd_soc_read(codec, WM9081_ANTI_POP_CONTROL); 793 reg = snd_soc_read(codec, WM9081_ANTI_POP_CONTROL);
810 reg &= ~WM9081_LINEOUT_DISCH; 794 reg &= ~WM9081_LINEOUT_DISCH;
@@ -864,7 +848,7 @@ static int wm9081_set_bias_level(struct snd_soc_codec *codec,
864 break; 848 break;
865 } 849 }
866 850
867 codec->bias_level = level; 851 codec->dapm.bias_level = level;
868 852
869 return 0; 853 return 0;
870} 854}
@@ -1078,21 +1062,22 @@ static int wm9081_hw_params(struct snd_pcm_substream *substream,
1078 aif4 |= wm9081->bclk / wm9081->fs; 1062 aif4 |= wm9081->bclk / wm9081->fs;
1079 1063
1080 /* Apply a ReTune Mobile configuration if it's in use */ 1064 /* Apply a ReTune Mobile configuration if it's in use */
1081 if (wm9081->retune) { 1065 if (wm9081->pdata.num_retune_configs) {
1082 struct wm9081_retune_mobile_config *retune = wm9081->retune; 1066 struct wm9081_pdata *pdata = &wm9081->pdata;
1083 struct wm9081_retune_mobile_setting *s; 1067 struct wm9081_retune_mobile_setting *s;
1084 int eq1; 1068 int eq1;
1085 1069
1086 best = 0; 1070 best = 0;
1087 best_val = abs(retune->configs[0].rate - wm9081->fs); 1071 best_val = abs(pdata->retune_configs[0].rate - wm9081->fs);
1088 for (i = 0; i < retune->num_configs; i++) { 1072 for (i = 0; i < pdata->num_retune_configs; i++) {
1089 cur_val = abs(retune->configs[i].rate - wm9081->fs); 1073 cur_val = abs(pdata->retune_configs[i].rate -
1074 wm9081->fs);
1090 if (cur_val < best_val) { 1075 if (cur_val < best_val) {
1091 best_val = cur_val; 1076 best_val = cur_val;
1092 best = i; 1077 best = i;
1093 } 1078 }
1094 } 1079 }
1095 s = &retune->configs[best]; 1080 s = &pdata->retune_configs[best];
1096 1081
1097 dev_dbg(codec->dev, "ReTune Mobile %s tuned for %dHz\n", 1082 dev_dbg(codec->dev, "ReTune Mobile %s tuned for %dHz\n",
1098 s->name, s->rate); 1083 s->name, s->rate);
@@ -1135,10 +1120,9 @@ static int wm9081_digital_mute(struct snd_soc_dai *codec_dai, int mute)
1135 return 0; 1120 return 0;
1136} 1121}
1137 1122
1138static int wm9081_set_sysclk(struct snd_soc_dai *codec_dai, 1123static int wm9081_set_sysclk(struct snd_soc_codec *codec,
1139 int clk_id, unsigned int freq, int dir) 1124 int clk_id, unsigned int freq, int dir)
1140{ 1125{
1141 struct snd_soc_codec *codec = codec_dai->codec;
1142 struct wm9081_priv *wm9081 = snd_soc_codec_get_drvdata(codec); 1126 struct wm9081_priv *wm9081 = snd_soc_codec_get_drvdata(codec);
1143 1127
1144 switch (clk_id) { 1128 switch (clk_id) {
@@ -1203,7 +1187,6 @@ static int wm9081_set_tdm_slot(struct snd_soc_dai *dai,
1203 1187
1204static struct snd_soc_dai_ops wm9081_dai_ops = { 1188static struct snd_soc_dai_ops wm9081_dai_ops = {
1205 .hw_params = wm9081_hw_params, 1189 .hw_params = wm9081_hw_params,
1206 .set_sysclk = wm9081_set_sysclk,
1207 .set_fmt = wm9081_set_dai_fmt, 1190 .set_fmt = wm9081_set_dai_fmt,
1208 .digital_mute = wm9081_digital_mute, 1191 .digital_mute = wm9081_digital_mute,
1209 .set_tdm_slot = wm9081_set_tdm_slot, 1192 .set_tdm_slot = wm9081_set_tdm_slot,
@@ -1212,8 +1195,8 @@ static struct snd_soc_dai_ops wm9081_dai_ops = {
1212/* We report two channels because the CODEC processes a stereo signal, even 1195/* We report two channels because the CODEC processes a stereo signal, even
1213 * though it is only capable of handling a mono output. 1196 * though it is only capable of handling a mono output.
1214 */ 1197 */
1215struct snd_soc_dai wm9081_dai = { 1198static struct snd_soc_dai_driver wm9081_dai = {
1216 .name = "WM9081", 1199 .name = "wm9081-hifi",
1217 .playback = { 1200 .playback = {
1218 .stream_name = "HiFi Playback", 1201 .stream_name = "HiFi Playback",
1219 .channels_min = 1, 1202 .channels_min = 1,
@@ -1223,82 +1206,82 @@ struct snd_soc_dai wm9081_dai = {
1223 }, 1206 },
1224 .ops = &wm9081_dai_ops, 1207 .ops = &wm9081_dai_ops,
1225}; 1208};
1226EXPORT_SYMBOL_GPL(wm9081_dai);
1227 1209
1228 1210static int wm9081_probe(struct snd_soc_codec *codec)
1229static struct snd_soc_codec *wm9081_codec;
1230
1231static int wm9081_probe(struct platform_device *pdev)
1232{ 1211{
1233 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 1212 struct wm9081_priv *wm9081 = snd_soc_codec_get_drvdata(codec);
1234 struct snd_soc_codec *codec; 1213 int ret;
1235 struct wm9081_priv *wm9081; 1214 u16 reg;
1236 int ret = 0;
1237 1215
1238 if (wm9081_codec == NULL) { 1216 codec->control_data = wm9081->control_data;
1239 dev_err(&pdev->dev, "Codec device not registered\n"); 1217 ret = snd_soc_codec_set_cache_io(codec, 8, 16, wm9081->control_type);
1240 return -ENODEV; 1218 if (ret != 0) {
1219 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
1220 return ret;
1241 } 1221 }
1242 1222
1243 socdev->card->codec = wm9081_codec; 1223 reg = snd_soc_read(codec, WM9081_SOFTWARE_RESET);
1244 codec = wm9081_codec; 1224 if (reg != 0x9081) {
1245 wm9081 = snd_soc_codec_get_drvdata(codec); 1225 dev_err(codec->dev, "Device is not a WM9081: ID=0x%x\n", reg);
1226 ret = -EINVAL;
1227 return ret;
1228 }
1246 1229
1247 /* register pcms */ 1230 ret = wm9081_reset(codec);
1248 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1);
1249 if (ret < 0) { 1231 if (ret < 0) {
1250 dev_err(codec->dev, "failed to create pcms: %d\n", ret); 1232 dev_err(codec->dev, "Failed to issue reset\n");
1251 goto pcm_err; 1233 return ret;
1252 } 1234 }
1253 1235
1236 reg = 0;
1237 if (wm9081->pdata.irq_high)
1238 reg |= WM9081_IRQ_POL;
1239 if (!wm9081->pdata.irq_cmos)
1240 reg |= WM9081_IRQ_OP_CTRL;
1241 snd_soc_update_bits(codec, WM9081_INTERRUPT_CONTROL,
1242 WM9081_IRQ_POL | WM9081_IRQ_OP_CTRL, reg);
1243
1244 wm9081_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
1245
1246 /* Enable zero cross by default */
1247 reg = snd_soc_read(codec, WM9081_ANALOGUE_LINEOUT);
1248 snd_soc_write(codec, WM9081_ANALOGUE_LINEOUT, reg | WM9081_LINEOUTZC);
1249 reg = snd_soc_read(codec, WM9081_ANALOGUE_SPEAKER_PGA);
1250 snd_soc_write(codec, WM9081_ANALOGUE_SPEAKER_PGA,
1251 reg | WM9081_SPKPGAZC);
1252
1254 snd_soc_add_controls(codec, wm9081_snd_controls, 1253 snd_soc_add_controls(codec, wm9081_snd_controls,
1255 ARRAY_SIZE(wm9081_snd_controls)); 1254 ARRAY_SIZE(wm9081_snd_controls));
1256 if (!wm9081->retune) { 1255 if (!wm9081->pdata.num_retune_configs) {
1257 dev_dbg(codec->dev, 1256 dev_dbg(codec->dev,
1258 "No ReTune Mobile data, using normal EQ\n"); 1257 "No ReTune Mobile data, using normal EQ\n");
1259 snd_soc_add_controls(codec, wm9081_eq_controls, 1258 snd_soc_add_controls(codec, wm9081_eq_controls,
1260 ARRAY_SIZE(wm9081_eq_controls)); 1259 ARRAY_SIZE(wm9081_eq_controls));
1261 } 1260 }
1262 1261
1263 snd_soc_dapm_new_controls(codec, wm9081_dapm_widgets,
1264 ARRAY_SIZE(wm9081_dapm_widgets));
1265 snd_soc_dapm_add_routes(codec, audio_paths, ARRAY_SIZE(audio_paths));
1266
1267 return ret;
1268
1269pcm_err:
1270 return ret; 1262 return ret;
1271} 1263}
1272 1264
1273static int wm9081_remove(struct platform_device *pdev) 1265static int wm9081_remove(struct snd_soc_codec *codec)
1274{ 1266{
1275 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 1267 wm9081_set_bias_level(codec, SND_SOC_BIAS_OFF);
1276
1277 snd_soc_free_pcms(socdev);
1278 snd_soc_dapm_free(socdev);
1279
1280 return 0; 1268 return 0;
1281} 1269}
1282 1270
1283#ifdef CONFIG_PM 1271#ifdef CONFIG_PM
1284static int wm9081_suspend(struct platform_device *pdev, pm_message_t state) 1272static int wm9081_suspend(struct snd_soc_codec *codec, pm_message_t state)
1285{ 1273{
1286 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1287 struct snd_soc_codec *codec = socdev->card->codec;
1288
1289 wm9081_set_bias_level(codec, SND_SOC_BIAS_OFF); 1274 wm9081_set_bias_level(codec, SND_SOC_BIAS_OFF);
1290 1275
1291 return 0; 1276 return 0;
1292} 1277}
1293 1278
1294static int wm9081_resume(struct platform_device *pdev) 1279static int wm9081_resume(struct snd_soc_codec *codec)
1295{ 1280{
1296 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1297 struct snd_soc_codec *codec = socdev->card->codec;
1298 u16 *reg_cache = codec->reg_cache; 1281 u16 *reg_cache = codec->reg_cache;
1299 int i; 1282 int i;
1300 1283
1301 for (i = 0; i < codec->reg_cache_size; i++) { 1284 for (i = 0; i < codec->driver->reg_cache_size; i++) {
1302 if (i == WM9081_SOFTWARE_RESET) 1285 if (i == WM9081_SOFTWARE_RESET)
1303 continue; 1286 continue;
1304 1287
@@ -1314,133 +1297,56 @@ static int wm9081_resume(struct platform_device *pdev)
1314#define wm9081_resume NULL 1297#define wm9081_resume NULL
1315#endif 1298#endif
1316 1299
1317struct snd_soc_codec_device soc_codec_dev_wm9081 = { 1300static struct snd_soc_codec_driver soc_codec_dev_wm9081 = {
1318 .probe = wm9081_probe, 1301 .probe = wm9081_probe,
1319 .remove = wm9081_remove, 1302 .remove = wm9081_remove,
1320 .suspend = wm9081_suspend, 1303 .suspend = wm9081_suspend,
1321 .resume = wm9081_resume, 1304 .resume = wm9081_resume,
1322};
1323EXPORT_SYMBOL_GPL(soc_codec_dev_wm9081);
1324 1305
1325static int wm9081_register(struct wm9081_priv *wm9081, 1306 .set_sysclk = wm9081_set_sysclk,
1326 enum snd_soc_control_type control) 1307 .set_bias_level = wm9081_set_bias_level,
1327{
1328 struct snd_soc_codec *codec = &wm9081->codec;
1329 int ret;
1330 u16 reg;
1331
1332 if (wm9081_codec) {
1333 dev_err(codec->dev, "Another WM9081 is registered\n");
1334 ret = -EINVAL;
1335 goto err;
1336 }
1337
1338 mutex_init(&codec->mutex);
1339 INIT_LIST_HEAD(&codec->dapm_widgets);
1340 INIT_LIST_HEAD(&codec->dapm_paths);
1341
1342 snd_soc_codec_set_drvdata(codec, wm9081);
1343 codec->name = "WM9081";
1344 codec->owner = THIS_MODULE;
1345 codec->dai = &wm9081_dai;
1346 codec->num_dai = 1;
1347 codec->reg_cache_size = ARRAY_SIZE(wm9081->reg_cache);
1348 codec->reg_cache = &wm9081->reg_cache;
1349 codec->bias_level = SND_SOC_BIAS_OFF;
1350 codec->set_bias_level = wm9081_set_bias_level;
1351 codec->volatile_register = wm9081_volatile_register;
1352
1353 memcpy(codec->reg_cache, wm9081_reg_defaults,
1354 sizeof(wm9081_reg_defaults));
1355
1356 ret = snd_soc_codec_set_cache_io(codec, 8, 16, control);
1357 if (ret != 0) {
1358 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
1359 goto err;
1360 }
1361
1362 reg = snd_soc_read(codec, WM9081_SOFTWARE_RESET);
1363 if (reg != 0x9081) {
1364 dev_err(codec->dev, "Device is not a WM9081: ID=0x%x\n", reg);
1365 ret = -EINVAL;
1366 goto err;
1367 }
1368
1369 ret = wm9081_reset(codec);
1370 if (ret < 0) {
1371 dev_err(codec->dev, "Failed to issue reset\n");
1372 goto err;
1373 }
1374
1375 wm9081_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
1376
1377 /* Enable zero cross by default */
1378 reg = snd_soc_read(codec, WM9081_ANALOGUE_LINEOUT);
1379 snd_soc_write(codec, WM9081_ANALOGUE_LINEOUT, reg | WM9081_LINEOUTZC);
1380 reg = snd_soc_read(codec, WM9081_ANALOGUE_SPEAKER_PGA);
1381 snd_soc_write(codec, WM9081_ANALOGUE_SPEAKER_PGA,
1382 reg | WM9081_SPKPGAZC);
1383
1384 wm9081_dai.dev = codec->dev;
1385
1386 wm9081_codec = codec;
1387
1388 ret = snd_soc_register_codec(codec);
1389 if (ret != 0) {
1390 dev_err(codec->dev, "Failed to register codec: %d\n", ret);
1391 goto err;
1392 }
1393
1394 ret = snd_soc_register_dai(&wm9081_dai);
1395 if (ret != 0) {
1396 dev_err(codec->dev, "Failed to register DAI: %d\n", ret);
1397 goto err_codec;
1398 }
1399
1400 return 0;
1401 1308
1402err_codec: 1309 .reg_cache_size = ARRAY_SIZE(wm9081_reg_defaults),
1403 snd_soc_unregister_codec(codec); 1310 .reg_word_size = sizeof(u16),
1404err: 1311 .reg_cache_default = wm9081_reg_defaults,
1405 kfree(wm9081); 1312 .volatile_register = wm9081_volatile_register,
1406 return ret;
1407}
1408 1313
1409static void wm9081_unregister(struct wm9081_priv *wm9081) 1314 .dapm_widgets = wm9081_dapm_widgets,
1410{ 1315 .num_dapm_widgets = ARRAY_SIZE(wm9081_dapm_widgets),
1411 wm9081_set_bias_level(&wm9081->codec, SND_SOC_BIAS_OFF); 1316 .dapm_routes = wm9081_audio_paths,
1412 snd_soc_unregister_dai(&wm9081_dai); 1317 .num_dapm_routes = ARRAY_SIZE(wm9081_audio_paths),
1413 snd_soc_unregister_codec(&wm9081->codec); 1318};
1414 kfree(wm9081);
1415 wm9081_codec = NULL;
1416}
1417 1319
1320#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1418static __devinit int wm9081_i2c_probe(struct i2c_client *i2c, 1321static __devinit int wm9081_i2c_probe(struct i2c_client *i2c,
1419 const struct i2c_device_id *id) 1322 const struct i2c_device_id *id)
1420{ 1323{
1421 struct wm9081_priv *wm9081; 1324 struct wm9081_priv *wm9081;
1422 struct snd_soc_codec *codec; 1325 int ret;
1423 1326
1424 wm9081 = kzalloc(sizeof(struct wm9081_priv), GFP_KERNEL); 1327 wm9081 = kzalloc(sizeof(struct wm9081_priv), GFP_KERNEL);
1425 if (wm9081 == NULL) 1328 if (wm9081 == NULL)
1426 return -ENOMEM; 1329 return -ENOMEM;
1427 1330
1428 codec = &wm9081->codec;
1429 codec->hw_write = (hw_write_t)i2c_master_send;
1430 wm9081->retune = i2c->dev.platform_data;
1431
1432 i2c_set_clientdata(i2c, wm9081); 1331 i2c_set_clientdata(i2c, wm9081);
1433 codec->control_data = i2c; 1332 wm9081->control_type = SND_SOC_I2C;
1333 wm9081->control_data = i2c;
1434 1334
1435 codec->dev = &i2c->dev; 1335 if (dev_get_platdata(&i2c->dev))
1336 memcpy(&wm9081->pdata, dev_get_platdata(&i2c->dev),
1337 sizeof(wm9081->pdata));
1436 1338
1437 return wm9081_register(wm9081, SND_SOC_I2C); 1339 ret = snd_soc_register_codec(&i2c->dev,
1340 &soc_codec_dev_wm9081, &wm9081_dai, 1);
1341 if (ret < 0)
1342 kfree(wm9081);
1343 return ret;
1438} 1344}
1439 1345
1440static __devexit int wm9081_i2c_remove(struct i2c_client *client) 1346static __devexit int wm9081_i2c_remove(struct i2c_client *client)
1441{ 1347{
1442 struct wm9081_priv *wm9081 = i2c_get_clientdata(client); 1348 snd_soc_unregister_codec(&client->dev);
1443 wm9081_unregister(wm9081); 1349 kfree(i2c_get_clientdata(client));
1444 return 0; 1350 return 0;
1445} 1351}
1446 1352
@@ -1459,24 +1365,27 @@ static struct i2c_driver wm9081_i2c_driver = {
1459 .remove = __devexit_p(wm9081_i2c_remove), 1365 .remove = __devexit_p(wm9081_i2c_remove),
1460 .id_table = wm9081_i2c_id, 1366 .id_table = wm9081_i2c_id,
1461}; 1367};
1368#endif
1462 1369
1463static int __init wm9081_modinit(void) 1370static int __init wm9081_modinit(void)
1464{ 1371{
1465 int ret; 1372 int ret = 0;
1466 1373#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1467 ret = i2c_add_driver(&wm9081_i2c_driver); 1374 ret = i2c_add_driver(&wm9081_i2c_driver);
1468 if (ret != 0) { 1375 if (ret != 0) {
1469 printk(KERN_ERR "Failed to register WM9081 I2C driver: %d\n", 1376 printk(KERN_ERR "Failed to register WM9081 I2C driver: %d\n",
1470 ret); 1377 ret);
1471 } 1378 }
1472 1379#endif
1473 return ret; 1380 return ret;
1474} 1381}
1475module_init(wm9081_modinit); 1382module_init(wm9081_modinit);
1476 1383
1477static void __exit wm9081_exit(void) 1384static void __exit wm9081_exit(void)
1478{ 1385{
1386#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1479 i2c_del_driver(&wm9081_i2c_driver); 1387 i2c_del_driver(&wm9081_i2c_driver);
1388#endif
1480} 1389}
1481module_exit(wm9081_exit); 1390module_exit(wm9081_exit);
1482 1391