aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm8960.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm8960.c')
-rw-r--r--sound/soc/codecs/wm8960.c242
1 files changed, 86 insertions, 156 deletions
diff --git a/sound/soc/codecs/wm8960.c b/sound/soc/codecs/wm8960.c
index 3c6ee61f6c95..4393394b7bc1 100644
--- a/sound/soc/codecs/wm8960.c
+++ b/sound/soc/codecs/wm8960.c
@@ -20,7 +20,6 @@
20#include <sound/pcm.h> 20#include <sound/pcm.h>
21#include <sound/pcm_params.h> 21#include <sound/pcm_params.h>
22#include <sound/soc.h> 22#include <sound/soc.h>
23#include <sound/soc-dapm.h>
24#include <sound/initval.h> 23#include <sound/initval.h>
25#include <sound/tlv.h> 24#include <sound/tlv.h>
26#include <sound/wm8960.h> 25#include <sound/wm8960.h>
@@ -29,8 +28,6 @@
29 28
30#define AUDIO_NAME "wm8960" 29#define AUDIO_NAME "wm8960"
31 30
32struct snd_soc_codec_device soc_codec_dev_wm8960;
33
34/* R25 - Power 1 */ 31/* R25 - Power 1 */
35#define WM8960_VMID_MASK 0x180 32#define WM8960_VMID_MASK 0x180
36#define WM8960_VREF 0x40 33#define WM8960_VREF 0x40
@@ -74,8 +71,10 @@ static const u16 wm8960_reg[WM8960_CACHEREGNUM] = {
74}; 71};
75 72
76struct wm8960_priv { 73struct wm8960_priv {
77 u16 reg_cache[WM8960_CACHEREGNUM]; 74 enum snd_soc_control_type control_type;
78 struct snd_soc_codec codec; 75 void *control_data;
76 int (*set_bias_level)(struct snd_soc_codec *,
77 enum snd_soc_bias_level level);
79 struct snd_soc_dapm_widget *lout1; 78 struct snd_soc_dapm_widget *lout1;
80 struct snd_soc_dapm_widget *rout1; 79 struct snd_soc_dapm_widget *rout1;
81 struct snd_soc_dapm_widget *out3; 80 struct snd_soc_dapm_widget *out3;
@@ -137,7 +136,8 @@ static int wm8960_get_deemph(struct snd_kcontrol *kcontrol,
137 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); 136 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
138 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec); 137 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec);
139 138
140 return wm8960->deemph; 139 ucontrol->value.enumerated.item[0] = wm8960->deemph;
140 return 0;
141} 141}
142 142
143static int wm8960_put_deemph(struct snd_kcontrol *kcontrol, 143static int wm8960_put_deemph(struct snd_kcontrol *kcontrol,
@@ -387,27 +387,28 @@ static int wm8960_add_widgets(struct snd_soc_codec *codec)
387{ 387{
388 struct wm8960_data *pdata = codec->dev->platform_data; 388 struct wm8960_data *pdata = codec->dev->platform_data;
389 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec); 389 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec);
390 struct snd_soc_dapm_context *dapm = &codec->dapm;
390 struct snd_soc_dapm_widget *w; 391 struct snd_soc_dapm_widget *w;
391 392
392 snd_soc_dapm_new_controls(codec, wm8960_dapm_widgets, 393 snd_soc_dapm_new_controls(dapm, wm8960_dapm_widgets,
393 ARRAY_SIZE(wm8960_dapm_widgets)); 394 ARRAY_SIZE(wm8960_dapm_widgets));
394 395
395 snd_soc_dapm_add_routes(codec, audio_paths, ARRAY_SIZE(audio_paths)); 396 snd_soc_dapm_add_routes(dapm, audio_paths, ARRAY_SIZE(audio_paths));
396 397
397 /* In capless mode OUT3 is used to provide VMID for the 398 /* In capless mode OUT3 is used to provide VMID for the
398 * headphone outputs, otherwise it is used as a mono mixer. 399 * headphone outputs, otherwise it is used as a mono mixer.
399 */ 400 */
400 if (pdata && pdata->capless) { 401 if (pdata && pdata->capless) {
401 snd_soc_dapm_new_controls(codec, wm8960_dapm_widgets_capless, 402 snd_soc_dapm_new_controls(dapm, wm8960_dapm_widgets_capless,
402 ARRAY_SIZE(wm8960_dapm_widgets_capless)); 403 ARRAY_SIZE(wm8960_dapm_widgets_capless));
403 404
404 snd_soc_dapm_add_routes(codec, audio_paths_capless, 405 snd_soc_dapm_add_routes(dapm, audio_paths_capless,
405 ARRAY_SIZE(audio_paths_capless)); 406 ARRAY_SIZE(audio_paths_capless));
406 } else { 407 } else {
407 snd_soc_dapm_new_controls(codec, wm8960_dapm_widgets_out3, 408 snd_soc_dapm_new_controls(dapm, wm8960_dapm_widgets_out3,
408 ARRAY_SIZE(wm8960_dapm_widgets_out3)); 409 ARRAY_SIZE(wm8960_dapm_widgets_out3));
409 410
410 snd_soc_dapm_add_routes(codec, audio_paths_out3, 411 snd_soc_dapm_add_routes(dapm, audio_paths_out3,
411 ARRAY_SIZE(audio_paths_out3)); 412 ARRAY_SIZE(audio_paths_out3));
412 } 413 }
413 414
@@ -416,7 +417,9 @@ static int wm8960_add_widgets(struct snd_soc_codec *codec)
416 * list each time to find the desired power state do so now 417 * list each time to find the desired power state do so now
417 * and save the result. 418 * and save the result.
418 */ 419 */
419 list_for_each_entry(w, &codec->dapm_widgets, list) { 420 list_for_each_entry(w, &codec->card->widgets, list) {
421 if (w->dapm != &codec->dapm)
422 continue;
420 if (strcmp(w->name, "LOUT1 PGA") == 0) 423 if (strcmp(w->name, "LOUT1 PGA") == 0)
421 wm8960->lout1 = w; 424 wm8960->lout1 = w;
422 if (strcmp(w->name, "ROUT1 PGA") == 0) 425 if (strcmp(w->name, "ROUT1 PGA") == 0)
@@ -507,8 +510,7 @@ static int wm8960_hw_params(struct snd_pcm_substream *substream,
507 struct snd_soc_dai *dai) 510 struct snd_soc_dai *dai)
508{ 511{
509 struct snd_soc_pcm_runtime *rtd = substream->private_data; 512 struct snd_soc_pcm_runtime *rtd = substream->private_data;
510 struct snd_soc_device *socdev = rtd->socdev; 513 struct snd_soc_codec *codec = rtd->codec;
511 struct snd_soc_codec *codec = socdev->card->codec;
512 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec); 514 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec);
513 u16 iface = snd_soc_read(codec, WM8960_IFACE1) & 0xfff3; 515 u16 iface = snd_soc_read(codec, WM8960_IFACE1) & 0xfff3;
514 int i; 516 int i;
@@ -572,7 +574,7 @@ static int wm8960_set_bias_level_out3(struct snd_soc_codec *codec,
572 break; 574 break;
573 575
574 case SND_SOC_BIAS_STANDBY: 576 case SND_SOC_BIAS_STANDBY:
575 if (codec->bias_level == SND_SOC_BIAS_OFF) { 577 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
576 /* Enable anti-pop features */ 578 /* Enable anti-pop features */
577 snd_soc_write(codec, WM8960_APOP1, 579 snd_soc_write(codec, WM8960_APOP1,
578 WM8960_POBCTRL | WM8960_SOFT_ST | 580 WM8960_POBCTRL | WM8960_SOFT_ST |
@@ -610,7 +612,7 @@ static int wm8960_set_bias_level_out3(struct snd_soc_codec *codec,
610 break; 612 break;
611 } 613 }
612 614
613 codec->bias_level = level; 615 codec->dapm.bias_level = level;
614 616
615 return 0; 617 return 0;
616} 618}
@@ -626,7 +628,7 @@ static int wm8960_set_bias_level_capless(struct snd_soc_codec *codec,
626 break; 628 break;
627 629
628 case SND_SOC_BIAS_PREPARE: 630 case SND_SOC_BIAS_PREPARE:
629 switch (codec->bias_level) { 631 switch (codec->dapm.bias_level) {
630 case SND_SOC_BIAS_STANDBY: 632 case SND_SOC_BIAS_STANDBY:
631 /* Enable anti pop mode */ 633 /* Enable anti pop mode */
632 snd_soc_update_bits(codec, WM8960_APOP1, 634 snd_soc_update_bits(codec, WM8960_APOP1,
@@ -681,7 +683,7 @@ static int wm8960_set_bias_level_capless(struct snd_soc_codec *codec,
681 break; 683 break;
682 684
683 case SND_SOC_BIAS_STANDBY: 685 case SND_SOC_BIAS_STANDBY:
684 switch (codec->bias_level) { 686 switch (codec->dapm.bias_level) {
685 case SND_SOC_BIAS_PREPARE: 687 case SND_SOC_BIAS_PREPARE:
686 /* Disable HP discharge */ 688 /* Disable HP discharge */
687 snd_soc_update_bits(codec, WM8960_APOP2, 689 snd_soc_update_bits(codec, WM8960_APOP2,
@@ -705,7 +707,7 @@ static int wm8960_set_bias_level_capless(struct snd_soc_codec *codec,
705 break; 707 break;
706 } 708 }
707 709
708 codec->bias_level = level; 710 codec->dapm.bias_level = level;
709 711
710 return 0; 712 return 0;
711} 713}
@@ -849,6 +851,14 @@ static int wm8960_set_dai_clkdiv(struct snd_soc_dai *codec_dai,
849 return 0; 851 return 0;
850} 852}
851 853
854static int wm8960_set_bias_level(struct snd_soc_codec *codec,
855 enum snd_soc_bias_level level)
856{
857 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec);
858
859 return wm8960->set_bias_level(codec, level);
860}
861
852#define WM8960_RATES SNDRV_PCM_RATE_8000_48000 862#define WM8960_RATES SNDRV_PCM_RATE_8000_48000
853 863
854#define WM8960_FORMATS \ 864#define WM8960_FORMATS \
@@ -863,8 +873,8 @@ static struct snd_soc_dai_ops wm8960_dai_ops = {
863 .set_pll = wm8960_set_dai_pll, 873 .set_pll = wm8960_set_dai_pll,
864}; 874};
865 875
866struct snd_soc_dai wm8960_dai = { 876static struct snd_soc_dai_driver wm8960_dai = {
867 .name = "WM8960", 877 .name = "wm8960-hifi",
868 .playback = { 878 .playback = {
869 .stream_name = "Playback", 879 .stream_name = "Playback",
870 .channels_min = 1, 880 .channels_min = 1,
@@ -880,21 +890,18 @@ struct snd_soc_dai wm8960_dai = {
880 .ops = &wm8960_dai_ops, 890 .ops = &wm8960_dai_ops,
881 .symmetric_rates = 1, 891 .symmetric_rates = 1,
882}; 892};
883EXPORT_SYMBOL_GPL(wm8960_dai);
884 893
885static int wm8960_suspend(struct platform_device *pdev, pm_message_t state) 894static int wm8960_suspend(struct snd_soc_codec *codec, pm_message_t state)
886{ 895{
887 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 896 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec);
888 struct snd_soc_codec *codec = socdev->card->codec;
889 897
890 codec->set_bias_level(codec, SND_SOC_BIAS_OFF); 898 wm8960->set_bias_level(codec, SND_SOC_BIAS_OFF);
891 return 0; 899 return 0;
892} 900}
893 901
894static int wm8960_resume(struct platform_device *pdev) 902static int wm8960_resume(struct snd_soc_codec *codec)
895{ 903{
896 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 904 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec);
897 struct snd_soc_codec *codec = socdev->card->codec;
898 int i; 905 int i;
899 u8 data[2]; 906 u8 data[2];
900 u16 *cache = codec->reg_cache; 907 u16 *cache = codec->reg_cache;
@@ -906,78 +913,19 @@ static int wm8960_resume(struct platform_device *pdev)
906 codec->hw_write(codec->control_data, data, 2); 913 codec->hw_write(codec->control_data, data, 2);
907 } 914 }
908 915
909 codec->set_bias_level(codec, SND_SOC_BIAS_STANDBY); 916 wm8960->set_bias_level(codec, SND_SOC_BIAS_STANDBY);
910
911 return 0; 917 return 0;
912} 918}
913 919
914static struct snd_soc_codec *wm8960_codec; 920static int wm8960_probe(struct snd_soc_codec *codec)
915
916static int wm8960_probe(struct platform_device *pdev)
917{
918 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
919 struct snd_soc_codec *codec;
920 int ret = 0;
921
922 if (wm8960_codec == NULL) {
923 dev_err(&pdev->dev, "Codec device not registered\n");
924 return -ENODEV;
925 }
926
927 socdev->card->codec = wm8960_codec;
928 codec = wm8960_codec;
929
930 /* register pcms */
931 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1);
932 if (ret < 0) {
933 dev_err(codec->dev, "failed to create pcms: %d\n", ret);
934 goto pcm_err;
935 }
936
937 snd_soc_add_controls(codec, wm8960_snd_controls,
938 ARRAY_SIZE(wm8960_snd_controls));
939 wm8960_add_widgets(codec);
940
941 return ret;
942
943pcm_err:
944 return ret;
945}
946
947/* power down chip */
948static int wm8960_remove(struct platform_device *pdev)
949{ 921{
950 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 922 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec);
951 923 struct wm8960_data *pdata = dev_get_platdata(codec->dev);
952 snd_soc_free_pcms(socdev);
953 snd_soc_dapm_free(socdev);
954
955 return 0;
956}
957
958struct snd_soc_codec_device soc_codec_dev_wm8960 = {
959 .probe = wm8960_probe,
960 .remove = wm8960_remove,
961 .suspend = wm8960_suspend,
962 .resume = wm8960_resume,
963};
964EXPORT_SYMBOL_GPL(soc_codec_dev_wm8960);
965
966static int wm8960_register(struct wm8960_priv *wm8960,
967 enum snd_soc_control_type control)
968{
969 struct wm8960_data *pdata = wm8960->codec.dev->platform_data;
970 struct snd_soc_codec *codec = &wm8960->codec;
971 int ret; 924 int ret;
972 u16 reg; 925 u16 reg;
973 926
974 if (wm8960_codec) { 927 wm8960->set_bias_level = wm8960_set_bias_level_out3;
975 dev_err(codec->dev, "Another WM8960 is registered\n"); 928 codec->control_data = wm8960->control_data;
976 ret = -EINVAL;
977 goto err;
978 }
979
980 codec->set_bias_level = wm8960_set_bias_level_out3;
981 929
982 if (!pdata) { 930 if (!pdata) {
983 dev_warn(codec->dev, "No platform data supplied\n"); 931 dev_warn(codec->dev, "No platform data supplied\n");
@@ -988,39 +936,22 @@ static int wm8960_register(struct wm8960_priv *wm8960,
988 } 936 }
989 937
990 if (pdata->capless) 938 if (pdata->capless)
991 codec->set_bias_level = wm8960_set_bias_level_capless; 939 wm8960->set_bias_level = wm8960_set_bias_level_capless;
992 } 940 }
993 941
994 mutex_init(&codec->mutex); 942 ret = snd_soc_codec_set_cache_io(codec, 7, 9, wm8960->control_type);
995 INIT_LIST_HEAD(&codec->dapm_widgets);
996 INIT_LIST_HEAD(&codec->dapm_paths);
997
998 snd_soc_codec_set_drvdata(codec, wm8960);
999 codec->name = "WM8960";
1000 codec->owner = THIS_MODULE;
1001 codec->bias_level = SND_SOC_BIAS_OFF;
1002 codec->dai = &wm8960_dai;
1003 codec->num_dai = 1;
1004 codec->reg_cache_size = WM8960_CACHEREGNUM;
1005 codec->reg_cache = &wm8960->reg_cache;
1006
1007 memcpy(codec->reg_cache, wm8960_reg, sizeof(wm8960_reg));
1008
1009 ret = snd_soc_codec_set_cache_io(codec, 7, 9, control);
1010 if (ret < 0) { 943 if (ret < 0) {
1011 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret); 944 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
1012 goto err; 945 return ret;
1013 } 946 }
1014 947
1015 ret = wm8960_reset(codec); 948 ret = wm8960_reset(codec);
1016 if (ret < 0) { 949 if (ret < 0) {
1017 dev_err(codec->dev, "Failed to issue reset\n"); 950 dev_err(codec->dev, "Failed to issue reset\n");
1018 goto err; 951 return ret;
1019 } 952 }
1020 953
1021 wm8960_dai.dev = codec->dev; 954 wm8960->set_bias_level(codec, SND_SOC_BIAS_STANDBY);
1022
1023 codec->set_bias_level(codec, SND_SOC_BIAS_STANDBY);
1024 955
1025 /* Latch the update bits */ 956 /* Latch the update bits */
1026 reg = snd_soc_read(codec, WM8960_LINVOL); 957 reg = snd_soc_read(codec, WM8960_LINVOL);
@@ -1044,62 +975,59 @@ static int wm8960_register(struct wm8960_priv *wm8960,
1044 reg = snd_soc_read(codec, WM8960_ROUT2); 975 reg = snd_soc_read(codec, WM8960_ROUT2);
1045 snd_soc_write(codec, WM8960_ROUT2, reg | 0x100); 976 snd_soc_write(codec, WM8960_ROUT2, reg | 0x100);
1046 977
1047 wm8960_codec = codec; 978 snd_soc_add_controls(codec, wm8960_snd_controls,
1048 979 ARRAY_SIZE(wm8960_snd_controls));
1049 ret = snd_soc_register_codec(codec); 980 wm8960_add_widgets(codec);
1050 if (ret != 0) {
1051 dev_err(codec->dev, "Failed to register codec: %d\n", ret);
1052 goto err;
1053 }
1054
1055 ret = snd_soc_register_dai(&wm8960_dai);
1056 if (ret != 0) {
1057 dev_err(codec->dev, "Failed to register DAI: %d\n", ret);
1058 goto err_codec;
1059 }
1060 981
1061 return 0; 982 return 0;
1062
1063err_codec:
1064 snd_soc_unregister_codec(codec);
1065err:
1066 kfree(wm8960);
1067 return ret;
1068} 983}
1069 984
1070static void wm8960_unregister(struct wm8960_priv *wm8960) 985/* power down chip */
986static int wm8960_remove(struct snd_soc_codec *codec)
1071{ 987{
1072 wm8960->codec.set_bias_level(&wm8960->codec, SND_SOC_BIAS_OFF); 988 struct wm8960_priv *wm8960 = snd_soc_codec_get_drvdata(codec);
1073 snd_soc_unregister_dai(&wm8960_dai); 989
1074 snd_soc_unregister_codec(&wm8960->codec); 990 wm8960->set_bias_level(codec, SND_SOC_BIAS_OFF);
1075 kfree(wm8960); 991 return 0;
1076 wm8960_codec = NULL;
1077} 992}
1078 993
994static struct snd_soc_codec_driver soc_codec_dev_wm8960 = {
995 .probe = wm8960_probe,
996 .remove = wm8960_remove,
997 .suspend = wm8960_suspend,
998 .resume = wm8960_resume,
999 .set_bias_level = wm8960_set_bias_level,
1000 .reg_cache_size = ARRAY_SIZE(wm8960_reg),
1001 .reg_word_size = sizeof(u16),
1002 .reg_cache_default = wm8960_reg,
1003};
1004
1005#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1079static __devinit int wm8960_i2c_probe(struct i2c_client *i2c, 1006static __devinit int wm8960_i2c_probe(struct i2c_client *i2c,
1080 const struct i2c_device_id *id) 1007 const struct i2c_device_id *id)
1081{ 1008{
1082 struct wm8960_priv *wm8960; 1009 struct wm8960_priv *wm8960;
1083 struct snd_soc_codec *codec; 1010 int ret;
1084 1011
1085 wm8960 = kzalloc(sizeof(struct wm8960_priv), GFP_KERNEL); 1012 wm8960 = kzalloc(sizeof(struct wm8960_priv), GFP_KERNEL);
1086 if (wm8960 == NULL) 1013 if (wm8960 == NULL)
1087 return -ENOMEM; 1014 return -ENOMEM;
1088 1015
1089 codec = &wm8960->codec;
1090
1091 i2c_set_clientdata(i2c, wm8960); 1016 i2c_set_clientdata(i2c, wm8960);
1092 codec->control_data = i2c; 1017 wm8960->control_type = SND_SOC_I2C;
1093 1018 wm8960->control_data = i2c;
1094 codec->dev = &i2c->dev;
1095 1019
1096 return wm8960_register(wm8960, SND_SOC_I2C); 1020 ret = snd_soc_register_codec(&i2c->dev,
1021 &soc_codec_dev_wm8960, &wm8960_dai, 1);
1022 if (ret < 0)
1023 kfree(wm8960);
1024 return ret;
1097} 1025}
1098 1026
1099static __devexit int wm8960_i2c_remove(struct i2c_client *client) 1027static __devexit int wm8960_i2c_remove(struct i2c_client *client)
1100{ 1028{
1101 struct wm8960_priv *wm8960 = i2c_get_clientdata(client); 1029 snd_soc_unregister_codec(&client->dev);
1102 wm8960_unregister(wm8960); 1030 kfree(i2c_get_clientdata(client));
1103 return 0; 1031 return 0;
1104} 1032}
1105 1033
@@ -1111,35 +1039,37 @@ MODULE_DEVICE_TABLE(i2c, wm8960_i2c_id);
1111 1039
1112static struct i2c_driver wm8960_i2c_driver = { 1040static struct i2c_driver wm8960_i2c_driver = {
1113 .driver = { 1041 .driver = {
1114 .name = "wm8960", 1042 .name = "wm8960-codec",
1115 .owner = THIS_MODULE, 1043 .owner = THIS_MODULE,
1116 }, 1044 },
1117 .probe = wm8960_i2c_probe, 1045 .probe = wm8960_i2c_probe,
1118 .remove = __devexit_p(wm8960_i2c_remove), 1046 .remove = __devexit_p(wm8960_i2c_remove),
1119 .id_table = wm8960_i2c_id, 1047 .id_table = wm8960_i2c_id,
1120}; 1048};
1049#endif
1121 1050
1122static int __init wm8960_modinit(void) 1051static int __init wm8960_modinit(void)
1123{ 1052{
1124 int ret; 1053 int ret = 0;
1125 1054#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1126 ret = i2c_add_driver(&wm8960_i2c_driver); 1055 ret = i2c_add_driver(&wm8960_i2c_driver);
1127 if (ret != 0) { 1056 if (ret != 0) {
1128 printk(KERN_ERR "Failed to register WM8960 I2C driver: %d\n", 1057 printk(KERN_ERR "Failed to register WM8960 I2C driver: %d\n",
1129 ret); 1058 ret);
1130 } 1059 }
1131 1060#endif
1132 return ret; 1061 return ret;
1133} 1062}
1134module_init(wm8960_modinit); 1063module_init(wm8960_modinit);
1135 1064
1136static void __exit wm8960_exit(void) 1065static void __exit wm8960_exit(void)
1137{ 1066{
1067#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
1138 i2c_del_driver(&wm8960_i2c_driver); 1068 i2c_del_driver(&wm8960_i2c_driver);
1069#endif
1139} 1070}
1140module_exit(wm8960_exit); 1071module_exit(wm8960_exit);
1141 1072
1142
1143MODULE_DESCRIPTION("ASoC WM8960 driver"); 1073MODULE_DESCRIPTION("ASoC WM8960 driver");
1144MODULE_AUTHOR("Liam Girdwood"); 1074MODULE_AUTHOR("Liam Girdwood");
1145MODULE_LICENSE("GPL"); 1075MODULE_LICENSE("GPL");