diff options
Diffstat (limited to 'sound/soc/codecs/wm9081.c')
-rw-r--r-- | sound/soc/codecs/wm9081.c | 297 |
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 | ||
158 | struct wm9081_priv { | 158 | struct 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 | ||
173 | static int wm9081_volatile_register(unsigned int reg) | 173 | static 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 | */ |
311 | static int speaker_mode_put(struct snd_kcontrol *kcontrol, | 311 | static int speaker_mode_put(struct snd_kcontrol *kcontrol, |
@@ -389,27 +389,6 @@ SOC_DAPM_SINGLE("IN2 Switch", WM9081_ANALOGUE_MIXER, 2, 1, 0), | |||
389 | SOC_DAPM_SINGLE("Playback Switch", WM9081_ANALOGUE_MIXER, 4, 1, 0), | 389 | SOC_DAPM_SINGLE("Playback Switch", WM9081_ANALOGUE_MIXER, 4, 1, 0), |
390 | }; | 390 | }; |
391 | 391 | ||
392 | static 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 | |||
413 | struct _fll_div { | 392 | struct _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 | ||
744 | SND_SOC_DAPM_PGA("LINEOUT PGA", WM9081_POWER_MANAGEMENT, 4, 0, NULL, 0), | 727 | SND_SOC_DAPM_PGA("LINEOUT PGA", WM9081_POWER_MANAGEMENT, 4, 0, NULL, 0), |
745 | 728 | ||
746 | SND_SOC_DAPM_PGA_E("Speaker PGA", WM9081_POWER_MANAGEMENT, 2, 0, NULL, 0, | 729 | SND_SOC_DAPM_PGA("Speaker PGA", WM9081_POWER_MANAGEMENT, 2, 0, NULL, 0), |
747 | speaker_event, | 730 | SND_SOC_DAPM_PGA("Speaker", WM9081_POWER_MANAGEMENT, 1, 0, NULL, 0), |
748 | SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD), | ||
749 | 731 | ||
750 | SND_SOC_DAPM_OUTPUT("LINEOUT"), | 732 | SND_SOC_DAPM_OUTPUT("LINEOUT"), |
751 | SND_SOC_DAPM_OUTPUT("SPKN"), | 733 | SND_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 | ||
761 | static const struct snd_soc_dapm_route audio_paths[] = { | 743 | static 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 | ||
783 | static int wm9081_set_bias_level(struct snd_soc_codec *codec, | 767 | static 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 | ||
1138 | static int wm9081_set_sysclk(struct snd_soc_dai *codec_dai, | 1123 | static 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 | ||
1204 | static struct snd_soc_dai_ops wm9081_dai_ops = { | 1188 | static 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 | */ |
1215 | struct snd_soc_dai wm9081_dai = { | 1198 | static 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 | }; |
1226 | EXPORT_SYMBOL_GPL(wm9081_dai); | ||
1227 | 1209 | ||
1228 | 1210 | static int wm9081_probe(struct snd_soc_codec *codec) | |
1229 | static struct snd_soc_codec *wm9081_codec; | ||
1230 | |||
1231 | static 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 | |||
1269 | pcm_err: | ||
1270 | return ret; | 1262 | return ret; |
1271 | } | 1263 | } |
1272 | 1264 | ||
1273 | static int wm9081_remove(struct platform_device *pdev) | 1265 | static 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 |
1284 | static int wm9081_suspend(struct platform_device *pdev, pm_message_t state) | 1272 | static 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 | ||
1294 | static int wm9081_resume(struct platform_device *pdev) | 1279 | static 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 | ||
1317 | struct snd_soc_codec_device soc_codec_dev_wm9081 = { | 1300 | static 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 | }; | ||
1323 | EXPORT_SYMBOL_GPL(soc_codec_dev_wm9081); | ||
1324 | 1305 | ||
1325 | static 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 | ||
1402 | err_codec: | 1309 | .reg_cache_size = ARRAY_SIZE(wm9081_reg_defaults), |
1403 | snd_soc_unregister_codec(codec); | 1310 | .reg_word_size = sizeof(u16), |
1404 | err: | 1311 | .reg_cache_default = wm9081_reg_defaults, |
1405 | kfree(wm9081); | 1312 | .volatile_register = wm9081_volatile_register, |
1406 | return ret; | ||
1407 | } | ||
1408 | 1313 | ||
1409 | static 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) | ||
1418 | static __devinit int wm9081_i2c_probe(struct i2c_client *i2c, | 1321 | static __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 | ||
1440 | static __devexit int wm9081_i2c_remove(struct i2c_client *client) | 1346 | static __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 | ||
1463 | static int __init wm9081_modinit(void) | 1370 | static 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 | } |
1475 | module_init(wm9081_modinit); | 1382 | module_init(wm9081_modinit); |
1476 | 1383 | ||
1477 | static void __exit wm9081_exit(void) | 1384 | static 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 | } |
1481 | module_exit(wm9081_exit); | 1390 | module_exit(wm9081_exit); |
1482 | 1391 | ||