aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm8978.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm8978.c')
-rw-r--r--sound/soc/codecs/wm8978.c222
1 files changed, 72 insertions, 150 deletions
diff --git a/sound/soc/codecs/wm8978.c b/sound/soc/codecs/wm8978.c
index 8a1ad778e7e3..85e3e630e763 100644
--- a/sound/soc/codecs/wm8978.c
+++ b/sound/soc/codecs/wm8978.c
@@ -24,15 +24,12 @@
24#include <sound/pcm.h> 24#include <sound/pcm.h>
25#include <sound/pcm_params.h> 25#include <sound/pcm_params.h>
26#include <sound/soc.h> 26#include <sound/soc.h>
27#include <sound/soc-dapm.h>
28#include <sound/initval.h> 27#include <sound/initval.h>
29#include <sound/tlv.h> 28#include <sound/tlv.h>
30#include <asm/div64.h> 29#include <asm/div64.h>
31 30
32#include "wm8978.h" 31#include "wm8978.h"
33 32
34static struct snd_soc_codec *wm8978_codec;
35
36/* wm8978 register cache. Note that register 0 is not included in the cache. */ 33/* wm8978 register cache. Note that register 0 is not included in the cache. */
37static const u16 wm8978_reg[WM8978_CACHEREGNUM] = { 34static const u16 wm8978_reg[WM8978_CACHEREGNUM] = {
38 0x0000, 0x0000, 0x0000, 0x0000, /* 0x00...0x03 */ 35 0x0000, 0x0000, 0x0000, 0x0000, /* 0x00...0x03 */
@@ -54,14 +51,14 @@ static const u16 wm8978_reg[WM8978_CACHEREGNUM] = {
54 51
55/* codec private data */ 52/* codec private data */
56struct wm8978_priv { 53struct wm8978_priv {
57 struct snd_soc_codec codec; 54 enum snd_soc_control_type control_type;
55 void *control_data;
58 unsigned int f_pllout; 56 unsigned int f_pllout;
59 unsigned int f_mclk; 57 unsigned int f_mclk;
60 unsigned int f_256fs; 58 unsigned int f_256fs;
61 unsigned int f_opclk; 59 unsigned int f_opclk;
62 int mclk_idx; 60 int mclk_idx;
63 enum wm8978_sysclk_src sysclk; 61 enum wm8978_sysclk_src sysclk;
64 u16 reg_cache[WM8978_CACHEREGNUM];
65}; 62};
66 63
67static const char *wm8978_companding[] = {"Off", "NC", "u-law", "A-law"}; 64static const char *wm8978_companding[] = {"Off", "NC", "u-law", "A-law"};
@@ -96,6 +93,7 @@ static const DECLARE_TLV_DB_SCALE(eq_tlv, -1200, 100, 0);
96static const DECLARE_TLV_DB_SCALE(inpga_tlv, -1200, 75, 0); 93static const DECLARE_TLV_DB_SCALE(inpga_tlv, -1200, 75, 0);
97static const DECLARE_TLV_DB_SCALE(spk_tlv, -5700, 100, 0); 94static const DECLARE_TLV_DB_SCALE(spk_tlv, -5700, 100, 0);
98static const DECLARE_TLV_DB_SCALE(boost_tlv, -1500, 300, 1); 95static const DECLARE_TLV_DB_SCALE(boost_tlv, -1500, 300, 1);
96static const DECLARE_TLV_DB_SCALE(limiter_tlv, 0, 100, 0);
99 97
100static const struct snd_kcontrol_new wm8978_snd_controls[] = { 98static const struct snd_kcontrol_new wm8978_snd_controls[] = {
101 99
@@ -147,19 +145,19 @@ static const struct snd_kcontrol_new wm8978_snd_controls[] = {
147 145
148 SOC_SINGLE("DAC Playback Limiter Threshold", 146 SOC_SINGLE("DAC Playback Limiter Threshold",
149 WM8978_DAC_LIMITER_2, 4, 7, 0), 147 WM8978_DAC_LIMITER_2, 4, 7, 0),
150 SOC_SINGLE("DAC Playback Limiter Boost", 148 SOC_SINGLE_TLV("DAC Playback Limiter Volume",
151 WM8978_DAC_LIMITER_2, 0, 15, 0), 149 WM8978_DAC_LIMITER_2, 0, 12, 0, limiter_tlv),
152 150
153 SOC_ENUM("ALC Enable Switch", alc1), 151 SOC_ENUM("ALC Enable Switch", alc1),
154 SOC_SINGLE("ALC Capture Min Gain", WM8978_ALC_CONTROL_1, 0, 7, 0), 152 SOC_SINGLE("ALC Capture Min Gain", WM8978_ALC_CONTROL_1, 0, 7, 0),
155 SOC_SINGLE("ALC Capture Max Gain", WM8978_ALC_CONTROL_1, 3, 7, 0), 153 SOC_SINGLE("ALC Capture Max Gain", WM8978_ALC_CONTROL_1, 3, 7, 0),
156 154
157 SOC_SINGLE("ALC Capture Hold", WM8978_ALC_CONTROL_2, 4, 7, 0), 155 SOC_SINGLE("ALC Capture Hold", WM8978_ALC_CONTROL_2, 4, 10, 0),
158 SOC_SINGLE("ALC Capture Target", WM8978_ALC_CONTROL_2, 0, 15, 0), 156 SOC_SINGLE("ALC Capture Target", WM8978_ALC_CONTROL_2, 0, 15, 0),
159 157
160 SOC_ENUM("ALC Capture Mode", alc3), 158 SOC_ENUM("ALC Capture Mode", alc3),
161 SOC_SINGLE("ALC Capture Decay", WM8978_ALC_CONTROL_3, 4, 15, 0), 159 SOC_SINGLE("ALC Capture Decay", WM8978_ALC_CONTROL_3, 4, 10, 0),
162 SOC_SINGLE("ALC Capture Attack", WM8978_ALC_CONTROL_3, 0, 15, 0), 160 SOC_SINGLE("ALC Capture Attack", WM8978_ALC_CONTROL_3, 0, 10, 0),
163 161
164 SOC_SINGLE("ALC Capture Noise Gate Switch", WM8978_NOISE_GATE, 3, 1, 0), 162 SOC_SINGLE("ALC Capture Noise Gate Switch", WM8978_NOISE_GATE, 3, 1, 0),
165 SOC_SINGLE("ALC Capture Noise Gate Threshold", 163 SOC_SINGLE("ALC Capture Noise Gate Threshold",
@@ -214,8 +212,10 @@ static const struct snd_kcontrol_new wm8978_snd_controls[] = {
214 WM8978_LOUT2_SPK_CONTROL, WM8978_ROUT2_SPK_CONTROL, 6, 1, 1), 212 WM8978_LOUT2_SPK_CONTROL, WM8978_ROUT2_SPK_CONTROL, 6, 1, 1),
215 213
216 /* DAC / ADC oversampling */ 214 /* DAC / ADC oversampling */
217 SOC_SINGLE("DAC 128x Oversampling Switch", WM8978_DAC_CONTROL, 8, 1, 0), 215 SOC_SINGLE("DAC 128x Oversampling Switch", WM8978_DAC_CONTROL,
218 SOC_SINGLE("ADC 128x Oversampling Switch", WM8978_ADC_CONTROL, 8, 1, 0), 216 5, 1, 0),
217 SOC_SINGLE("ADC 128x Oversampling Switch", WM8978_ADC_CONTROL,
218 5, 1, 0),
219}; 219};
220 220
221/* Mixer #1: Output (OUT1, OUT2) Mixer: mix AUX, Input mixer output and DAC */ 221/* Mixer #1: Output (OUT1, OUT2) Mixer: mix AUX, Input mixer output and DAC */
@@ -356,11 +356,12 @@ static const struct snd_soc_dapm_route audio_map[] = {
356 356
357static int wm8978_add_widgets(struct snd_soc_codec *codec) 357static int wm8978_add_widgets(struct snd_soc_codec *codec)
358{ 358{
359 snd_soc_dapm_new_controls(codec, wm8978_dapm_widgets, 359 struct snd_soc_dapm_context *dapm = &codec->dapm;
360 ARRAY_SIZE(wm8978_dapm_widgets));
361 360
361 snd_soc_dapm_new_controls(dapm, wm8978_dapm_widgets,
362 ARRAY_SIZE(wm8978_dapm_widgets));
362 /* set up the WM8978 audio map */ 363 /* set up the WM8978 audio map */
363 snd_soc_dapm_add_routes(codec, audio_map, ARRAY_SIZE(audio_map)); 364 snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map));
364 365
365 return 0; 366 return 0;
366} 367}
@@ -374,8 +375,8 @@ struct wm8978_pll_div {
374 375
375#define FIXED_PLL_SIZE (1 << 24) 376#define FIXED_PLL_SIZE (1 << 24)
376 377
377static void pll_factors(struct wm8978_pll_div *pll_div, unsigned int target, 378static void pll_factors(struct snd_soc_codec *codec,
378 unsigned int source) 379 struct wm8978_pll_div *pll_div, unsigned int target, unsigned int source)
379{ 380{
380 u64 k_part; 381 u64 k_part;
381 unsigned int k, n_div, n_mod; 382 unsigned int k, n_div, n_mod;
@@ -390,7 +391,7 @@ static void pll_factors(struct wm8978_pll_div *pll_div, unsigned int target,
390 } 391 }
391 392
392 if (n_div < 6 || n_div > 12) 393 if (n_div < 6 || n_div > 12)
393 dev_warn(wm8978_codec->dev, 394 dev_warn(codec->dev,
394 "WM8978 N value exceeds recommended range! N = %u\n", 395 "WM8978 N value exceeds recommended range! N = %u\n",
395 n_div); 396 n_div);
396 397
@@ -505,7 +506,7 @@ static int wm8978_configure_pll(struct snd_soc_codec *codec)
505 dev_dbg(codec->dev, "%s: f_MCLK=%uHz, f_PLLOUT=%uHz\n", __func__, 506 dev_dbg(codec->dev, "%s: f_MCLK=%uHz, f_PLLOUT=%uHz\n", __func__,
506 wm8978->f_mclk, wm8978->f_pllout); 507 wm8978->f_mclk, wm8978->f_pllout);
507 508
508 pll_factors(&pll_div, f2, wm8978->f_mclk); 509 pll_factors(codec, &pll_div, f2, wm8978->f_mclk);
509 510
510 dev_dbg(codec->dev, "%s: calculated PLL N=0x%x, K=0x%x, div2=%d\n", 511 dev_dbg(codec->dev, "%s: calculated PLL N=0x%x, K=0x%x, div2=%d\n",
511 __func__, pll_div.n, pll_div.k, pll_div.div2); 512 __func__, pll_div.n, pll_div.k, pll_div.div2);
@@ -690,8 +691,7 @@ static int wm8978_hw_params(struct snd_pcm_substream *substream,
690 struct snd_soc_dai *dai) 691 struct snd_soc_dai *dai)
691{ 692{
692 struct snd_soc_pcm_runtime *rtd = substream->private_data; 693 struct snd_soc_pcm_runtime *rtd = substream->private_data;
693 struct snd_soc_device *socdev = rtd->socdev; 694 struct snd_soc_codec *codec = rtd->codec;
694 struct snd_soc_codec *codec = socdev->card->codec;
695 struct wm8978_priv *wm8978 = snd_soc_codec_get_drvdata(codec); 695 struct wm8978_priv *wm8978 = snd_soc_codec_get_drvdata(codec);
696 /* Word length mask = 0x60 */ 696 /* Word length mask = 0x60 */
697 u16 iface_ctl = snd_soc_read(codec, WM8978_AUDIO_INTERFACE) & ~0x60; 697 u16 iface_ctl = snd_soc_read(codec, WM8978_AUDIO_INTERFACE) & ~0x60;
@@ -839,7 +839,7 @@ static int wm8978_set_bias_level(struct snd_soc_codec *codec,
839 /* bit 3: enable bias, bit 2: enable I/O tie off buffer */ 839 /* bit 3: enable bias, bit 2: enable I/O tie off buffer */
840 power1 |= 0xc; 840 power1 |= 0xc;
841 841
842 if (codec->bias_level == SND_SOC_BIAS_OFF) { 842 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
843 /* Initial cap charge at VMID 5k */ 843 /* Initial cap charge at VMID 5k */
844 snd_soc_write(codec, WM8978_POWER_MANAGEMENT_1, 844 snd_soc_write(codec, WM8978_POWER_MANAGEMENT_1,
845 power1 | 0x3); 845 power1 | 0x3);
@@ -859,7 +859,7 @@ static int wm8978_set_bias_level(struct snd_soc_codec *codec,
859 859
860 dev_dbg(codec->dev, "%s: %d, %x\n", __func__, level, power1); 860 dev_dbg(codec->dev, "%s: %d, %x\n", __func__, level, power1);
861 861
862 codec->bias_level = level; 862 codec->dapm.bias_level = level;
863 return 0; 863 return 0;
864} 864}
865 865
@@ -875,9 +875,8 @@ static struct snd_soc_dai_ops wm8978_dai_ops = {
875}; 875};
876 876
877/* Also supports 12kHz */ 877/* Also supports 12kHz */
878struct snd_soc_dai wm8978_dai = { 878static struct snd_soc_dai_driver wm8978_dai = {
879 .name = "WM8978 HiFi", 879 .name = "wm8978-hifi",
880 .id = 1,
881 .playback = { 880 .playback = {
882 .stream_name = "Playback", 881 .stream_name = "Playback",
883 .channels_min = 1, 882 .channels_min = 1,
@@ -894,13 +893,9 @@ struct snd_soc_dai wm8978_dai = {
894 }, 893 },
895 .ops = &wm8978_dai_ops, 894 .ops = &wm8978_dai_ops,
896}; 895};
897EXPORT_SYMBOL_GPL(wm8978_dai);
898 896
899static int wm8978_suspend(struct platform_device *pdev, pm_message_t state) 897static int wm8978_suspend(struct snd_soc_codec *codec, pm_message_t state)
900{ 898{
901 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
902 struct snd_soc_codec *codec = socdev->card->codec;
903
904 wm8978_set_bias_level(codec, SND_SOC_BIAS_OFF); 899 wm8978_set_bias_level(codec, SND_SOC_BIAS_OFF);
905 /* Also switch PLL off */ 900 /* Also switch PLL off */
906 snd_soc_write(codec, WM8978_POWER_MANAGEMENT_1, 0); 901 snd_soc_write(codec, WM8978_POWER_MANAGEMENT_1, 0);
@@ -908,10 +903,8 @@ static int wm8978_suspend(struct platform_device *pdev, pm_message_t state)
908 return 0; 903 return 0;
909} 904}
910 905
911static int wm8978_resume(struct platform_device *pdev) 906static int wm8978_resume(struct snd_soc_codec *codec)
912{ 907{
913 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
914 struct snd_soc_codec *codec = socdev->card->codec;
915 struct wm8978_priv *wm8978 = snd_soc_codec_get_drvdata(codec); 908 struct wm8978_priv *wm8978 = snd_soc_codec_get_drvdata(codec);
916 int i; 909 int i;
917 u16 *cache = codec->reg_cache; 910 u16 *cache = codec->reg_cache;
@@ -933,54 +926,6 @@ static int wm8978_resume(struct platform_device *pdev)
933 return 0; 926 return 0;
934} 927}
935 928
936static int wm8978_probe(struct platform_device *pdev)
937{
938 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
939 struct snd_soc_codec *codec;
940 int ret = 0;
941
942 if (wm8978_codec == NULL) {
943 dev_err(&pdev->dev, "Codec device not registered\n");
944 return -ENODEV;
945 }
946
947 socdev->card->codec = wm8978_codec;
948 codec = wm8978_codec;
949
950 /* register pcms */
951 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1);
952 if (ret < 0) {
953 dev_err(codec->dev, "failed to create pcms: %d\n", ret);
954 goto pcm_err;
955 }
956
957 snd_soc_add_controls(codec, wm8978_snd_controls,
958 ARRAY_SIZE(wm8978_snd_controls));
959 wm8978_add_widgets(codec);
960
961pcm_err:
962 return ret;
963}
964
965/* power down chip */
966static int wm8978_remove(struct platform_device *pdev)
967{
968 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
969
970 snd_soc_free_pcms(socdev);
971 snd_soc_dapm_free(socdev);
972
973 return 0;
974}
975
976struct snd_soc_codec_device soc_codec_dev_wm8978 = {
977 .probe = wm8978_probe,
978 .remove = wm8978_remove,
979 .suspend = wm8978_suspend,
980 .resume = wm8978_resume,
981};
982EXPORT_SYMBOL_GPL(soc_codec_dev_wm8978);
983
984/* 929/*
985 * These registers contain an "update" bit - bit 8. This means, for example, 930 * These registers contain an "update" bit - bit 8. This means, for example,
986 * that one can write new DAC digital volume for both channels, but only when 931 * that one can write new DAC digital volume for both channels, but only when
@@ -1000,124 +945,90 @@ static const int update_reg[] = {
1000 WM8978_ROUT2_SPK_CONTROL, 945 WM8978_ROUT2_SPK_CONTROL,
1001}; 946};
1002 947
1003static __devinit int wm8978_register(struct wm8978_priv *wm8978) 948static int wm8978_probe(struct snd_soc_codec *codec)
1004{ 949{
1005 int ret, i; 950 struct wm8978_priv *wm8978 = snd_soc_codec_get_drvdata(codec);
1006 struct snd_soc_codec *codec = &wm8978->codec; 951 int ret = 0, i;
1007
1008 if (wm8978_codec) {
1009 dev_err(codec->dev, "Another WM8978 is registered\n");
1010 return -EINVAL;
1011 }
1012 952
1013 /* 953 /*
1014 * Set default system clock to PLL, it is more precise, this is also the 954 * Set default system clock to PLL, it is more precise, this is also the
1015 * default hardware setting 955 * default hardware setting
1016 */ 956 */
1017 wm8978->sysclk = WM8978_PLL; 957 wm8978->sysclk = WM8978_PLL;
1018 958 codec->control_data = wm8978->control_data;
1019 mutex_init(&codec->mutex);
1020 INIT_LIST_HEAD(&codec->dapm_widgets);
1021 INIT_LIST_HEAD(&codec->dapm_paths);
1022
1023 snd_soc_codec_set_drvdata(codec, wm8978);
1024 codec->name = "WM8978";
1025 codec->owner = THIS_MODULE;
1026 codec->bias_level = SND_SOC_BIAS_OFF;
1027 codec->set_bias_level = wm8978_set_bias_level;
1028 codec->dai = &wm8978_dai;
1029 codec->num_dai = 1;
1030 codec->reg_cache_size = WM8978_CACHEREGNUM;
1031 codec->reg_cache = &wm8978->reg_cache;
1032
1033 ret = snd_soc_codec_set_cache_io(codec, 7, 9, SND_SOC_I2C); 959 ret = snd_soc_codec_set_cache_io(codec, 7, 9, SND_SOC_I2C);
1034 if (ret < 0) { 960 if (ret < 0) {
1035 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret); 961 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
1036 goto err; 962 return ret;
1037 } 963 }
1038 964
1039 memcpy(codec->reg_cache, wm8978_reg, sizeof(wm8978_reg));
1040
1041 /* 965 /*
1042 * Set the update bit in all registers, that have one. This way all 966 * Set the update bit in all registers, that have one. This way all
1043 * writes to those registers will also cause the update bit to be 967 * writes to those registers will also cause the update bit to be
1044 * written. 968 * written.
1045 */ 969 */
1046 for (i = 0; i < ARRAY_SIZE(update_reg); i++) 970 for (i = 0; i < ARRAY_SIZE(update_reg); i++)
1047 ((u16 *)codec->reg_cache)[update_reg[i]] |= 0x100; 971 snd_soc_update_bits(codec, update_reg[i], 0x100, 0x100);
1048 972
1049 /* Reset the codec */ 973 /* Reset the codec */
1050 ret = snd_soc_write(codec, WM8978_RESET, 0); 974 ret = snd_soc_write(codec, WM8978_RESET, 0);
1051 if (ret < 0) { 975 if (ret < 0) {
1052 dev_err(codec->dev, "Failed to issue reset\n"); 976 dev_err(codec->dev, "Failed to issue reset\n");
1053 goto err; 977 return ret;
1054 } 978 }
1055 979
1056 wm8978_dai.dev = codec->dev;
1057
1058 wm8978_set_bias_level(codec, SND_SOC_BIAS_STANDBY); 980 wm8978_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
1059 981
1060 wm8978_codec = codec; 982 snd_soc_add_controls(codec, wm8978_snd_controls,
1061 983 ARRAY_SIZE(wm8978_snd_controls));
1062 ret = snd_soc_register_codec(codec); 984 wm8978_add_widgets(codec);
1063 if (ret != 0) {
1064 dev_err(codec->dev, "Failed to register codec: %d\n", ret);
1065 goto err;
1066 }
1067
1068 ret = snd_soc_register_dai(&wm8978_dai);
1069 if (ret != 0) {
1070 dev_err(codec->dev, "Failed to register DAI: %d\n", ret);
1071 goto err_codec;
1072 }
1073 985
1074 return 0; 986 return 0;
1075
1076err_codec:
1077 snd_soc_unregister_codec(codec);
1078err:
1079 return ret;
1080} 987}
1081 988
1082static __devexit void wm8978_unregister(struct wm8978_priv *wm8978) 989/* power down chip */
990static int wm8978_remove(struct snd_soc_codec *codec)
1083{ 991{
1084 wm8978_set_bias_level(&wm8978->codec, SND_SOC_BIAS_OFF); 992 wm8978_set_bias_level(codec, SND_SOC_BIAS_OFF);
1085 snd_soc_unregister_dai(&wm8978_dai); 993 return 0;
1086 snd_soc_unregister_codec(&wm8978->codec);
1087 wm8978_codec = NULL;
1088} 994}
1089 995
996static struct snd_soc_codec_driver soc_codec_dev_wm8978 = {
997 .probe = wm8978_probe,
998 .remove = wm8978_remove,
999 .suspend = wm8978_suspend,
1000 .resume = wm8978_resume,
1001 .set_bias_level = wm8978_set_bias_level,
1002 .reg_cache_size = ARRAY_SIZE(wm8978_reg),
1003 .reg_word_size = sizeof(u16),
1004 .reg_cache_default = wm8978_reg,
1005};
1006
1007#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1090static __devinit int wm8978_i2c_probe(struct i2c_client *i2c, 1008static __devinit int wm8978_i2c_probe(struct i2c_client *i2c,
1091 const struct i2c_device_id *id) 1009 const struct i2c_device_id *id)
1092{ 1010{
1093 int ret;
1094 struct wm8978_priv *wm8978; 1011 struct wm8978_priv *wm8978;
1095 struct snd_soc_codec *codec; 1012 int ret;
1096 1013
1097 wm8978 = kzalloc(sizeof(struct wm8978_priv), GFP_KERNEL); 1014 wm8978 = kzalloc(sizeof(struct wm8978_priv), GFP_KERNEL);
1098 if (wm8978 == NULL) 1015 if (wm8978 == NULL)
1099 return -ENOMEM; 1016 return -ENOMEM;
1100 1017
1101 codec = &wm8978->codec;
1102 codec->hw_write = (hw_write_t)i2c_master_send;
1103
1104 i2c_set_clientdata(i2c, wm8978); 1018 i2c_set_clientdata(i2c, wm8978);
1105 codec->control_data = i2c; 1019 wm8978->control_data = i2c;
1106
1107 codec->dev = &i2c->dev;
1108 1020
1109 ret = wm8978_register(wm8978); 1021 ret = snd_soc_register_codec(&i2c->dev,
1022 &soc_codec_dev_wm8978, &wm8978_dai, 1);
1110 if (ret < 0) 1023 if (ret < 0)
1111 kfree(wm8978); 1024 kfree(wm8978);
1112
1113 return ret; 1025 return ret;
1114} 1026}
1115 1027
1116static __devexit int wm8978_i2c_remove(struct i2c_client *client) 1028static __devexit int wm8978_i2c_remove(struct i2c_client *client)
1117{ 1029{
1118 struct wm8978_priv *wm8978 = i2c_get_clientdata(client); 1030 snd_soc_unregister_codec(&client->dev);
1119 wm8978_unregister(wm8978); 1031 kfree(i2c_get_clientdata(client));
1120 kfree(wm8978);
1121 return 0; 1032 return 0;
1122} 1033}
1123 1034
@@ -1129,23 +1040,34 @@ MODULE_DEVICE_TABLE(i2c, wm8978_i2c_id);
1129 1040
1130static struct i2c_driver wm8978_i2c_driver = { 1041static struct i2c_driver wm8978_i2c_driver = {
1131 .driver = { 1042 .driver = {
1132 .name = "WM8978", 1043 .name = "wm8978",
1133 .owner = THIS_MODULE, 1044 .owner = THIS_MODULE,
1134 }, 1045 },
1135 .probe = wm8978_i2c_probe, 1046 .probe = wm8978_i2c_probe,
1136 .remove = __devexit_p(wm8978_i2c_remove), 1047 .remove = __devexit_p(wm8978_i2c_remove),
1137 .id_table = wm8978_i2c_id, 1048 .id_table = wm8978_i2c_id,
1138}; 1049};
1050#endif
1139 1051
1140static int __init wm8978_modinit(void) 1052static int __init wm8978_modinit(void)
1141{ 1053{
1142 return i2c_add_driver(&wm8978_i2c_driver); 1054 int ret = 0;
1055#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1056 ret = i2c_add_driver(&wm8978_i2c_driver);
1057 if (ret != 0) {
1058 printk(KERN_ERR "Failed to register WM8978 I2C driver: %d\n",
1059 ret);
1060 }
1061#endif
1062 return ret;
1143} 1063}
1144module_init(wm8978_modinit); 1064module_init(wm8978_modinit);
1145 1065
1146static void __exit wm8978_exit(void) 1066static void __exit wm8978_exit(void)
1147{ 1067{
1068#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1148 i2c_del_driver(&wm8978_i2c_driver); 1069 i2c_del_driver(&wm8978_i2c_driver);
1070#endif
1149} 1071}
1150module_exit(wm8978_exit); 1072module_exit(wm8978_exit);
1151 1073