diff options
Diffstat (limited to 'sound/soc/codecs/wm8960.c')
-rw-r--r-- | sound/soc/codecs/wm8960.c | 242 |
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 | ||
32 | struct 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 | ||
76 | struct wm8960_priv { | 73 | struct 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 | ||
143 | static int wm8960_put_deemph(struct snd_kcontrol *kcontrol, | 143 | static 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 | ||
854 | static 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 | ||
866 | struct snd_soc_dai wm8960_dai = { | 876 | static 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 | }; |
883 | EXPORT_SYMBOL_GPL(wm8960_dai); | ||
884 | 893 | ||
885 | static int wm8960_suspend(struct platform_device *pdev, pm_message_t state) | 894 | static 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 | ||
894 | static int wm8960_resume(struct platform_device *pdev) | 902 | static 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 | ||
914 | static struct snd_soc_codec *wm8960_codec; | 920 | static int wm8960_probe(struct snd_soc_codec *codec) |
915 | |||
916 | static 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 | |||
943 | pcm_err: | ||
944 | return ret; | ||
945 | } | ||
946 | |||
947 | /* power down chip */ | ||
948 | static 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 | |||
958 | struct 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 | }; | ||
964 | EXPORT_SYMBOL_GPL(soc_codec_dev_wm8960); | ||
965 | |||
966 | static 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 | |||
1063 | err_codec: | ||
1064 | snd_soc_unregister_codec(codec); | ||
1065 | err: | ||
1066 | kfree(wm8960); | ||
1067 | return ret; | ||
1068 | } | 983 | } |
1069 | 984 | ||
1070 | static void wm8960_unregister(struct wm8960_priv *wm8960) | 985 | /* power down chip */ |
986 | static 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 | ||
994 | static 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) | ||
1079 | static __devinit int wm8960_i2c_probe(struct i2c_client *i2c, | 1006 | static __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 | ||
1099 | static __devexit int wm8960_i2c_remove(struct i2c_client *client) | 1027 | static __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 | ||
1112 | static struct i2c_driver wm8960_i2c_driver = { | 1040 | static 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 | ||
1122 | static int __init wm8960_modinit(void) | 1051 | static 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 | } |
1134 | module_init(wm8960_modinit); | 1063 | module_init(wm8960_modinit); |
1135 | 1064 | ||
1136 | static void __exit wm8960_exit(void) | 1065 | static 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 | } |
1140 | module_exit(wm8960_exit); | 1071 | module_exit(wm8960_exit); |
1141 | 1072 | ||
1142 | |||
1143 | MODULE_DESCRIPTION("ASoC WM8960 driver"); | 1073 | MODULE_DESCRIPTION("ASoC WM8960 driver"); |
1144 | MODULE_AUTHOR("Liam Girdwood"); | 1074 | MODULE_AUTHOR("Liam Girdwood"); |
1145 | MODULE_LICENSE("GPL"); | 1075 | MODULE_LICENSE("GPL"); |