diff options
Diffstat (limited to 'sound/soc/codecs/wm8978.c')
-rw-r--r-- | sound/soc/codecs/wm8978.c | 222 |
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 | ||
34 | static 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. */ |
37 | static const u16 wm8978_reg[WM8978_CACHEREGNUM] = { | 34 | static 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 */ |
56 | struct wm8978_priv { | 53 | struct 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 | ||
67 | static const char *wm8978_companding[] = {"Off", "NC", "u-law", "A-law"}; | 64 | static 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); | |||
96 | static const DECLARE_TLV_DB_SCALE(inpga_tlv, -1200, 75, 0); | 93 | static const DECLARE_TLV_DB_SCALE(inpga_tlv, -1200, 75, 0); |
97 | static const DECLARE_TLV_DB_SCALE(spk_tlv, -5700, 100, 0); | 94 | static const DECLARE_TLV_DB_SCALE(spk_tlv, -5700, 100, 0); |
98 | static const DECLARE_TLV_DB_SCALE(boost_tlv, -1500, 300, 1); | 95 | static const DECLARE_TLV_DB_SCALE(boost_tlv, -1500, 300, 1); |
96 | static const DECLARE_TLV_DB_SCALE(limiter_tlv, 0, 100, 0); | ||
99 | 97 | ||
100 | static const struct snd_kcontrol_new wm8978_snd_controls[] = { | 98 | static 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 | ||
357 | static int wm8978_add_widgets(struct snd_soc_codec *codec) | 357 | static 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 | ||
377 | static void pll_factors(struct wm8978_pll_div *pll_div, unsigned int target, | 378 | static 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 */ |
878 | struct snd_soc_dai wm8978_dai = { | 878 | static 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 | }; |
897 | EXPORT_SYMBOL_GPL(wm8978_dai); | ||
898 | 896 | ||
899 | static int wm8978_suspend(struct platform_device *pdev, pm_message_t state) | 897 | static 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 | ||
911 | static int wm8978_resume(struct platform_device *pdev) | 906 | static 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 | ||
936 | static 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 | |||
961 | pcm_err: | ||
962 | return ret; | ||
963 | } | ||
964 | |||
965 | /* power down chip */ | ||
966 | static 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 | |||
976 | struct 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 | }; | ||
982 | EXPORT_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 | ||
1003 | static __devinit int wm8978_register(struct wm8978_priv *wm8978) | 948 | static 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 | |||
1076 | err_codec: | ||
1077 | snd_soc_unregister_codec(codec); | ||
1078 | err: | ||
1079 | return ret; | ||
1080 | } | 987 | } |
1081 | 988 | ||
1082 | static __devexit void wm8978_unregister(struct wm8978_priv *wm8978) | 989 | /* power down chip */ |
990 | static 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 | ||
996 | static 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) | ||
1090 | static __devinit int wm8978_i2c_probe(struct i2c_client *i2c, | 1008 | static __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 | ||
1116 | static __devexit int wm8978_i2c_remove(struct i2c_client *client) | 1028 | static __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 | ||
1130 | static struct i2c_driver wm8978_i2c_driver = { | 1041 | static 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 | ||
1140 | static int __init wm8978_modinit(void) | 1052 | static 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 | } |
1144 | module_init(wm8978_modinit); | 1064 | module_init(wm8978_modinit); |
1145 | 1065 | ||
1146 | static void __exit wm8978_exit(void) | 1066 | static 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 | } |
1150 | module_exit(wm8978_exit); | 1072 | module_exit(wm8978_exit); |
1151 | 1073 | ||