Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 1 | /* |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 2 | * Tobermory audio support |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 3 | * |
| 4 | * Copyright 2011 Wolfson Microelectronics |
| 5 | * |
| 6 | * This program is free software; you can redistribute it and/or modify it |
| 7 | * under the terms of the GNU General Public License as published by the |
| 8 | * Free Software Foundation; either version 2 of the License, or (at your |
| 9 | * option) any later version. |
| 10 | */ |
| 11 | |
| 12 | #include <sound/soc.h> |
| 13 | #include <sound/soc-dapm.h> |
| 14 | #include <sound/jack.h> |
| 15 | #include <linux/gpio.h> |
Paul Gortmaker | da155d5 | 2011-07-15 12:38:28 -0400 | [diff] [blame] | 16 | #include <linux/module.h> |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 17 | |
| 18 | #include "../codecs/wm8962.h" |
| 19 | |
Mark Brown | ef473fe | 2011-09-23 16:05:00 +0100 | [diff] [blame] | 20 | static int sample_rate = 44100; |
| 21 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 22 | static int tobermory_set_bias_level(struct snd_soc_card *card, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 23 | struct snd_soc_dapm_context *dapm, |
| 24 | enum snd_soc_bias_level level) |
| 25 | { |
Mengdong Lin | 5015920 | 2015-11-18 02:34:01 -0500 | [diff] [blame] | 26 | struct snd_soc_pcm_runtime *rtd; |
| 27 | struct snd_soc_dai *codec_dai; |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 28 | int ret; |
| 29 | |
Mengdong Lin | 5015920 | 2015-11-18 02:34:01 -0500 | [diff] [blame] | 30 | rtd = snd_soc_get_pcm_runtime(card, card->dai_link[0].name); |
| 31 | codec_dai = rtd->codec_dai; |
| 32 | |
Mark Brown | f79e7ff | 2011-08-21 12:20:00 +0100 | [diff] [blame] | 33 | if (dapm->dev != codec_dai->dev) |
| 34 | return 0; |
| 35 | |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 36 | switch (level) { |
| 37 | case SND_SOC_BIAS_PREPARE: |
| 38 | if (dapm->bias_level == SND_SOC_BIAS_STANDBY) { |
| 39 | ret = snd_soc_dai_set_pll(codec_dai, WM8962_FLL, |
| 40 | WM8962_FLL_MCLK, 32768, |
Mark Brown | ef473fe | 2011-09-23 16:05:00 +0100 | [diff] [blame] | 41 | sample_rate * 512); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 42 | if (ret < 0) |
Mark Brown | 67d0c47 | 2011-06-29 14:07:24 -0700 | [diff] [blame] | 43 | pr_err("Failed to start FLL: %d\n", ret); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 44 | |
| 45 | ret = snd_soc_dai_set_sysclk(codec_dai, |
| 46 | WM8962_SYSCLK_FLL, |
Mark Brown | ef473fe | 2011-09-23 16:05:00 +0100 | [diff] [blame] | 47 | sample_rate * 512, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 48 | SND_SOC_CLOCK_IN); |
Mark Brown | 67d0c47 | 2011-06-29 14:07:24 -0700 | [diff] [blame] | 49 | if (ret < 0) { |
Mark Brown | 371e730 | 2011-08-04 10:54:17 +0900 | [diff] [blame] | 50 | pr_err("Failed to set SYSCLK: %d\n", ret); |
Mark Brown | 631f8e9 | 2014-01-30 18:26:07 +0000 | [diff] [blame] | 51 | snd_soc_dai_set_pll(codec_dai, WM8962_FLL, |
| 52 | 0, 0, 0); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 53 | return ret; |
Mark Brown | 67d0c47 | 2011-06-29 14:07:24 -0700 | [diff] [blame] | 54 | } |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 55 | } |
| 56 | break; |
| 57 | |
| 58 | default: |
| 59 | break; |
| 60 | } |
| 61 | |
| 62 | return 0; |
| 63 | } |
| 64 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 65 | static int tobermory_set_bias_level_post(struct snd_soc_card *card, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 66 | struct snd_soc_dapm_context *dapm, |
| 67 | enum snd_soc_bias_level level) |
| 68 | { |
Mengdong Lin | 5015920 | 2015-11-18 02:34:01 -0500 | [diff] [blame] | 69 | struct snd_soc_pcm_runtime *rtd; |
| 70 | struct snd_soc_dai *codec_dai; |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 71 | int ret; |
| 72 | |
Mengdong Lin | 5015920 | 2015-11-18 02:34:01 -0500 | [diff] [blame] | 73 | rtd = snd_soc_get_pcm_runtime(card, card->dai_link[0].name); |
| 74 | codec_dai = rtd->codec_dai; |
| 75 | |
Mark Brown | f79e7ff | 2011-08-21 12:20:00 +0100 | [diff] [blame] | 76 | if (dapm->dev != codec_dai->dev) |
| 77 | return 0; |
| 78 | |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 79 | switch (level) { |
| 80 | case SND_SOC_BIAS_STANDBY: |
| 81 | ret = snd_soc_dai_set_sysclk(codec_dai, WM8962_SYSCLK_MCLK, |
| 82 | 32768, SND_SOC_CLOCK_IN); |
Mark Brown | 67d0c47 | 2011-06-29 14:07:24 -0700 | [diff] [blame] | 83 | if (ret < 0) { |
| 84 | pr_err("Failed to switch away from FLL: %d\n", ret); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 85 | return ret; |
Mark Brown | 67d0c47 | 2011-06-29 14:07:24 -0700 | [diff] [blame] | 86 | } |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 87 | |
| 88 | ret = snd_soc_dai_set_pll(codec_dai, WM8962_FLL, |
| 89 | 0, 0, 0); |
| 90 | if (ret < 0) { |
Mark Brown | 67d0c47 | 2011-06-29 14:07:24 -0700 | [diff] [blame] | 91 | pr_err("Failed to stop FLL: %d\n", ret); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 92 | return ret; |
| 93 | } |
| 94 | break; |
| 95 | |
| 96 | default: |
| 97 | break; |
| 98 | } |
| 99 | |
| 100 | dapm->bias_level = level; |
| 101 | |
| 102 | return 0; |
| 103 | } |
| 104 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 105 | static int tobermory_hw_params(struct snd_pcm_substream *substream, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 106 | struct snd_pcm_hw_params *params) |
| 107 | { |
Mark Brown | ef473fe | 2011-09-23 16:05:00 +0100 | [diff] [blame] | 108 | sample_rate = params_rate(params); |
| 109 | |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 110 | return 0; |
| 111 | } |
| 112 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 113 | static struct snd_soc_ops tobermory_ops = { |
| 114 | .hw_params = tobermory_hw_params, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 115 | }; |
| 116 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 117 | static struct snd_soc_dai_link tobermory_dai[] = { |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 118 | { |
| 119 | .name = "CPU", |
| 120 | .stream_name = "CPU", |
| 121 | .cpu_dai_name = "samsung-i2s.0", |
| 122 | .codec_dai_name = "wm8962", |
Padmavathi Venna | a08485d8 | 2012-12-07 13:59:21 +0530 | [diff] [blame] | 123 | .platform_name = "samsung-i2s.0", |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 124 | .codec_name = "wm8962.1-001a", |
Mark Brown | 44fdd43 | 2011-09-27 16:42:27 +0100 | [diff] [blame] | 125 | .dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
| 126 | | SND_SOC_DAIFMT_CBM_CFM, |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 127 | .ops = &tobermory_ops, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 128 | }, |
| 129 | }; |
| 130 | |
| 131 | static const struct snd_kcontrol_new controls[] = { |
| 132 | SOC_DAPM_PIN_SWITCH("Main Speaker"), |
Mark Brown | 8c75615 | 2011-09-23 16:46:24 +0100 | [diff] [blame] | 133 | SOC_DAPM_PIN_SWITCH("DMIC"), |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 134 | }; |
| 135 | |
| 136 | static struct snd_soc_dapm_widget widgets[] = { |
| 137 | SND_SOC_DAPM_HP("Headphone", NULL), |
| 138 | SND_SOC_DAPM_MIC("Headset Mic", NULL), |
| 139 | |
| 140 | SND_SOC_DAPM_MIC("DMIC", NULL), |
Mark Brown | 7564463 | 2011-09-23 16:23:11 +0100 | [diff] [blame] | 141 | SND_SOC_DAPM_MIC("AMIC", NULL), |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 142 | |
| 143 | SND_SOC_DAPM_SPK("Main Speaker", NULL), |
| 144 | }; |
| 145 | |
| 146 | static struct snd_soc_dapm_route audio_paths[] = { |
| 147 | { "Headphone", NULL, "HPOUTL" }, |
| 148 | { "Headphone", NULL, "HPOUTR" }, |
| 149 | |
| 150 | { "Main Speaker", NULL, "SPKOUTL" }, |
| 151 | { "Main Speaker", NULL, "SPKOUTR" }, |
| 152 | |
Mark Brown | 086d7f8 | 2011-09-23 16:22:48 +0100 | [diff] [blame] | 153 | { "Headset Mic", NULL, "MICBIAS" }, |
| 154 | { "IN4L", NULL, "Headset Mic" }, |
| 155 | { "IN4R", NULL, "Headset Mic" }, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 156 | |
Mark Brown | 7564463 | 2011-09-23 16:23:11 +0100 | [diff] [blame] | 157 | { "AMIC", NULL, "MICBIAS" }, |
| 158 | { "IN1L", NULL, "AMIC" }, |
| 159 | { "IN1R", NULL, "AMIC" }, |
| 160 | |
Mark Brown | 086d7f8 | 2011-09-23 16:22:48 +0100 | [diff] [blame] | 161 | { "DMIC", NULL, "MICBIAS" }, |
| 162 | { "DMICDAT", NULL, "DMIC" }, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 163 | }; |
| 164 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 165 | static struct snd_soc_jack tobermory_headset; |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 166 | |
| 167 | /* Headset jack detection DAPM pins */ |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 168 | static struct snd_soc_jack_pin tobermory_headset_pins[] = { |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 169 | { |
| 170 | .pin = "Headset Mic", |
| 171 | .mask = SND_JACK_MICROPHONE, |
| 172 | }, |
| 173 | { |
| 174 | .pin = "Headphone", |
| 175 | .mask = SND_JACK_MICROPHONE, |
| 176 | }, |
| 177 | }; |
| 178 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 179 | static int tobermory_late_probe(struct snd_soc_card *card) |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 180 | { |
Mengdong Lin | 5015920 | 2015-11-18 02:34:01 -0500 | [diff] [blame] | 181 | struct snd_soc_pcm_runtime *rtd; |
| 182 | struct snd_soc_codec *codec; |
| 183 | struct snd_soc_dai *codec_dai; |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 184 | int ret; |
| 185 | |
Mengdong Lin | 5015920 | 2015-11-18 02:34:01 -0500 | [diff] [blame] | 186 | rtd = snd_soc_get_pcm_runtime(card, card->dai_link[0].name); |
| 187 | codec = rtd->codec; |
| 188 | codec_dai = rtd->codec_dai; |
| 189 | |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 190 | ret = snd_soc_dai_set_sysclk(codec_dai, WM8962_SYSCLK_MCLK, |
| 191 | 32768, SND_SOC_CLOCK_IN); |
| 192 | if (ret < 0) |
| 193 | return ret; |
| 194 | |
Lars-Peter Clausen | 3fd94f3 | 2015-03-04 10:33:39 +0100 | [diff] [blame] | 195 | ret = snd_soc_card_jack_new(card, "Headset", SND_JACK_HEADSET | |
| 196 | SND_JACK_BTN_0, &tobermory_headset, |
| 197 | tobermory_headset_pins, |
| 198 | ARRAY_SIZE(tobermory_headset_pins)); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 199 | if (ret) |
| 200 | return ret; |
| 201 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 202 | wm8962_mic_detect(codec, &tobermory_headset); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 203 | |
| 204 | return 0; |
| 205 | } |
| 206 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 207 | static struct snd_soc_card tobermory = { |
| 208 | .name = "Tobermory", |
Axel Lin | 095d79d | 2011-12-22 10:53:15 +0800 | [diff] [blame] | 209 | .owner = THIS_MODULE, |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 210 | .dai_link = tobermory_dai, |
| 211 | .num_links = ARRAY_SIZE(tobermory_dai), |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 212 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 213 | .set_bias_level = tobermory_set_bias_level, |
| 214 | .set_bias_level_post = tobermory_set_bias_level_post, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 215 | |
| 216 | .controls = controls, |
| 217 | .num_controls = ARRAY_SIZE(controls), |
| 218 | .dapm_widgets = widgets, |
| 219 | .num_dapm_widgets = ARRAY_SIZE(widgets), |
| 220 | .dapm_routes = audio_paths, |
| 221 | .num_dapm_routes = ARRAY_SIZE(audio_paths), |
Mark Brown | 39afd66 | 2011-11-23 21:33:07 +0000 | [diff] [blame] | 222 | .fully_routed = true, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 223 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 224 | .late_probe = tobermory_late_probe, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 225 | }; |
| 226 | |
Bill Pemberton | fdca21a | 2012-12-07 09:26:15 -0500 | [diff] [blame] | 227 | static int tobermory_probe(struct platform_device *pdev) |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 228 | { |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 229 | struct snd_soc_card *card = &tobermory; |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 230 | int ret; |
| 231 | |
| 232 | card->dev = &pdev->dev; |
| 233 | |
Tushar Behera | c583883 | 2014-05-21 08:52:17 +0530 | [diff] [blame] | 234 | ret = devm_snd_soc_register_card(&pdev->dev, card); |
| 235 | if (ret) |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 236 | dev_err(&pdev->dev, "snd_soc_register_card() failed: %d\n", |
| 237 | ret); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 238 | |
Tushar Behera | c583883 | 2014-05-21 08:52:17 +0530 | [diff] [blame] | 239 | return ret; |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 240 | } |
| 241 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 242 | static struct platform_driver tobermory_driver = { |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 243 | .driver = { |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 244 | .name = "tobermory", |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 245 | .pm = &snd_soc_pm_ops, |
| 246 | }, |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 247 | .probe = tobermory_probe, |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 248 | }; |
| 249 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 250 | module_platform_driver(tobermory_driver); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 251 | |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 252 | MODULE_DESCRIPTION("Tobermory audio support"); |
Mark Brown | 22cb839 | 2011-04-25 18:30:45 +0100 | [diff] [blame] | 253 | MODULE_AUTHOR("Mark Brown <broonie@opensource.wolfsonmicro.com>"); |
| 254 | MODULE_LICENSE("GPL"); |
Mark Brown | 6414261 | 2011-11-30 13:30:27 +0000 | [diff] [blame] | 255 | MODULE_ALIAS("platform:tobermory"); |