blob: cb73ab6e637c2bc6e3eb58d75c5ee062ad932275 [file] [log] [blame]
Joonwoo Park0976d012011-12-22 11:48:18 -08001/* Copyright (c) 2011-2012, Code Aurora Forum. All rights reserved.
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
13#include <linux/clk.h>
Kiran Kandidb0a4b02011-08-23 09:32:09 -070014#include <linux/delay.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070015#include <linux/gpio.h>
16#include <linux/mfd/pm8xxx/pm8921.h>
17#include <linux/platform_device.h>
18#include <linux/gpio.h>
19#include <linux/mfd/pm8xxx/pm8921.h>
Joonwoo Park0976d012011-12-22 11:48:18 -080020#include <linux/slab.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070021#include <sound/core.h>
22#include <sound/soc.h>
23#include <sound/soc-dapm.h>
24#include <sound/soc-dsp.h>
25#include <sound/pcm.h>
26#include <sound/jack.h>
Bradley Rubin229c6a52011-07-12 16:18:48 -070027#include <asm/mach-types.h>
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -080028#include <mach/socinfo.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070029#include "msm-pcm-routing.h"
Bhalchandra Gajare0b9d4d52011-10-21 16:17:47 -070030#include "../codecs/wcd9310.h"
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070031
32/* 8960 machine driver */
33
34#define PM8921_GPIO_BASE NR_GPIO_IRQS
35#define PM8921_GPIO_PM_TO_SYS(pm_gpio) (pm_gpio - 1 + PM8921_GPIO_BASE)
36
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070037#define MSM8960_SPK_ON 1
38#define MSM8960_SPK_OFF 0
39
40#define msm8960_SLIM_0_RX_MAX_CHANNELS 2
41#define msm8960_SLIM_0_TX_MAX_CHANNELS 4
42
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -070043#define BTSCO_RATE_8KHZ 8000
44#define BTSCO_RATE_16KHZ 16000
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070045
Kiran Kandi462acea2011-08-31 23:53:14 -070046#define BOTTOM_SPK_AMP_POS 0x1
47#define BOTTOM_SPK_AMP_NEG 0x2
48#define TOP_SPK_AMP_POS 0x4
49#define TOP_SPK_AMP_NEG 0x8
50
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -070051#define GPIO_AUX_PCM_DOUT 63
52#define GPIO_AUX_PCM_DIN 64
53#define GPIO_AUX_PCM_SYNC 65
54#define GPIO_AUX_PCM_CLK 66
55
Joonwoo Park0976d012011-12-22 11:48:18 -080056#define TABLA_EXT_CLK_RATE 12288000
57
58#define TABLA_MBHC_DEF_BUTTONS 3
59#define TABLA_MBHC_DEF_RLOADS 5
60
Harmandeep Singh0dd82412011-11-11 09:46:17 -080061static u32 top_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(18);
62static u32 bottom_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(19);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070063static int msm8960_spk_control;
Kiran Kandi462acea2011-08-31 23:53:14 -070064static int msm8960_ext_bottom_spk_pamp;
65static int msm8960_ext_top_spk_pamp;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070066static int msm8960_slim_0_rx_ch = 1;
67static int msm8960_slim_0_tx_ch = 1;
68
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -070069static int msm8960_btsco_rate = BTSCO_RATE_8KHZ;
70static int msm8960_btsco_ch = 1;
71
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070072static struct clk *codec_clk;
73static int clk_users;
74
75static int msm8960_headset_gpios_configured;
76
77static struct snd_soc_jack hs_jack;
Bradley Rubincb1e2732011-06-23 16:49:20 -070078static struct snd_soc_jack button_jack;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070079
Joonwoo Park0976d012011-12-22 11:48:18 -080080static void *tabla_mbhc_cal;
81
Kiran Kandi462acea2011-08-31 23:53:14 -070082static void msm8960_enable_ext_spk_amp_gpio(u32 spk_amp_gpio)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070083{
84 int ret = 0;
85
86 struct pm_gpio param = {
87 .direction = PM_GPIO_DIR_OUT,
88 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
89 .output_value = 1,
90 .pull = PM_GPIO_PULL_NO,
91 .vin_sel = PM_GPIO_VIN_S4,
92 .out_strength = PM_GPIO_STRENGTH_MED,
Kiran Kandi462acea2011-08-31 23:53:14 -070093 .
94 function = PM_GPIO_FUNC_NORMAL,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070095 };
96
Harmandeep Singh0dd82412011-11-11 09:46:17 -080097 if (spk_amp_gpio == bottom_spk_pamp_gpio) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070098
Harmandeep Singh0dd82412011-11-11 09:46:17 -080099 ret = gpio_request(bottom_spk_pamp_gpio, "BOTTOM_SPK_AMP");
Kiran Kandi462acea2011-08-31 23:53:14 -0700100 if (ret) {
101 pr_err("%s: Error requesting BOTTOM SPK AMP GPIO %u\n",
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800102 __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700103 return;
104 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800105 ret = pm8xxx_gpio_config(bottom_spk_pamp_gpio, &param);
Kiran Kandi462acea2011-08-31 23:53:14 -0700106 if (ret)
107 pr_err("%s: Failed to configure Bottom Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800108 " gpio %u\n", __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700109 else {
110 pr_debug("%s: enable Bottom spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800111 gpio_direction_output(bottom_spk_pamp_gpio, 1);
Kiran Kandi462acea2011-08-31 23:53:14 -0700112 }
113
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800114 } else if (spk_amp_gpio == top_spk_pamp_gpio) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700115
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800116 ret = gpio_request(top_spk_pamp_gpio, "TOP_SPK_AMP");
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700117 if (ret) {
118 pr_err("%s: Error requesting GPIO %d\n", __func__,
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800119 top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700120 return;
121 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800122 ret = pm8xxx_gpio_config(top_spk_pamp_gpio, &param);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700123 if (ret)
Kiran Kandi462acea2011-08-31 23:53:14 -0700124 pr_err("%s: Failed to configure Top Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800125 " gpio %u\n", __func__, top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700126 else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700127 pr_debug("%s: enable Top spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800128 gpio_direction_output(top_spk_pamp_gpio, 1);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700129 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700130 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700131 pr_err("%s: ERROR : Invalid External Speaker Ampl GPIO."
132 " gpio = %u\n", __func__, spk_amp_gpio);
133 return;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700134 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700135}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700136
Kiran Kandi462acea2011-08-31 23:53:14 -0700137static void msm8960_ext_spk_power_amp_on(u32 spk)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700138{
Kiran Kandi462acea2011-08-31 23:53:14 -0700139 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
140
141 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
142 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
143
144 pr_debug("%s() External Bottom Speaker Ampl already "
145 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700146 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700147 }
148
149 msm8960_ext_bottom_spk_pamp |= spk;
150
151 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
152 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
153
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800154 msm8960_enable_ext_spk_amp_gpio(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700155 pr_debug("%s: slepping 4 ms after turning on external "
156 " Bottom Speaker Ampl\n", __func__);
157 usleep_range(4000, 4000);
158 }
159
160 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
161
162 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
163 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
164
165 pr_debug("%s() External Top Speaker Ampl already"
166 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700167 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700168 }
169
170 msm8960_ext_top_spk_pamp |= spk;
171
172 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
173 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
174
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800175 msm8960_enable_ext_spk_amp_gpio(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700176 pr_debug("%s: sleeping 4 ms after turning on "
177 " external Top Speaker Ampl\n", __func__);
178 usleep_range(4000, 4000);
179 }
180 } else {
181
182 pr_err("%s: ERROR : Invalid External Speaker Ampl. spk = 0x%08x\n",
183 __func__, spk);
184 return;
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700185 }
Kiran Kandi462acea2011-08-31 23:53:14 -0700186}
187
188static void msm8960_ext_spk_power_amp_off(u32 spk)
189{
190 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
191
192 if (!msm8960_ext_bottom_spk_pamp)
193 return;
194
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800195 gpio_direction_output(bottom_spk_pamp_gpio, 0);
196 gpio_free(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700197 msm8960_ext_bottom_spk_pamp = 0;
198
199 pr_debug("%s: sleeping 4 ms after turning off external Bottom"
200 " Speaker Ampl\n", __func__);
201
202 usleep_range(4000, 4000);
203
204 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
205
206 if (!msm8960_ext_top_spk_pamp)
207 return;
208
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800209 gpio_direction_output(top_spk_pamp_gpio, 0);
210 gpio_free(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700211 msm8960_ext_top_spk_pamp = 0;
212
213 pr_debug("%s: sleeping 4 ms after turning off external Top"
214 " Spkaker Ampl\n", __func__);
215
216 usleep_range(4000, 4000);
217 } else {
218
219 pr_err("%s: ERROR : Invalid Ext Spk Ampl. spk = 0x%08x\n",
220 __func__, spk);
221 return;
222 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700223}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700224
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700225static void msm8960_ext_control(struct snd_soc_codec *codec)
226{
227 struct snd_soc_dapm_context *dapm = &codec->dapm;
228
229 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700230 if (msm8960_spk_control == MSM8960_SPK_ON) {
231 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
232 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
233 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
234 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
235 } else {
236 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Pos");
237 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Neg");
238 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Pos");
239 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Neg");
240 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700241
242 snd_soc_dapm_sync(dapm);
243}
244
245static int msm8960_get_spk(struct snd_kcontrol *kcontrol,
246 struct snd_ctl_elem_value *ucontrol)
247{
248 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
249 ucontrol->value.integer.value[0] = msm8960_spk_control;
250 return 0;
251}
252static int msm8960_set_spk(struct snd_kcontrol *kcontrol,
253 struct snd_ctl_elem_value *ucontrol)
254{
255 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
256
257 pr_debug("%s()\n", __func__);
258 if (msm8960_spk_control == ucontrol->value.integer.value[0])
259 return 0;
260
261 msm8960_spk_control = ucontrol->value.integer.value[0];
262 msm8960_ext_control(codec);
263 return 1;
264}
265static int msm8960_spkramp_event(struct snd_soc_dapm_widget *w,
266 struct snd_kcontrol *k, int event)
267{
268 pr_debug("%s() %x\n", __func__, SND_SOC_DAPM_EVENT_ON(event));
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700269
270 if (SND_SOC_DAPM_EVENT_ON(event)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700271 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
272 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_POS);
273 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
274 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_NEG);
275 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
276 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_POS);
277 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
278 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_NEG);
279 else {
280 pr_err("%s() Invalid Speaker Widget = %s\n",
281 __func__, w->name);
282 return -EINVAL;
283 }
284
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700285 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700286 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
287 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_POS);
288 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
289 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_NEG);
290 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
291 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_POS);
292 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
293 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_NEG);
294 else {
295 pr_err("%s() Invalid Speaker Widget = %s\n",
296 __func__, w->name);
297 return -EINVAL;
298 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700299 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700300 return 0;
301}
Joonwoo Park0976d012011-12-22 11:48:18 -0800302int msm8960_enable_codec_ext_clk(
303 struct snd_soc_codec *codec, int enable)
304{
305 pr_debug("%s: enable = %d\n", __func__, enable);
306 if (enable) {
307 clk_users++;
308 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
309 if (clk_users != 1)
310 return 0;
311
312 codec_clk = clk_get(NULL, "i2s_spkr_osr_clk");
313 if (codec_clk) {
314 clk_set_rate(codec_clk, TABLA_EXT_CLK_RATE);
315 clk_enable(codec_clk);
316 tabla_mclk_enable(codec, 1);
317 } else {
318 pr_err("%s: Error setting Tabla MCLK\n", __func__);
319 clk_users--;
320 return -EINVAL;
321 }
322 } else {
323 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
324 if (clk_users == 0)
325 return 0;
326 clk_users--;
327 if (!clk_users) {
328 pr_debug("%s: disabling MCLK. clk_users = %d\n",
329 __func__, clk_users);
330 clk_disable(codec_clk);
331 clk_put(codec_clk);
332 tabla_mclk_enable(codec, 0);
333 }
334 }
335 return 0;
336}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700337
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700338static int msm8960_mclk_event(struct snd_soc_dapm_widget *w,
339 struct snd_kcontrol *kcontrol, int event)
340{
341 pr_debug("%s: event = %d\n", __func__, event);
342
343 switch (event) {
344 case SND_SOC_DAPM_PRE_PMU:
Joonwoo Park0976d012011-12-22 11:48:18 -0800345 return msm8960_enable_codec_ext_clk(w->codec, 1);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700346 case SND_SOC_DAPM_POST_PMD:
Joonwoo Park0976d012011-12-22 11:48:18 -0800347 return msm8960_enable_codec_ext_clk(w->codec, 0);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700348 }
349 return 0;
350}
351
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700352static const struct snd_soc_dapm_widget msm8960_dapm_widgets[] = {
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700353
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700354 SND_SOC_DAPM_SUPPLY("MCLK", SND_SOC_NOPM, 0, 0,
355 msm8960_mclk_event, SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
356
Kiran Kandi462acea2011-08-31 23:53:14 -0700357 SND_SOC_DAPM_SPK("Ext Spk Bottom Pos", msm8960_spkramp_event),
358 SND_SOC_DAPM_SPK("Ext Spk Bottom Neg", msm8960_spkramp_event),
359
360 SND_SOC_DAPM_SPK("Ext Spk Top Pos", msm8960_spkramp_event),
361 SND_SOC_DAPM_SPK("Ext Spk Top Neg", msm8960_spkramp_event),
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700362
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700363 SND_SOC_DAPM_MIC("Handset Mic", NULL),
364 SND_SOC_DAPM_MIC("Headset Mic", NULL),
365 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700366 SND_SOC_DAPM_MIC("ANCRight Headset Mic", NULL),
367 SND_SOC_DAPM_MIC("ANCLeft Headset Mic", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700368
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700369 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700370 SND_SOC_DAPM_MIC("Digital Mic2", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700371 SND_SOC_DAPM_MIC("Digital Mic3", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700372 SND_SOC_DAPM_MIC("Digital Mic4", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700373 SND_SOC_DAPM_MIC("Digital Mic5", NULL),
374 SND_SOC_DAPM_MIC("Digital Mic6", NULL),
375
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700376};
377
Bradley Rubin229c6a52011-07-12 16:18:48 -0700378static const struct snd_soc_dapm_route common_audio_map[] = {
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700379
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700380 {"RX_BIAS", NULL, "MCLK"},
381 {"LDO_H", NULL, "MCLK"},
382
383 /* Speaker path */
Kiran Kandi462acea2011-08-31 23:53:14 -0700384 {"Ext Spk Bottom Pos", NULL, "LINEOUT1"},
385 {"Ext Spk Bottom Neg", NULL, "LINEOUT3"},
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700386
Kiran Kandi462acea2011-08-31 23:53:14 -0700387 {"Ext Spk Top Pos", NULL, "LINEOUT2"},
388 {"Ext Spk Top Neg", NULL, "LINEOUT4"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700389
390 /* Microphone path */
Bradley Rubin229c6a52011-07-12 16:18:48 -0700391 {"AMIC1", NULL, "MIC BIAS1 Internal1"},
392 {"MIC BIAS1 Internal1", NULL, "Handset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700393
394 {"AMIC2", NULL, "MIC BIAS2 External"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700395 {"MIC BIAS2 External", NULL, "Headset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700396
Patrick Laie519dd62011-09-28 12:37:11 -0700397 /**
398 * AMIC3 and AMIC4 inputs are connected to ANC microphones
399 * These mics are biased differently on CDP and FLUID
400 * routing entries below are based on bias arrangement
401 * on FLUID.
402 */
403 {"AMIC3", NULL, "MIC BIAS3 Internal1"},
404 {"MIC BIAS3 Internal1", NULL, "ANCRight Headset Mic"},
405
406 {"AMIC4", NULL, "MIC BIAS1 Internal2"},
407 {"MIC BIAS1 Internal2", NULL, "ANCLeft Headset Mic"},
408
Kiran Kandie9bf86a2011-07-21 16:50:41 -0700409 {"HEADPHONE", NULL, "LDO_H"},
410
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700411 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700412 * The digital Mic routes are setup considering
413 * fluid as default device.
414 */
415
416 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700417 * Digital Mic1. Front Bottom left Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700418 * Digital Mic GM5 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700419 * Conncted to DMIC2 Input on Tabla codec.
420 */
421 {"DMIC2", NULL, "MIC BIAS1 External"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700422 {"MIC BIAS1 External", NULL, "Digital Mic1"},
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700423
424 /**
425 * Digital Mic2. Front Bottom right Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700426 * Digital Mic GM6 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700427 * Conncted to DMIC1 Input on Tabla codec.
428 */
429 {"DMIC1", NULL, "MIC BIAS1 External"},
430 {"MIC BIAS1 External", NULL, "Digital Mic2"},
431
432 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700433 * Digital Mic3. Back Bottom Digital Mic on Fluid.
434 * Digital Mic GM1 on CDP mainboard.
435 * Conncted to DMIC4 Input on Tabla codec.
436 */
437 {"DMIC4", NULL, "MIC BIAS3 External"},
438 {"MIC BIAS3 External", NULL, "Digital Mic3"},
439
440 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700441 * Digital Mic4. Back top Digital Mic on Fluid.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700442 * Digital Mic GM2 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700443 * Conncted to DMIC3 Input on Tabla codec.
444 */
445 {"DMIC3", NULL, "MIC BIAS3 External"},
446 {"MIC BIAS3 External", NULL, "Digital Mic4"},
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700447
448 /**
449 * Digital Mic5. Front top Digital Mic on Fluid.
450 * Digital Mic GM3 on CDP mainboard.
451 * Conncted to DMIC5 Input on Tabla codec.
452 */
453 {"DMIC5", NULL, "MIC BIAS4 External"},
454 {"MIC BIAS4 External", NULL, "Digital Mic5"},
455
Patrick Laicb7802b2011-10-04 12:39:18 -0700456 /* Tabla digital Mic6 - back bottom digital Mic on Liquid and
457 * bottom mic on CDP. FLUID/MTP do not have dmic6 installed.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700458 */
459 {"DMIC6", NULL, "MIC BIAS4 External"},
460 {"MIC BIAS4 External", NULL, "Digital Mic6"},
Bradley Rubin229c6a52011-07-12 16:18:48 -0700461};
462
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700463static const char *spk_function[] = {"Off", "On"};
Patrick Lai9f4b4292011-07-16 22:11:09 -0700464static const char *slim0_rx_ch_text[] = {"One", "Two"};
465static const char *slim0_tx_ch_text[] = {"One", "Two", "Three", "Four"};
466
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700467static const struct soc_enum msm8960_enum[] = {
468 SOC_ENUM_SINGLE_EXT(2, spk_function),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700469 SOC_ENUM_SINGLE_EXT(2, slim0_rx_ch_text),
470 SOC_ENUM_SINGLE_EXT(4, slim0_tx_ch_text),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700471};
472
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700473static const char *btsco_rate_text[] = {"8000", "16000"};
474static const struct soc_enum msm8960_btsco_enum[] = {
475 SOC_ENUM_SINGLE_EXT(2, btsco_rate_text),
476};
477
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700478static int msm8960_slim_0_rx_ch_get(struct snd_kcontrol *kcontrol,
479 struct snd_ctl_elem_value *ucontrol)
480{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700481 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800482 msm8960_slim_0_rx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700483 ucontrol->value.integer.value[0] = msm8960_slim_0_rx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700484 return 0;
485}
486
487static int msm8960_slim_0_rx_ch_put(struct snd_kcontrol *kcontrol,
488 struct snd_ctl_elem_value *ucontrol)
489{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700490 msm8960_slim_0_rx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700491
Patrick Lai9f4b4292011-07-16 22:11:09 -0700492 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800493 msm8960_slim_0_rx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700494 return 1;
495}
496
497static int msm8960_slim_0_tx_ch_get(struct snd_kcontrol *kcontrol,
498 struct snd_ctl_elem_value *ucontrol)
499{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700500 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800501 msm8960_slim_0_tx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700502 ucontrol->value.integer.value[0] = msm8960_slim_0_tx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700503 return 0;
504}
505
506static int msm8960_slim_0_tx_ch_put(struct snd_kcontrol *kcontrol,
507 struct snd_ctl_elem_value *ucontrol)
508{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700509 msm8960_slim_0_tx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700510
511 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800512 msm8960_slim_0_tx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700513 return 1;
514}
515
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700516static int msm8960_btsco_rate_get(struct snd_kcontrol *kcontrol,
517 struct snd_ctl_elem_value *ucontrol)
518{
Joonwoo Park0976d012011-12-22 11:48:18 -0800519 pr_debug("%s: msm8960_btsco_rate = %d", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700520 ucontrol->value.integer.value[0] = msm8960_btsco_rate;
521 return 0;
522}
523
524static int msm8960_btsco_rate_put(struct snd_kcontrol *kcontrol,
525 struct snd_ctl_elem_value *ucontrol)
526{
527 switch (ucontrol->value.integer.value[0]) {
528 case 0:
529 msm8960_btsco_rate = BTSCO_RATE_8KHZ;
530 break;
531 case 1:
532 msm8960_btsco_rate = BTSCO_RATE_16KHZ;
533 break;
534 default:
535 msm8960_btsco_rate = BTSCO_RATE_8KHZ;
536 break;
537 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800538 pr_debug("%s: msm8960_btsco_rate = %d\n", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700539 return 0;
540}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700541
542static const struct snd_kcontrol_new tabla_msm8960_controls[] = {
543 SOC_ENUM_EXT("Speaker Function", msm8960_enum[0], msm8960_get_spk,
544 msm8960_set_spk),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700545 SOC_ENUM_EXT("SLIM_0_RX Channels", msm8960_enum[1],
546 msm8960_slim_0_rx_ch_get, msm8960_slim_0_rx_ch_put),
547 SOC_ENUM_EXT("SLIM_0_TX Channels", msm8960_enum[2],
548 msm8960_slim_0_tx_ch_get, msm8960_slim_0_tx_ch_put),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700549};
550
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700551static const struct snd_kcontrol_new int_btsco_rate_mixer_controls[] = {
552 SOC_ENUM_EXT("Internal BTSCO SampleRate", msm8960_btsco_enum[0],
553 msm8960_btsco_rate_get, msm8960_btsco_rate_put),
554};
555
556static int msm8960_btsco_init(struct snd_soc_pcm_runtime *rtd)
557{
558 int err = 0;
559 struct snd_soc_platform *platform = rtd->platform;
560
561 err = snd_soc_add_platform_controls(platform,
562 int_btsco_rate_mixer_controls,
563 ARRAY_SIZE(int_btsco_rate_mixer_controls));
564 if (err < 0)
565 return err;
566 return 0;
567}
568
Joonwoo Park0976d012011-12-22 11:48:18 -0800569static void *def_tabla_mbhc_cal(void)
570{
571 void *tabla_cal;
572 struct tabla_mbhc_btn_detect_cfg *btn_cfg;
573 u16 *btn_low, *btn_high;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800574 u8 *n_ready, *n_cic, *gain;
Joonwoo Park0976d012011-12-22 11:48:18 -0800575
576 tabla_cal = kzalloc(TABLA_MBHC_CAL_SIZE(TABLA_MBHC_DEF_BUTTONS,
577 TABLA_MBHC_DEF_RLOADS),
578 GFP_KERNEL);
579 if (!tabla_cal) {
580 pr_err("%s: out of memory\n", __func__);
581 return NULL;
582 }
583
584#define S(X, Y) ((TABLA_MBHC_CAL_GENERAL_PTR(tabla_cal)->X) = (Y))
585 S(t_ldoh, 100);
586 S(t_bg_fast_settle, 100);
587 S(t_shutdown_plug_rem, 255);
588 S(mbhc_nsa, 4);
589 S(mbhc_navg, 4);
590#undef S
591#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_DET_PTR(tabla_cal)->X) = (Y))
592 S(mic_current, TABLA_PID_MIC_5_UA);
593 S(hph_current, TABLA_PID_MIC_5_UA);
594 S(t_mic_pid, 100);
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800595 S(t_ins_complete, 250);
Joonwoo Park0976d012011-12-22 11:48:18 -0800596 S(t_ins_retry, 200);
597#undef S
598#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla_cal)->X) = (Y))
599 S(v_no_mic, 30);
600 S(v_hs_max, 1450);
601#undef S
602#define S(X, Y) ((TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal)->X) = (Y))
603 S(c[0], 6);
604 S(c[1], 10);
605 S(c[2], 10);
606 S(c[3], 14);
607 S(c[4], 14);
608 S(c[5], 16);
609 S(c[6], 0);
610 S(c[7], 0);
611 S(nc, 5);
612 S(n_meas, 11);
613 S(mbhc_nsc, 11);
614 S(num_btn, TABLA_MBHC_DEF_BUTTONS);
615 S(v_btn_press_delta_sta, 100);
616 S(v_btn_press_delta_cic, 50);
617#undef S
618 btn_cfg = TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal);
619 btn_low = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_LOW);
620 btn_high = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_HIGH);
621 btn_low[0] = 0;
622 btn_high[0] = 40;
623 btn_low[1] = 60;
624 btn_high[1] = 140;
625 btn_low[2] = 160;
626 btn_high[2] = 240;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800627 n_ready = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_READY);
628 n_ready[0] = 48;
629 n_ready[1] = 38;
Joonwoo Park0976d012011-12-22 11:48:18 -0800630 n_cic = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_CIC);
631 n_cic[0] = 120;
632 n_cic[1] = 94;
633 gain = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_GAIN);
634 gain[0] = 11;
635 gain[1] = 9;
636
637 return tabla_cal;
638}
639
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700640static int msm8960_audrx_init(struct snd_soc_pcm_runtime *rtd)
641{
642 int err;
643 struct snd_soc_codec *codec = rtd->codec;
644 struct snd_soc_dapm_context *dapm = &codec->dapm;
645
646 pr_debug("%s()\n", __func__);
647
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800648 if (machine_is_msm8960_liquid()) {
649 top_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(19));
650 bottom_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(18));
651 }
652
Kiran Kandi28ef14e2011-09-16 15:45:47 -0700653 rtd->pmdown_time = 0;
654
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700655 err = snd_soc_add_controls(codec, tabla_msm8960_controls,
656 ARRAY_SIZE(tabla_msm8960_controls));
657 if (err < 0)
658 return err;
659
660 snd_soc_dapm_new_controls(dapm, msm8960_dapm_widgets,
661 ARRAY_SIZE(msm8960_dapm_widgets));
662
Bradley Rubin229c6a52011-07-12 16:18:48 -0700663 snd_soc_dapm_add_routes(dapm, common_audio_map,
664 ARRAY_SIZE(common_audio_map));
665
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700666 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
667 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
668 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
669 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700670
671 snd_soc_dapm_sync(dapm);
672
673 err = snd_soc_jack_new(codec, "Headset Jack",
Patrick Lai456c4ea2011-11-03 16:16:10 -0700674 (SND_JACK_HEADSET | SND_JACK_OC_HPHL | SND_JACK_OC_HPHR),
675 &hs_jack);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700676 if (err) {
677 pr_err("failed to create new jack\n");
678 return err;
679 }
Bradley Rubincb1e2732011-06-23 16:49:20 -0700680
681 err = snd_soc_jack_new(codec, "Button Jack",
682 SND_JACK_BTN_0, &button_jack);
683 if (err) {
684 pr_err("failed to create new jack\n");
685 return err;
686 }
687
Joonwoo Park0976d012011-12-22 11:48:18 -0800688 tabla_hs_detect(codec, &hs_jack, &button_jack, tabla_mbhc_cal,
689 TABLA_MICBIAS2, msm8960_enable_codec_ext_clk, 0,
690 TABLA_EXT_CLK_RATE);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700691
692 return 0;
693}
694
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700695static struct snd_soc_dsp_link lpa_fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700696 .playback = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700697 .trigger = {
698 SND_SOC_DSP_TRIGGER_POST,
699 SND_SOC_DSP_TRIGGER_POST
700 },
701};
702
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700703static struct snd_soc_dsp_link fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700704 .playback = true,
705 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700706 .trigger = {
707 SND_SOC_DSP_TRIGGER_POST,
708 SND_SOC_DSP_TRIGGER_POST
709 },
710};
711
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700712static struct snd_soc_dsp_link slimbus0_hl_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700713 .playback = true,
714 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700715 .trigger = {
716 SND_SOC_DSP_TRIGGER_POST,
717 SND_SOC_DSP_TRIGGER_POST
718 },
719};
720
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700721static struct snd_soc_dsp_link int_fm_hl_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700722 .playback = true,
723 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700724 .trigger = {
725 SND_SOC_DSP_TRIGGER_POST,
726 SND_SOC_DSP_TRIGGER_POST
727 },
728};
729
Alex Wongb3d06a02012-01-12 10:00:41 -0800730static struct snd_soc_dsp_link hdmi_rx_hl = {
731 .playback = true,
732 .trigger = {
733 SND_SOC_DSP_TRIGGER_POST,
734 SND_SOC_DSP_TRIGGER_POST
735 },
736};
737
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700738static int msm8960_slim_0_rx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
739 struct snd_pcm_hw_params *params)
740{
741 struct snd_interval *rate = hw_param_interval(params,
742 SNDRV_PCM_HW_PARAM_RATE);
743
744 struct snd_interval *channels = hw_param_interval(params,
745 SNDRV_PCM_HW_PARAM_CHANNELS);
746
747 pr_debug("%s()\n", __func__);
748 rate->min = rate->max = 48000;
749 channels->min = channels->max = msm8960_slim_0_rx_ch;
750
751 return 0;
752}
753
754static int msm8960_slim_0_tx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
755 struct snd_pcm_hw_params *params)
756{
757 struct snd_interval *rate = hw_param_interval(params,
758 SNDRV_PCM_HW_PARAM_RATE);
759
760 struct snd_interval *channels = hw_param_interval(params,
761 SNDRV_PCM_HW_PARAM_CHANNELS);
762
763 pr_debug("%s()\n", __func__);
764 rate->min = rate->max = 48000;
765 channels->min = channels->max = msm8960_slim_0_tx_ch;
766
767 return 0;
768}
769
770static int msm8960_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
771 struct snd_pcm_hw_params *params)
772{
773 struct snd_interval *rate = hw_param_interval(params,
774 SNDRV_PCM_HW_PARAM_RATE);
775
776 pr_debug("%s()\n", __func__);
777 rate->min = rate->max = 48000;
778
779 return 0;
780}
781
Helen Zeng73f7fc62011-11-18 16:29:59 -0800782static int msm8960_hdmi_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
783 struct snd_pcm_hw_params *params)
784{
785 struct snd_interval *rate = hw_param_interval(params,
786 SNDRV_PCM_HW_PARAM_RATE);
787
788 struct snd_interval *channels = hw_param_interval(params,
789 SNDRV_PCM_HW_PARAM_CHANNELS);
790
791 rate->min = rate->max = 48000;
792 channels->min = channels->max = 2;
793
794 return 0;
795}
796
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700797static int msm8960_btsco_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
798 struct snd_pcm_hw_params *params)
799{
800 struct snd_interval *rate = hw_param_interval(params,
801 SNDRV_PCM_HW_PARAM_RATE);
802
803 struct snd_interval *channels = hw_param_interval(params,
804 SNDRV_PCM_HW_PARAM_CHANNELS);
805
806 rate->min = rate->max = msm8960_btsco_rate;
807 channels->min = channels->max = msm8960_btsco_ch;
808
809 return 0;
810}
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700811static int msm8960_auxpcm_be_params_fixup(struct snd_soc_pcm_runtime *rtd,
812 struct snd_pcm_hw_params *params)
813{
814 struct snd_interval *rate = hw_param_interval(params,
815 SNDRV_PCM_HW_PARAM_RATE);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700816
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700817 struct snd_interval *channels = hw_param_interval(params,
818 SNDRV_PCM_HW_PARAM_CHANNELS);
819
820 /* PCM only supports mono output with 8khz sample rate */
821 rate->min = rate->max = 8000;
822 channels->min = channels->max = 1;
823
824 return 0;
825}
826static int msm8960_aux_pcm_get_gpios(void)
827{
828 int ret = 0;
829
830 pr_debug("%s\n", __func__);
831
832 ret = gpio_request(GPIO_AUX_PCM_DOUT, "AUX PCM DOUT");
833 if (ret < 0) {
834 pr_err("%s: Failed to request gpio(%d): AUX PCM DOUT",
835 __func__, GPIO_AUX_PCM_DOUT);
836 goto fail_dout;
837 }
838
839 ret = gpio_request(GPIO_AUX_PCM_DIN, "AUX PCM DIN");
840 if (ret < 0) {
841 pr_err("%s: Failed to request gpio(%d): AUX PCM DIN",
842 __func__, GPIO_AUX_PCM_DIN);
843 goto fail_din;
844 }
845
846 ret = gpio_request(GPIO_AUX_PCM_SYNC, "AUX PCM SYNC");
847 if (ret < 0) {
848 pr_err("%s: Failed to request gpio(%d): AUX PCM SYNC",
849 __func__, GPIO_AUX_PCM_SYNC);
850 goto fail_sync;
851 }
852 ret = gpio_request(GPIO_AUX_PCM_CLK, "AUX PCM CLK");
853 if (ret < 0) {
854 pr_err("%s: Failed to request gpio(%d): AUX PCM CLK",
855 __func__, GPIO_AUX_PCM_CLK);
856 goto fail_clk;
857 }
858
859 return 0;
860
861fail_clk:
862 gpio_free(GPIO_AUX_PCM_SYNC);
863fail_sync:
864 gpio_free(GPIO_AUX_PCM_DIN);
865fail_din:
866 gpio_free(GPIO_AUX_PCM_DOUT);
867fail_dout:
868
869 return ret;
870}
871
872static int msm8960_aux_pcm_free_gpios(void)
873{
874 gpio_free(GPIO_AUX_PCM_DIN);
875 gpio_free(GPIO_AUX_PCM_DOUT);
876 gpio_free(GPIO_AUX_PCM_SYNC);
877 gpio_free(GPIO_AUX_PCM_CLK);
878
879 return 0;
880}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700881static int msm8960_startup(struct snd_pcm_substream *substream)
882{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700883 pr_debug("%s(): substream = %s stream = %d\n", __func__,
884 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700885 return 0;
886}
887
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700888static int msm8960_auxpcm_startup(struct snd_pcm_substream *substream)
889{
890 int ret = 0;
891
892 pr_debug("%s(): substream = %s\n", __func__, substream->name);
893 ret = msm8960_aux_pcm_get_gpios();
894 if (ret < 0) {
895 pr_err("%s: Aux PCM GPIO request failed\n", __func__);
896 return -EINVAL;
897 }
898 return 0;
899}
900
901static void msm8960_auxpcm_shutdown(struct snd_pcm_substream *substream)
902{
903
904 pr_debug("%s(): substream = %s\n", __func__, substream->name);
905 msm8960_aux_pcm_free_gpios();
906}
907
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700908static void msm8960_shutdown(struct snd_pcm_substream *substream)
909{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700910 pr_debug("%s(): substream = %s stream = %d\n", __func__,
911 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700912}
913
914static struct snd_soc_ops msm8960_be_ops = {
915 .startup = msm8960_startup,
916 .shutdown = msm8960_shutdown,
917};
918
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700919static struct snd_soc_ops msm8960_auxpcm_be_ops = {
920 .startup = msm8960_auxpcm_startup,
921 .shutdown = msm8960_auxpcm_shutdown,
922};
923
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700924/* Digital audio interface glue - connects codec <---> CPU */
925static struct snd_soc_dai_link msm8960_dai[] = {
926 /* FrontEnd DAI Links */
927 {
928 .name = "MSM8960 Media1",
929 .stream_name = "MultiMedia1",
930 .cpu_dai_name = "MultiMedia1",
931 .platform_name = "msm-pcm-dsp",
932 .dynamic = 1,
933 .dsp_link = &fe_media,
934 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA1
935 },
936 {
937 .name = "MSM8960 Media2",
938 .stream_name = "MultiMedia2",
939 .cpu_dai_name = "MultiMedia2",
940 .platform_name = "msm-pcm-dsp",
941 .dynamic = 1,
942 .dsp_link = &fe_media,
943 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA2,
944 },
945 {
946 .name = "Circuit-Switch Voice",
947 .stream_name = "CS-Voice",
948 .cpu_dai_name = "CS-VOICE",
949 .platform_name = "msm-pcm-voice",
950 .dynamic = 1,
951 .dsp_link = &fe_media,
952 .be_id = MSM_FRONTEND_DAI_CS_VOICE,
953 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -0700954 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700955 },
956 {
957 .name = "MSM VoIP",
958 .stream_name = "VoIP",
959 .cpu_dai_name = "VoIP",
960 .platform_name = "msm-voip-dsp",
961 .dynamic = 1,
962 .dsp_link = &fe_media,
963 .be_id = MSM_FRONTEND_DAI_VOIP,
964 },
965 {
966 .name = "MSM8960 LPA",
967 .stream_name = "LPA",
968 .cpu_dai_name = "MultiMedia3",
969 .platform_name = "msm-pcm-lpa",
970 .dynamic = 1,
971 .dsp_link = &lpa_fe_media,
972 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA3,
973 },
974 /* Hostless PMC purpose */
975 {
976 .name = "SLIMBUS_0 Hostless",
977 .stream_name = "SLIMBUS_0 Hostless",
978 .cpu_dai_name = "SLIMBUS0_HOSTLESS",
979 .platform_name = "msm-pcm-hostless",
980 .dynamic = 1,
981 .dsp_link = &slimbus0_hl_media,
982 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -0700983 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700984 /* .be_id = do not care */
985 },
986 {
987 .name = "INT_FM Hostless",
988 .stream_name = "INT_FM Hostless",
989 .cpu_dai_name = "INT_FM_HOSTLESS",
990 .platform_name = "msm-pcm-hostless",
991 .dynamic = 1,
992 .dsp_link = &int_fm_hl_media,
993 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -0700994 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700995 /* .be_id = do not care */
996 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +0530997 {
998 .name = "MSM AFE-PCM RX",
999 .stream_name = "AFE-PROXY RX",
1000 .cpu_dai_name = "msm-dai-q6.241",
1001 .codec_name = "msm-stub-codec.1",
1002 .codec_dai_name = "msm-stub-rx",
1003 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001004 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301005 },
1006 {
1007 .name = "MSM AFE-PCM TX",
1008 .stream_name = "AFE-PROXY TX",
1009 .cpu_dai_name = "msm-dai-q6.240",
1010 .codec_name = "msm-stub-codec.1",
1011 .codec_dai_name = "msm-stub-tx",
1012 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001013 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301014 },
Asish Bhattacharya09f9e0a2011-11-11 13:22:47 +05301015 {
1016 .name = "MSM8960 Compr",
1017 .stream_name = "COMPR",
1018 .cpu_dai_name = "MultiMedia4",
1019 .platform_name = "msm-compr-dsp",
1020 .dynamic = 1,
1021 .dsp_link = &lpa_fe_media,
1022 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA4,
1023 },
Alex Wongb3d06a02012-01-12 10:00:41 -08001024 /* HDMI Hostless */
1025 {
1026 .name = "HDMI_RX_HOSTLESS",
1027 .stream_name = "HDMI_RX_HOSTLESS",
1028 .cpu_dai_name = "HDMI_HOSTLESS",
1029 .platform_name = "msm-pcm-hostless",
1030 .dynamic = 1,
1031 .dsp_link = &hdmi_rx_hl,
1032 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1033 .no_codec = 1,
1034 .ignore_suspend = 1,
1035 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001036 /* Backend DAI Links */
1037 {
1038 .name = LPASS_BE_SLIMBUS_0_RX,
1039 .stream_name = "Slimbus Playback",
1040 .cpu_dai_name = "msm-dai-q6.16384",
1041 .platform_name = "msm-pcm-routing",
1042 .codec_name = "tabla_codec",
1043 .codec_dai_name = "tabla_rx1",
1044 .no_pcm = 1,
1045 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1046 .init = &msm8960_audrx_init,
1047 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1048 .ops = &msm8960_be_ops,
1049 },
1050 {
1051 .name = LPASS_BE_SLIMBUS_0_TX,
1052 .stream_name = "Slimbus Capture",
1053 .cpu_dai_name = "msm-dai-q6.16385",
1054 .platform_name = "msm-pcm-routing",
1055 .codec_name = "tabla_codec",
1056 .codec_dai_name = "tabla_tx1",
1057 .no_pcm = 1,
1058 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1059 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1060 .ops = &msm8960_be_ops,
1061 },
1062 /* Backend BT/FM DAI Links */
1063 {
1064 .name = LPASS_BE_INT_BT_SCO_RX,
1065 .stream_name = "Internal BT-SCO Playback",
1066 .cpu_dai_name = "msm-dai-q6.12288",
1067 .platform_name = "msm-pcm-routing",
1068 .codec_name = "msm-stub-codec.1",
1069 .codec_dai_name = "msm-stub-rx",
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001070 .init = &msm8960_btsco_init,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001071 .no_pcm = 1,
1072 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_RX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001073 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001074 },
1075 {
1076 .name = LPASS_BE_INT_BT_SCO_TX,
1077 .stream_name = "Internal BT-SCO Capture",
1078 .cpu_dai_name = "msm-dai-q6.12289",
1079 .platform_name = "msm-pcm-routing",
1080 .codec_name = "msm-stub-codec.1",
1081 .codec_dai_name = "msm-stub-tx",
1082 .no_pcm = 1,
1083 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_TX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001084 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001085 },
1086 {
1087 .name = LPASS_BE_INT_FM_RX,
1088 .stream_name = "Internal FM Playback",
1089 .cpu_dai_name = "msm-dai-q6.12292",
1090 .platform_name = "msm-pcm-routing",
1091 .codec_name = "msm-stub-codec.1",
1092 .codec_dai_name = "msm-stub-rx",
1093 .no_pcm = 1,
1094 .be_id = MSM_BACKEND_DAI_INT_FM_RX,
1095 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1096 },
1097 {
1098 .name = LPASS_BE_INT_FM_TX,
1099 .stream_name = "Internal FM Capture",
1100 .cpu_dai_name = "msm-dai-q6.12293",
1101 .platform_name = "msm-pcm-routing",
1102 .codec_name = "msm-stub-codec.1",
1103 .codec_dai_name = "msm-stub-tx",
1104 .no_pcm = 1,
1105 .be_id = MSM_BACKEND_DAI_INT_FM_TX,
1106 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1107 },
1108 /* HDMI BACK END DAI Link */
1109 {
1110 .name = LPASS_BE_HDMI,
1111 .stream_name = "HDMI Playback",
1112 .cpu_dai_name = "msm-dai-q6.8",
1113 .platform_name = "msm-pcm-routing",
1114 .codec_name = "msm-stub-codec.1",
1115 .codec_dai_name = "msm-stub-rx",
1116 .no_pcm = 1,
1117 .no_codec = 1,
1118 .be_id = MSM_BACKEND_DAI_HDMI_RX,
Helen Zeng73f7fc62011-11-18 16:29:59 -08001119 .be_hw_params_fixup = msm8960_hdmi_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001120 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301121 /* Backend AFE DAI Links */
1122 {
1123 .name = LPASS_BE_AFE_PCM_RX,
1124 .stream_name = "AFE Playback",
1125 .cpu_dai_name = "msm-dai-q6.224",
1126 .platform_name = "msm-pcm-routing",
1127 .codec_name = "msm-stub-codec.1",
1128 .codec_dai_name = "msm-stub-rx",
1129 .no_codec = 1,
1130 .no_pcm = 1,
1131 .be_id = MSM_BACKEND_DAI_AFE_PCM_RX,
1132 },
1133 {
1134 .name = LPASS_BE_AFE_PCM_TX,
1135 .stream_name = "AFE Capture",
1136 .cpu_dai_name = "msm-dai-q6.225",
1137 .platform_name = "msm-pcm-routing",
1138 .codec_name = "msm-stub-codec.1",
1139 .codec_dai_name = "msm-stub-tx",
1140 .no_codec = 1,
1141 .no_pcm = 1,
1142 .be_id = MSM_BACKEND_DAI_AFE_PCM_TX,
1143 },
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001144 /* AUX PCM Backend DAI Links */
1145 {
1146 .name = LPASS_BE_AUXPCM_RX,
1147 .stream_name = "AUX PCM Playback",
1148 .cpu_dai_name = "msm-dai-q6.2",
1149 .platform_name = "msm-pcm-routing",
1150 .codec_name = "msm-stub-codec.1",
1151 .codec_dai_name = "msm-stub-rx",
1152 .no_pcm = 1,
1153 .be_id = MSM_BACKEND_DAI_AUXPCM_RX,
1154 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1155 .ops = &msm8960_auxpcm_be_ops,
1156 },
1157 {
1158 .name = LPASS_BE_AUXPCM_TX,
1159 .stream_name = "AUX PCM Capture",
1160 .cpu_dai_name = "msm-dai-q6.3",
1161 .platform_name = "msm-pcm-routing",
1162 .codec_name = "msm-stub-codec.1",
1163 .codec_dai_name = "msm-stub-tx",
1164 .no_pcm = 1,
1165 .be_id = MSM_BACKEND_DAI_AUXPCM_TX,
1166 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1167 },
Helen Zeng0705a5f2011-10-14 15:29:52 -07001168 /* Incall Music BACK END DAI Link */
1169 {
1170 .name = LPASS_BE_VOICE_PLAYBACK_TX,
1171 .stream_name = "Voice Farend Playback",
1172 .cpu_dai_name = "msm-dai-q6.32773",
1173 .platform_name = "msm-pcm-routing",
1174 .codec_name = "msm-stub-codec.1",
1175 .codec_dai_name = "msm-stub-rx",
1176 .no_pcm = 1,
1177 .no_codec = 1,
1178 .be_id = MSM_BACKEND_DAI_VOICE_PLAYBACK_TX,
1179 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1180 },
Helen Zenge3d716a2011-10-14 16:32:16 -07001181 /* Incall Record Uplink BACK END DAI Link */
1182 {
1183 .name = LPASS_BE_INCALL_RECORD_TX,
1184 .stream_name = "Voice Uplink Capture",
1185 .cpu_dai_name = "msm-dai-q6.32772",
1186 .platform_name = "msm-pcm-routing",
1187 .codec_name = "msm-stub-codec.1",
1188 .codec_dai_name = "msm-stub-tx",
1189 .no_pcm = 1,
1190 .no_codec = 1,
1191 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_TX,
1192 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1193 },
1194 /* Incall Record Downlink BACK END DAI Link */
1195 {
1196 .name = LPASS_BE_INCALL_RECORD_RX,
1197 .stream_name = "Voice Downlink Capture",
1198 .cpu_dai_name = "msm-dai-q6.32771",
1199 .platform_name = "msm-pcm-routing",
1200 .codec_name = "msm-stub-codec.1",
1201 .codec_dai_name = "msm-stub-tx",
1202 .no_pcm = 1,
1203 .no_codec = 1,
1204 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_RX,
1205 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1206 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001207};
1208
1209struct snd_soc_card snd_soc_card_msm8960 = {
1210 .name = "msm8960-snd-card",
1211 .dai_link = msm8960_dai,
1212 .num_links = ARRAY_SIZE(msm8960_dai),
1213};
1214
1215static struct platform_device *msm8960_snd_device;
1216
1217static int msm8960_configure_headset_mic_gpios(void)
1218{
1219 int ret;
1220 struct pm_gpio param = {
1221 .direction = PM_GPIO_DIR_OUT,
1222 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
1223 .output_value = 1,
1224 .pull = PM_GPIO_PULL_NO,
1225 .vin_sel = PM_GPIO_VIN_S4,
1226 .out_strength = PM_GPIO_STRENGTH_MED,
1227 .function = PM_GPIO_FUNC_NORMAL,
1228 };
1229
1230 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(23), "AV_SWITCH");
1231 if (ret) {
1232 pr_err("%s: Failed to request gpio %d\n", __func__,
1233 PM8921_GPIO_PM_TO_SYS(23));
1234 return ret;
1235 }
1236
1237 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(23), &param);
1238 if (ret)
1239 pr_err("%s: Failed to configure gpio %d\n", __func__,
1240 PM8921_GPIO_PM_TO_SYS(23));
1241 else
1242 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(23), 0);
1243
1244 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(35), "US_EURO_SWITCH");
1245 if (ret) {
1246 pr_err("%s: Failed to request gpio %d\n", __func__,
1247 PM8921_GPIO_PM_TO_SYS(35));
1248 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1249 return ret;
1250 }
1251 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(35), &param);
1252 if (ret)
1253 pr_err("%s: Failed to configure gpio %d\n", __func__,
1254 PM8921_GPIO_PM_TO_SYS(35));
1255 else
Patrick Lai2ae0a002011-09-26 14:51:09 -07001256 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(35), 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001257
1258 return 0;
1259}
1260static void msm8960_free_headset_mic_gpios(void)
1261{
1262 if (msm8960_headset_gpios_configured) {
1263 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1264 gpio_free(PM8921_GPIO_PM_TO_SYS(35));
1265 }
1266}
1267
1268static int __init msm8960_audio_init(void)
1269{
1270 int ret;
1271
Ajay Dudani79e268c2011-12-28 13:14:44 -08001272 if (!cpu_is_msm8960() && !cpu_is_msm8930()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001273 pr_err("%s: Not the right machine type\n", __func__);
1274 return -ENODEV ;
1275 }
Joonwoo Park0976d012011-12-22 11:48:18 -08001276
1277 tabla_mbhc_cal = def_tabla_mbhc_cal();
1278 if (!tabla_mbhc_cal) {
1279 pr_err("Calibration data allocation failed\n");
1280 return -ENOMEM;
1281 }
1282
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001283 msm8960_snd_device = platform_device_alloc("soc-audio", 0);
1284 if (!msm8960_snd_device) {
1285 pr_err("Platform device allocation failed\n");
Joonwoo Park0976d012011-12-22 11:48:18 -08001286 kfree(tabla_mbhc_cal);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001287 return -ENOMEM;
1288 }
1289
1290 platform_set_drvdata(msm8960_snd_device, &snd_soc_card_msm8960);
1291 ret = platform_device_add(msm8960_snd_device);
1292 if (ret) {
1293 platform_device_put(msm8960_snd_device);
Joonwoo Park0976d012011-12-22 11:48:18 -08001294 kfree(tabla_mbhc_cal);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001295 return ret;
1296 }
1297
1298 if (msm8960_configure_headset_mic_gpios()) {
1299 pr_err("%s Fail to configure headset mic gpios\n", __func__);
1300 msm8960_headset_gpios_configured = 0;
1301 } else
1302 msm8960_headset_gpios_configured = 1;
1303
1304 return ret;
1305
1306}
1307module_init(msm8960_audio_init);
1308
1309static void __exit msm8960_audio_exit(void)
1310{
Ajay Dudani79e268c2011-12-28 13:14:44 -08001311 if (!cpu_is_msm8960() && !cpu_is_msm8930()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001312 pr_err("%s: Not the right machine type\n", __func__);
1313 return ;
1314 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001315 msm8960_free_headset_mic_gpios();
1316 platform_device_unregister(msm8960_snd_device);
Joonwoo Park0976d012011-12-22 11:48:18 -08001317 kfree(tabla_mbhc_cal);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001318}
1319module_exit(msm8960_audio_exit);
1320
1321MODULE_DESCRIPTION("ALSA SoC MSM8960");
1322MODULE_LICENSE("GPL v2");