blob: d2f259c038819bf2be3cb03986ba5675e73c631e [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>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070018#include <linux/mfd/pm8xxx/pm8921.h>
Joonwoo Park0976d012011-12-22 11:48:18 -080019#include <linux/slab.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070020#include <sound/core.h>
21#include <sound/soc.h>
22#include <sound/soc-dapm.h>
23#include <sound/soc-dsp.h>
24#include <sound/pcm.h>
25#include <sound/jack.h>
Bradley Rubin229c6a52011-07-12 16:18:48 -070026#include <asm/mach-types.h>
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -080027#include <mach/socinfo.h>
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070028#include <linux/mfd/wcd9xxx/core.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
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070035#define PM8921_IRQ_BASE (NR_MSM_IRQS + NR_GPIO_IRQS)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070036#define PM8921_GPIO_PM_TO_SYS(pm_gpio) (pm_gpio - 1 + PM8921_GPIO_BASE)
37
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070038#define MSM8960_SPK_ON 1
39#define MSM8960_SPK_OFF 0
40
41#define msm8960_SLIM_0_RX_MAX_CHANNELS 2
42#define msm8960_SLIM_0_TX_MAX_CHANNELS 4
43
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -070044#define BTSCO_RATE_8KHZ 8000
45#define BTSCO_RATE_16KHZ 16000
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070046
Kiran Kandi462acea2011-08-31 23:53:14 -070047#define BOTTOM_SPK_AMP_POS 0x1
48#define BOTTOM_SPK_AMP_NEG 0x2
49#define TOP_SPK_AMP_POS 0x4
50#define TOP_SPK_AMP_NEG 0x8
51
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -070052#define GPIO_AUX_PCM_DOUT 63
53#define GPIO_AUX_PCM_DIN 64
54#define GPIO_AUX_PCM_SYNC 65
55#define GPIO_AUX_PCM_CLK 66
56
Joonwoo Park0976d012011-12-22 11:48:18 -080057#define TABLA_EXT_CLK_RATE 12288000
58
Joonwoo Park6b9b03f2012-01-23 18:48:54 -080059#define TABLA_MBHC_DEF_BUTTONS 8
Joonwoo Park0976d012011-12-22 11:48:18 -080060#define TABLA_MBHC_DEF_RLOADS 5
61
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070062#define JACK_DETECT_GPIO 38
63#define JACK_DETECT_INT PM8921_GPIO_IRQ(PM8921_IRQ_BASE, JACK_DETECT_GPIO)
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070064
Harmandeep Singh0dd82412011-11-11 09:46:17 -080065static u32 top_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(18);
66static u32 bottom_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(19);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070067static int msm8960_spk_control;
Kiran Kandi462acea2011-08-31 23:53:14 -070068static int msm8960_ext_bottom_spk_pamp;
69static int msm8960_ext_top_spk_pamp;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070070static int msm8960_slim_0_rx_ch = 1;
71static int msm8960_slim_0_tx_ch = 1;
72
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -070073static int msm8960_btsco_rate = BTSCO_RATE_8KHZ;
74static int msm8960_btsco_ch = 1;
75
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070076static struct clk *codec_clk;
77static int clk_users;
78
79static int msm8960_headset_gpios_configured;
80
81static struct snd_soc_jack hs_jack;
Bradley Rubincb1e2732011-06-23 16:49:20 -070082static struct snd_soc_jack button_jack;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070083
Joonwoo Park9115ade2012-04-18 13:14:10 -070084static bool hs_detect_use_gpio;
85module_param(hs_detect_use_gpio, bool, 0444);
86MODULE_PARM_DESC(hs_detect_use_gpio, "Use GPIO for headset detection");
87
88static bool hs_detect_use_firmware;
89module_param(hs_detect_use_firmware, bool, 0444);
90MODULE_PARM_DESC(hs_detect_use_firmware, "Use firmware for headset detection");
91
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070092static int msm8960_enable_codec_ext_clk(struct snd_soc_codec *codec, int enable,
93 bool dapm);
94
95static struct tabla_mbhc_config mbhc_cfg = {
96 .headset_jack = &hs_jack,
97 .button_jack = &button_jack,
98 .read_fw_bin = false,
99 .calibration = NULL,
100 .micbias = TABLA_MICBIAS2,
101 .mclk_cb_fn = msm8960_enable_codec_ext_clk,
102 .mclk_rate = TABLA_EXT_CLK_RATE,
103 .gpio = 0,
104 .gpio_irq = 0,
105 .gpio_level_insert = 1,
106};
Joonwoo Park0976d012011-12-22 11:48:18 -0800107
Joonwoo Park28f49c82012-03-16 12:29:21 -0700108static struct mutex cdc_mclk_mutex;
109
Kiran Kandi462acea2011-08-31 23:53:14 -0700110static void msm8960_enable_ext_spk_amp_gpio(u32 spk_amp_gpio)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700111{
112 int ret = 0;
113
114 struct pm_gpio param = {
115 .direction = PM_GPIO_DIR_OUT,
116 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
117 .output_value = 1,
118 .pull = PM_GPIO_PULL_NO,
119 .vin_sel = PM_GPIO_VIN_S4,
120 .out_strength = PM_GPIO_STRENGTH_MED,
Kiran Kandi462acea2011-08-31 23:53:14 -0700121 .
122 function = PM_GPIO_FUNC_NORMAL,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700123 };
124
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800125 if (spk_amp_gpio == bottom_spk_pamp_gpio) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700126
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800127 ret = gpio_request(bottom_spk_pamp_gpio, "BOTTOM_SPK_AMP");
Kiran Kandi462acea2011-08-31 23:53:14 -0700128 if (ret) {
129 pr_err("%s: Error requesting BOTTOM SPK AMP GPIO %u\n",
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800130 __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700131 return;
132 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800133 ret = pm8xxx_gpio_config(bottom_spk_pamp_gpio, &param);
Kiran Kandi462acea2011-08-31 23:53:14 -0700134 if (ret)
135 pr_err("%s: Failed to configure Bottom Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800136 " gpio %u\n", __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700137 else {
138 pr_debug("%s: enable Bottom spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800139 gpio_direction_output(bottom_spk_pamp_gpio, 1);
Kiran Kandi462acea2011-08-31 23:53:14 -0700140 }
141
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800142 } else if (spk_amp_gpio == top_spk_pamp_gpio) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700143
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800144 ret = gpio_request(top_spk_pamp_gpio, "TOP_SPK_AMP");
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700145 if (ret) {
146 pr_err("%s: Error requesting GPIO %d\n", __func__,
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800147 top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700148 return;
149 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800150 ret = pm8xxx_gpio_config(top_spk_pamp_gpio, &param);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700151 if (ret)
Kiran Kandi462acea2011-08-31 23:53:14 -0700152 pr_err("%s: Failed to configure Top Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800153 " gpio %u\n", __func__, top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700154 else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700155 pr_debug("%s: enable Top spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800156 gpio_direction_output(top_spk_pamp_gpio, 1);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700157 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700158 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700159 pr_err("%s: ERROR : Invalid External Speaker Ampl GPIO."
160 " gpio = %u\n", __func__, spk_amp_gpio);
161 return;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700162 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700163}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700164
Kiran Kandi462acea2011-08-31 23:53:14 -0700165static void msm8960_ext_spk_power_amp_on(u32 spk)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700166{
Kiran Kandi462acea2011-08-31 23:53:14 -0700167 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
168
169 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
170 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
171
172 pr_debug("%s() External Bottom Speaker Ampl already "
173 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700174 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700175 }
176
177 msm8960_ext_bottom_spk_pamp |= spk;
178
179 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
180 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
181
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800182 msm8960_enable_ext_spk_amp_gpio(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700183 pr_debug("%s: slepping 4 ms after turning on external "
184 " Bottom Speaker Ampl\n", __func__);
185 usleep_range(4000, 4000);
186 }
187
188 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
189
190 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
191 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
192
193 pr_debug("%s() External Top Speaker Ampl already"
194 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700195 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700196 }
197
198 msm8960_ext_top_spk_pamp |= spk;
199
200 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
201 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
202
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800203 msm8960_enable_ext_spk_amp_gpio(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700204 pr_debug("%s: sleeping 4 ms after turning on "
205 " external Top Speaker Ampl\n", __func__);
206 usleep_range(4000, 4000);
207 }
208 } else {
209
210 pr_err("%s: ERROR : Invalid External Speaker Ampl. spk = 0x%08x\n",
211 __func__, spk);
212 return;
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700213 }
Kiran Kandi462acea2011-08-31 23:53:14 -0700214}
215
216static void msm8960_ext_spk_power_amp_off(u32 spk)
217{
218 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
219
220 if (!msm8960_ext_bottom_spk_pamp)
221 return;
222
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800223 gpio_direction_output(bottom_spk_pamp_gpio, 0);
224 gpio_free(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700225 msm8960_ext_bottom_spk_pamp = 0;
226
227 pr_debug("%s: sleeping 4 ms after turning off external Bottom"
228 " Speaker Ampl\n", __func__);
229
230 usleep_range(4000, 4000);
231
232 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
233
234 if (!msm8960_ext_top_spk_pamp)
235 return;
236
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800237 gpio_direction_output(top_spk_pamp_gpio, 0);
238 gpio_free(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700239 msm8960_ext_top_spk_pamp = 0;
240
241 pr_debug("%s: sleeping 4 ms after turning off external Top"
242 " Spkaker Ampl\n", __func__);
243
244 usleep_range(4000, 4000);
245 } else {
246
247 pr_err("%s: ERROR : Invalid Ext Spk Ampl. spk = 0x%08x\n",
248 __func__, spk);
249 return;
250 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700251}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700252
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700253static void msm8960_ext_control(struct snd_soc_codec *codec)
254{
255 struct snd_soc_dapm_context *dapm = &codec->dapm;
256
257 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700258 if (msm8960_spk_control == MSM8960_SPK_ON) {
259 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
260 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
261 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
262 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
263 } else {
264 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Pos");
265 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Neg");
266 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Pos");
267 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Neg");
268 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700269
270 snd_soc_dapm_sync(dapm);
271}
272
273static int msm8960_get_spk(struct snd_kcontrol *kcontrol,
274 struct snd_ctl_elem_value *ucontrol)
275{
276 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
277 ucontrol->value.integer.value[0] = msm8960_spk_control;
278 return 0;
279}
280static int msm8960_set_spk(struct snd_kcontrol *kcontrol,
281 struct snd_ctl_elem_value *ucontrol)
282{
283 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
284
285 pr_debug("%s()\n", __func__);
286 if (msm8960_spk_control == ucontrol->value.integer.value[0])
287 return 0;
288
289 msm8960_spk_control = ucontrol->value.integer.value[0];
290 msm8960_ext_control(codec);
291 return 1;
292}
293static int msm8960_spkramp_event(struct snd_soc_dapm_widget *w,
294 struct snd_kcontrol *k, int event)
295{
296 pr_debug("%s() %x\n", __func__, SND_SOC_DAPM_EVENT_ON(event));
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700297
298 if (SND_SOC_DAPM_EVENT_ON(event)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700299 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
300 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_POS);
301 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
302 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_NEG);
303 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
304 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_POS);
305 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
306 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_NEG);
307 else {
308 pr_err("%s() Invalid Speaker Widget = %s\n",
309 __func__, w->name);
310 return -EINVAL;
311 }
312
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700313 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700314 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
315 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_POS);
316 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
317 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_NEG);
318 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
319 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_POS);
320 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
321 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_NEG);
322 else {
323 pr_err("%s() Invalid Speaker Widget = %s\n",
324 __func__, w->name);
325 return -EINVAL;
326 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700327 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700328 return 0;
329}
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700330
331static int msm8960_enable_codec_ext_clk(struct snd_soc_codec *codec, int enable,
332 bool dapm)
Joonwoo Park0976d012011-12-22 11:48:18 -0800333{
Joonwoo Park28f49c82012-03-16 12:29:21 -0700334 int r = 0;
Joonwoo Park0976d012011-12-22 11:48:18 -0800335 pr_debug("%s: enable = %d\n", __func__, enable);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700336
337 mutex_lock(&cdc_mclk_mutex);
Joonwoo Park0976d012011-12-22 11:48:18 -0800338 if (enable) {
339 clk_users++;
340 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700341 if (clk_users == 1) {
342 if (codec_clk) {
343 clk_set_rate(codec_clk, TABLA_EXT_CLK_RATE);
344 clk_enable(codec_clk);
345 tabla_mclk_enable(codec, 1, dapm);
346 } else {
347 pr_err("%s: Error setting Tabla MCLK\n",
348 __func__);
349 clk_users--;
350 r = -EINVAL;
351 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800352 }
353 } else {
Joonwoo Park28f49c82012-03-16 12:29:21 -0700354 if (clk_users > 0) {
355 clk_users--;
356 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
357 if (clk_users == 0) {
358 pr_debug("%s: disabling MCLK. clk_users = %d\n",
Joonwoo Park0976d012011-12-22 11:48:18 -0800359 __func__, clk_users);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700360 tabla_mclk_enable(codec, 0, dapm);
Joonwoo Park20b5c8b2012-04-05 15:23:44 -0700361 clk_disable(codec_clk);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700362 }
363 } else {
364 pr_err("%s: Error releasing Tabla MCLK\n", __func__);
365 r = -EINVAL;
Joonwoo Park0976d012011-12-22 11:48:18 -0800366 }
367 }
Joonwoo Park28f49c82012-03-16 12:29:21 -0700368 mutex_unlock(&cdc_mclk_mutex);
369 return r;
Joonwoo Park0976d012011-12-22 11:48:18 -0800370}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700371
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700372static int msm8960_mclk_event(struct snd_soc_dapm_widget *w,
373 struct snd_kcontrol *kcontrol, int event)
374{
375 pr_debug("%s: event = %d\n", __func__, event);
376
377 switch (event) {
378 case SND_SOC_DAPM_PRE_PMU:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700379 return msm8960_enable_codec_ext_clk(w->codec, 1, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700380 case SND_SOC_DAPM_POST_PMD:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700381 return msm8960_enable_codec_ext_clk(w->codec, 0, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700382 }
383 return 0;
384}
385
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700386static const struct snd_soc_dapm_widget msm8960_dapm_widgets[] = {
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700387
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700388 SND_SOC_DAPM_SUPPLY("MCLK", SND_SOC_NOPM, 0, 0,
389 msm8960_mclk_event, SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
390
Kiran Kandi462acea2011-08-31 23:53:14 -0700391 SND_SOC_DAPM_SPK("Ext Spk Bottom Pos", msm8960_spkramp_event),
392 SND_SOC_DAPM_SPK("Ext Spk Bottom Neg", msm8960_spkramp_event),
393
394 SND_SOC_DAPM_SPK("Ext Spk Top Pos", msm8960_spkramp_event),
395 SND_SOC_DAPM_SPK("Ext Spk Top Neg", msm8960_spkramp_event),
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700396
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700397 SND_SOC_DAPM_MIC("Handset Mic", NULL),
398 SND_SOC_DAPM_MIC("Headset Mic", NULL),
399 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700400 SND_SOC_DAPM_MIC("ANCRight Headset Mic", NULL),
401 SND_SOC_DAPM_MIC("ANCLeft Headset Mic", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700402
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700403 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700404 SND_SOC_DAPM_MIC("Digital Mic2", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700405 SND_SOC_DAPM_MIC("Digital Mic3", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700406 SND_SOC_DAPM_MIC("Digital Mic4", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700407 SND_SOC_DAPM_MIC("Digital Mic5", NULL),
408 SND_SOC_DAPM_MIC("Digital Mic6", NULL),
409
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700410};
411
Bradley Rubin229c6a52011-07-12 16:18:48 -0700412static const struct snd_soc_dapm_route common_audio_map[] = {
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700413
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700414 {"RX_BIAS", NULL, "MCLK"},
415 {"LDO_H", NULL, "MCLK"},
416
417 /* Speaker path */
Kiran Kandi462acea2011-08-31 23:53:14 -0700418 {"Ext Spk Bottom Pos", NULL, "LINEOUT1"},
419 {"Ext Spk Bottom Neg", NULL, "LINEOUT3"},
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700420
Kiran Kandi462acea2011-08-31 23:53:14 -0700421 {"Ext Spk Top Pos", NULL, "LINEOUT2"},
422 {"Ext Spk Top Neg", NULL, "LINEOUT4"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700423
424 /* Microphone path */
Bradley Rubin229c6a52011-07-12 16:18:48 -0700425 {"AMIC1", NULL, "MIC BIAS1 Internal1"},
426 {"MIC BIAS1 Internal1", NULL, "Handset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700427
428 {"AMIC2", NULL, "MIC BIAS2 External"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700429 {"MIC BIAS2 External", NULL, "Headset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700430
Patrick Laie519dd62011-09-28 12:37:11 -0700431 /**
432 * AMIC3 and AMIC4 inputs are connected to ANC microphones
433 * These mics are biased differently on CDP and FLUID
434 * routing entries below are based on bias arrangement
435 * on FLUID.
436 */
437 {"AMIC3", NULL, "MIC BIAS3 Internal1"},
438 {"MIC BIAS3 Internal1", NULL, "ANCRight Headset Mic"},
439
440 {"AMIC4", NULL, "MIC BIAS1 Internal2"},
441 {"MIC BIAS1 Internal2", NULL, "ANCLeft Headset Mic"},
442
Kiran Kandie9bf86a2011-07-21 16:50:41 -0700443 {"HEADPHONE", NULL, "LDO_H"},
444
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700445 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700446 * The digital Mic routes are setup considering
447 * fluid as default device.
448 */
449
450 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700451 * Digital Mic1. Front Bottom left Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700452 * Digital Mic GM5 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700453 * Conncted to DMIC2 Input on Tabla codec.
454 */
455 {"DMIC2", NULL, "MIC BIAS1 External"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700456 {"MIC BIAS1 External", NULL, "Digital Mic1"},
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700457
458 /**
459 * Digital Mic2. Front Bottom right Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700460 * Digital Mic GM6 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700461 * Conncted to DMIC1 Input on Tabla codec.
462 */
463 {"DMIC1", NULL, "MIC BIAS1 External"},
464 {"MIC BIAS1 External", NULL, "Digital Mic2"},
465
466 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700467 * Digital Mic3. Back Bottom Digital Mic on Fluid.
468 * Digital Mic GM1 on CDP mainboard.
469 * Conncted to DMIC4 Input on Tabla codec.
470 */
471 {"DMIC4", NULL, "MIC BIAS3 External"},
472 {"MIC BIAS3 External", NULL, "Digital Mic3"},
473
474 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700475 * Digital Mic4. Back top Digital Mic on Fluid.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700476 * Digital Mic GM2 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700477 * Conncted to DMIC3 Input on Tabla codec.
478 */
479 {"DMIC3", NULL, "MIC BIAS3 External"},
480 {"MIC BIAS3 External", NULL, "Digital Mic4"},
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700481
482 /**
483 * Digital Mic5. Front top Digital Mic on Fluid.
484 * Digital Mic GM3 on CDP mainboard.
485 * Conncted to DMIC5 Input on Tabla codec.
486 */
487 {"DMIC5", NULL, "MIC BIAS4 External"},
488 {"MIC BIAS4 External", NULL, "Digital Mic5"},
489
Patrick Laicb7802b2011-10-04 12:39:18 -0700490 /* Tabla digital Mic6 - back bottom digital Mic on Liquid and
491 * bottom mic on CDP. FLUID/MTP do not have dmic6 installed.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700492 */
493 {"DMIC6", NULL, "MIC BIAS4 External"},
494 {"MIC BIAS4 External", NULL, "Digital Mic6"},
Bradley Rubin229c6a52011-07-12 16:18:48 -0700495};
496
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700497static const char *spk_function[] = {"Off", "On"};
Patrick Lai9f4b4292011-07-16 22:11:09 -0700498static const char *slim0_rx_ch_text[] = {"One", "Two"};
499static const char *slim0_tx_ch_text[] = {"One", "Two", "Three", "Four"};
500
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700501static const struct soc_enum msm8960_enum[] = {
502 SOC_ENUM_SINGLE_EXT(2, spk_function),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700503 SOC_ENUM_SINGLE_EXT(2, slim0_rx_ch_text),
504 SOC_ENUM_SINGLE_EXT(4, slim0_tx_ch_text),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700505};
506
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700507static const char *btsco_rate_text[] = {"8000", "16000"};
508static const struct soc_enum msm8960_btsco_enum[] = {
509 SOC_ENUM_SINGLE_EXT(2, btsco_rate_text),
510};
511
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700512static int msm8960_slim_0_rx_ch_get(struct snd_kcontrol *kcontrol,
513 struct snd_ctl_elem_value *ucontrol)
514{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700515 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800516 msm8960_slim_0_rx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700517 ucontrol->value.integer.value[0] = msm8960_slim_0_rx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700518 return 0;
519}
520
521static int msm8960_slim_0_rx_ch_put(struct snd_kcontrol *kcontrol,
522 struct snd_ctl_elem_value *ucontrol)
523{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700524 msm8960_slim_0_rx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700525
Patrick Lai9f4b4292011-07-16 22:11:09 -0700526 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800527 msm8960_slim_0_rx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700528 return 1;
529}
530
531static int msm8960_slim_0_tx_ch_get(struct snd_kcontrol *kcontrol,
532 struct snd_ctl_elem_value *ucontrol)
533{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700534 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800535 msm8960_slim_0_tx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700536 ucontrol->value.integer.value[0] = msm8960_slim_0_tx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700537 return 0;
538}
539
540static int msm8960_slim_0_tx_ch_put(struct snd_kcontrol *kcontrol,
541 struct snd_ctl_elem_value *ucontrol)
542{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700543 msm8960_slim_0_tx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700544
545 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800546 msm8960_slim_0_tx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700547 return 1;
548}
549
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700550static int msm8960_btsco_rate_get(struct snd_kcontrol *kcontrol,
551 struct snd_ctl_elem_value *ucontrol)
552{
Joonwoo Park0976d012011-12-22 11:48:18 -0800553 pr_debug("%s: msm8960_btsco_rate = %d", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700554 ucontrol->value.integer.value[0] = msm8960_btsco_rate;
555 return 0;
556}
557
558static int msm8960_btsco_rate_put(struct snd_kcontrol *kcontrol,
559 struct snd_ctl_elem_value *ucontrol)
560{
561 switch (ucontrol->value.integer.value[0]) {
562 case 0:
563 msm8960_btsco_rate = BTSCO_RATE_8KHZ;
564 break;
565 case 1:
566 msm8960_btsco_rate = BTSCO_RATE_16KHZ;
567 break;
568 default:
569 msm8960_btsco_rate = BTSCO_RATE_8KHZ;
570 break;
571 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800572 pr_debug("%s: msm8960_btsco_rate = %d\n", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700573 return 0;
574}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700575
576static const struct snd_kcontrol_new tabla_msm8960_controls[] = {
577 SOC_ENUM_EXT("Speaker Function", msm8960_enum[0], msm8960_get_spk,
578 msm8960_set_spk),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700579 SOC_ENUM_EXT("SLIM_0_RX Channels", msm8960_enum[1],
580 msm8960_slim_0_rx_ch_get, msm8960_slim_0_rx_ch_put),
581 SOC_ENUM_EXT("SLIM_0_TX Channels", msm8960_enum[2],
582 msm8960_slim_0_tx_ch_get, msm8960_slim_0_tx_ch_put),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700583};
584
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700585static const struct snd_kcontrol_new int_btsco_rate_mixer_controls[] = {
586 SOC_ENUM_EXT("Internal BTSCO SampleRate", msm8960_btsco_enum[0],
587 msm8960_btsco_rate_get, msm8960_btsco_rate_put),
588};
589
590static int msm8960_btsco_init(struct snd_soc_pcm_runtime *rtd)
591{
592 int err = 0;
593 struct snd_soc_platform *platform = rtd->platform;
594
595 err = snd_soc_add_platform_controls(platform,
596 int_btsco_rate_mixer_controls,
597 ARRAY_SIZE(int_btsco_rate_mixer_controls));
598 if (err < 0)
599 return err;
600 return 0;
601}
602
Joonwoo Park0976d012011-12-22 11:48:18 -0800603static void *def_tabla_mbhc_cal(void)
604{
605 void *tabla_cal;
606 struct tabla_mbhc_btn_detect_cfg *btn_cfg;
607 u16 *btn_low, *btn_high;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800608 u8 *n_ready, *n_cic, *gain;
Joonwoo Park0976d012011-12-22 11:48:18 -0800609
610 tabla_cal = kzalloc(TABLA_MBHC_CAL_SIZE(TABLA_MBHC_DEF_BUTTONS,
611 TABLA_MBHC_DEF_RLOADS),
612 GFP_KERNEL);
613 if (!tabla_cal) {
614 pr_err("%s: out of memory\n", __func__);
615 return NULL;
616 }
617
618#define S(X, Y) ((TABLA_MBHC_CAL_GENERAL_PTR(tabla_cal)->X) = (Y))
619 S(t_ldoh, 100);
620 S(t_bg_fast_settle, 100);
621 S(t_shutdown_plug_rem, 255);
622 S(mbhc_nsa, 4);
623 S(mbhc_navg, 4);
624#undef S
625#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_DET_PTR(tabla_cal)->X) = (Y))
626 S(mic_current, TABLA_PID_MIC_5_UA);
627 S(hph_current, TABLA_PID_MIC_5_UA);
628 S(t_mic_pid, 100);
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800629 S(t_ins_complete, 250);
Joonwoo Park0976d012011-12-22 11:48:18 -0800630 S(t_ins_retry, 200);
631#undef S
632#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla_cal)->X) = (Y))
633 S(v_no_mic, 30);
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700634 S(v_hs_max, 2400);
Joonwoo Park0976d012011-12-22 11:48:18 -0800635#undef S
636#define S(X, Y) ((TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal)->X) = (Y))
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800637 S(c[0], 62);
638 S(c[1], 124);
639 S(nc, 1);
640 S(n_meas, 3);
Joonwoo Park0976d012011-12-22 11:48:18 -0800641 S(mbhc_nsc, 11);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800642 S(n_btn_meas, 1);
643 S(n_btn_con, 2);
Joonwoo Park0976d012011-12-22 11:48:18 -0800644 S(num_btn, TABLA_MBHC_DEF_BUTTONS);
645 S(v_btn_press_delta_sta, 100);
646 S(v_btn_press_delta_cic, 50);
647#undef S
648 btn_cfg = TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal);
649 btn_low = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_LOW);
650 btn_high = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_HIGH);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800651 btn_low[0] = -50;
652 btn_high[0] = 10;
653 btn_low[1] = 11;
654 btn_high[1] = 38;
655 btn_low[2] = 39;
656 btn_high[2] = 64;
657 btn_low[3] = 65;
658 btn_high[3] = 91;
659 btn_low[4] = 92;
660 btn_high[4] = 115;
661 btn_low[5] = 116;
662 btn_high[5] = 141;
663 btn_low[6] = 142;
664 btn_high[6] = 163;
665 btn_low[7] = 164;
666 btn_high[7] = 250;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800667 n_ready = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_READY);
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700668 n_ready[0] = 80;
669 n_ready[1] = 68;
Joonwoo Park0976d012011-12-22 11:48:18 -0800670 n_cic = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_CIC);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800671 n_cic[0] = 60;
672 n_cic[1] = 47;
Joonwoo Park0976d012011-12-22 11:48:18 -0800673 gain = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_GAIN);
674 gain[0] = 11;
675 gain[1] = 9;
676
677 return tabla_cal;
678}
679
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800680static int msm8960_hw_params(struct snd_pcm_substream *substream,
681 struct snd_pcm_hw_params *params)
682{
683 struct snd_soc_pcm_runtime *rtd = substream->private_data;
684 struct snd_soc_dai *codec_dai = rtd->codec_dai;
685 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
686 int ret = 0;
687 unsigned int rx_ch[SLIM_MAX_RX_PORTS], tx_ch[SLIM_MAX_TX_PORTS];
688 unsigned int rx_ch_cnt = 0, tx_ch_cnt = 0;
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700689 unsigned int user_set_tx_ch = 0;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800690
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700691
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800692 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700693
694 pr_debug("%s: rx_0_ch=%d\n", __func__, msm8960_slim_0_rx_ch);
695
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800696 ret = snd_soc_dai_get_channel_map(codec_dai,
697 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
698 if (ret < 0) {
699 pr_err("%s: failed to get codec chan map\n", __func__);
700 goto end;
701 }
702
703 ret = snd_soc_dai_set_channel_map(cpu_dai, 0, 0,
704 msm8960_slim_0_rx_ch, rx_ch);
705 if (ret < 0) {
706 pr_err("%s: failed to set cpu chan map\n", __func__);
707 goto end;
708 }
709 ret = snd_soc_dai_set_channel_map(codec_dai, 0, 0,
710 msm8960_slim_0_rx_ch, rx_ch);
711 if (ret < 0) {
712 pr_err("%s: failed to set codec channel map\n",
713 __func__);
714 goto end;
715 }
716 } else {
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700717
718 if (codec_dai->id == 2)
719 user_set_tx_ch = msm8960_slim_0_tx_ch;
720 else if (codec_dai->id == 4)
721 user_set_tx_ch = params_channels(params);
722
723 pr_debug("%s: %s_tx_dai_id_%d_ch=%d\n", __func__,
724 codec_dai->name, codec_dai->id, user_set_tx_ch);
725
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800726 ret = snd_soc_dai_get_channel_map(codec_dai,
727 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
728 if (ret < 0) {
729 pr_err("%s: failed to get codec chan map\n", __func__);
730 goto end;
731 }
732 ret = snd_soc_dai_set_channel_map(cpu_dai,
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700733 user_set_tx_ch, tx_ch, 0 , 0);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800734 if (ret < 0) {
735 pr_err("%s: failed to set cpu chan map\n", __func__);
736 goto end;
737 }
738 ret = snd_soc_dai_set_channel_map(codec_dai,
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700739 user_set_tx_ch, tx_ch, 0, 0);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800740 if (ret < 0) {
741 pr_err("%s: failed to set codec channel map\n",
742 __func__);
743 goto end;
744 }
745
746
747 }
748end:
749 return ret;
750}
751
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700752static int msm8960_audrx_init(struct snd_soc_pcm_runtime *rtd)
753{
754 int err;
755 struct snd_soc_codec *codec = rtd->codec;
756 struct snd_soc_dapm_context *dapm = &codec->dapm;
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800757 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700758 struct pm_gpio jack_gpio_cfg = {
759 .direction = PM_GPIO_DIR_IN,
760 .pull = PM_GPIO_PULL_UP_1P5,
761 .function = PM_GPIO_FUNC_NORMAL,
762 .vin_sel = 2,
763 .inv_int_pol = 0,
764 };
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700765
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800766 pr_debug("%s(), dev_name%s\n", __func__, dev_name(cpu_dai->dev));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700767
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800768 if (machine_is_msm8960_liquid()) {
769 top_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(19));
770 bottom_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(18));
771 }
772
Kiran Kandi28ef14e2011-09-16 15:45:47 -0700773 rtd->pmdown_time = 0;
774
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700775 err = snd_soc_add_controls(codec, tabla_msm8960_controls,
776 ARRAY_SIZE(tabla_msm8960_controls));
777 if (err < 0)
778 return err;
779
780 snd_soc_dapm_new_controls(dapm, msm8960_dapm_widgets,
781 ARRAY_SIZE(msm8960_dapm_widgets));
782
Bradley Rubin229c6a52011-07-12 16:18:48 -0700783 snd_soc_dapm_add_routes(dapm, common_audio_map,
784 ARRAY_SIZE(common_audio_map));
785
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700786 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
787 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
788 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
789 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700790
791 snd_soc_dapm_sync(dapm);
792
793 err = snd_soc_jack_new(codec, "Headset Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800794 (SND_JACK_HEADSET | SND_JACK_OC_HPHL |
795 SND_JACK_OC_HPHR),
796 &hs_jack);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700797 if (err) {
798 pr_err("failed to create new jack\n");
799 return err;
800 }
Bradley Rubincb1e2732011-06-23 16:49:20 -0700801
802 err = snd_soc_jack_new(codec, "Button Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800803 TABLA_JACK_BUTTON_MASK, &button_jack);
Bradley Rubincb1e2732011-06-23 16:49:20 -0700804 if (err) {
805 pr_err("failed to create new jack\n");
806 return err;
807 }
808
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800809 codec_clk = clk_get(cpu_dai->dev, "osr_clk");
810
Joonwoo Park9115ade2012-04-18 13:14:10 -0700811 if (hs_detect_use_gpio) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700812 mbhc_cfg.gpio = PM8921_GPIO_PM_TO_SYS(JACK_DETECT_GPIO);
813 mbhc_cfg.gpio_irq = JACK_DETECT_INT;
814 }
815
816 if (mbhc_cfg.gpio) {
817 err = pm8xxx_gpio_config(mbhc_cfg.gpio, &jack_gpio_cfg);
818 if (err) {
819 pr_err("%s: pm8xxx_gpio_config failed %d\n", __func__,
820 err);
821 return err;
822 }
823 }
Joonwoo Park9115ade2012-04-18 13:14:10 -0700824
825 mbhc_cfg.read_fw_bin = hs_detect_use_firmware;
826
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700827 err = tabla_hs_detect(codec, &mbhc_cfg);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700828
Joonwoo Park03324832012-03-19 19:36:16 -0700829 return err;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700830}
831
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700832static struct snd_soc_dsp_link lpa_fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700833 .playback = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700834 .trigger = {
835 SND_SOC_DSP_TRIGGER_POST,
836 SND_SOC_DSP_TRIGGER_POST
837 },
838};
839
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700840static struct snd_soc_dsp_link fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700841 .playback = true,
842 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700843 .trigger = {
844 SND_SOC_DSP_TRIGGER_POST,
845 SND_SOC_DSP_TRIGGER_POST
846 },
847};
848
Patrick Lai9b56e1d2012-01-22 22:14:15 -0800849/* bi-directional media definition for hostless PCM device */
850static struct snd_soc_dsp_link bidir_hl_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700851 .playback = true,
852 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700853 .trigger = {
854 SND_SOC_DSP_TRIGGER_POST,
855 SND_SOC_DSP_TRIGGER_POST
856 },
857};
858
Alex Wongb3d06a02012-01-12 10:00:41 -0800859static struct snd_soc_dsp_link hdmi_rx_hl = {
860 .playback = true,
861 .trigger = {
862 SND_SOC_DSP_TRIGGER_POST,
863 SND_SOC_DSP_TRIGGER_POST
864 },
865};
866
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700867static int msm8960_slim_0_rx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
868 struct snd_pcm_hw_params *params)
869{
870 struct snd_interval *rate = hw_param_interval(params,
871 SNDRV_PCM_HW_PARAM_RATE);
872
873 struct snd_interval *channels = hw_param_interval(params,
874 SNDRV_PCM_HW_PARAM_CHANNELS);
875
876 pr_debug("%s()\n", __func__);
877 rate->min = rate->max = 48000;
878 channels->min = channels->max = msm8960_slim_0_rx_ch;
879
880 return 0;
881}
882
883static int msm8960_slim_0_tx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
884 struct snd_pcm_hw_params *params)
885{
886 struct snd_interval *rate = hw_param_interval(params,
887 SNDRV_PCM_HW_PARAM_RATE);
888
889 struct snd_interval *channels = hw_param_interval(params,
890 SNDRV_PCM_HW_PARAM_CHANNELS);
891
892 pr_debug("%s()\n", __func__);
893 rate->min = rate->max = 48000;
894 channels->min = channels->max = msm8960_slim_0_tx_ch;
895
896 return 0;
897}
898
899static int msm8960_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
900 struct snd_pcm_hw_params *params)
901{
902 struct snd_interval *rate = hw_param_interval(params,
903 SNDRV_PCM_HW_PARAM_RATE);
904
905 pr_debug("%s()\n", __func__);
906 rate->min = rate->max = 48000;
907
908 return 0;
909}
910
Helen Zeng73f7fc62011-11-18 16:29:59 -0800911static int msm8960_hdmi_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
912 struct snd_pcm_hw_params *params)
913{
914 struct snd_interval *rate = hw_param_interval(params,
915 SNDRV_PCM_HW_PARAM_RATE);
916
917 struct snd_interval *channels = hw_param_interval(params,
918 SNDRV_PCM_HW_PARAM_CHANNELS);
919
Kiran Kandi5e809b02012-01-31 00:24:33 -0800920 pr_debug("%s channels->min %u channels->max %u ()\n", __func__,
921 channels->min, channels->max);
922
Helen Zeng73f7fc62011-11-18 16:29:59 -0800923 rate->min = rate->max = 48000;
Helen Zeng73f7fc62011-11-18 16:29:59 -0800924
925 return 0;
926}
927
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700928static int msm8960_btsco_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
929 struct snd_pcm_hw_params *params)
930{
931 struct snd_interval *rate = hw_param_interval(params,
932 SNDRV_PCM_HW_PARAM_RATE);
933
934 struct snd_interval *channels = hw_param_interval(params,
935 SNDRV_PCM_HW_PARAM_CHANNELS);
936
937 rate->min = rate->max = msm8960_btsco_rate;
938 channels->min = channels->max = msm8960_btsco_ch;
939
940 return 0;
941}
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700942static int msm8960_auxpcm_be_params_fixup(struct snd_soc_pcm_runtime *rtd,
943 struct snd_pcm_hw_params *params)
944{
945 struct snd_interval *rate = hw_param_interval(params,
946 SNDRV_PCM_HW_PARAM_RATE);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700947
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700948 struct snd_interval *channels = hw_param_interval(params,
949 SNDRV_PCM_HW_PARAM_CHANNELS);
950
951 /* PCM only supports mono output with 8khz sample rate */
952 rate->min = rate->max = 8000;
953 channels->min = channels->max = 1;
954
955 return 0;
956}
957static int msm8960_aux_pcm_get_gpios(void)
958{
959 int ret = 0;
960
961 pr_debug("%s\n", __func__);
962
963 ret = gpio_request(GPIO_AUX_PCM_DOUT, "AUX PCM DOUT");
964 if (ret < 0) {
965 pr_err("%s: Failed to request gpio(%d): AUX PCM DOUT",
966 __func__, GPIO_AUX_PCM_DOUT);
967 goto fail_dout;
968 }
969
970 ret = gpio_request(GPIO_AUX_PCM_DIN, "AUX PCM DIN");
971 if (ret < 0) {
972 pr_err("%s: Failed to request gpio(%d): AUX PCM DIN",
973 __func__, GPIO_AUX_PCM_DIN);
974 goto fail_din;
975 }
976
977 ret = gpio_request(GPIO_AUX_PCM_SYNC, "AUX PCM SYNC");
978 if (ret < 0) {
979 pr_err("%s: Failed to request gpio(%d): AUX PCM SYNC",
980 __func__, GPIO_AUX_PCM_SYNC);
981 goto fail_sync;
982 }
983 ret = gpio_request(GPIO_AUX_PCM_CLK, "AUX PCM CLK");
984 if (ret < 0) {
985 pr_err("%s: Failed to request gpio(%d): AUX PCM CLK",
986 __func__, GPIO_AUX_PCM_CLK);
987 goto fail_clk;
988 }
989
990 return 0;
991
992fail_clk:
993 gpio_free(GPIO_AUX_PCM_SYNC);
994fail_sync:
995 gpio_free(GPIO_AUX_PCM_DIN);
996fail_din:
997 gpio_free(GPIO_AUX_PCM_DOUT);
998fail_dout:
999
1000 return ret;
1001}
1002
1003static int msm8960_aux_pcm_free_gpios(void)
1004{
1005 gpio_free(GPIO_AUX_PCM_DIN);
1006 gpio_free(GPIO_AUX_PCM_DOUT);
1007 gpio_free(GPIO_AUX_PCM_SYNC);
1008 gpio_free(GPIO_AUX_PCM_CLK);
1009
1010 return 0;
1011}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001012static int msm8960_startup(struct snd_pcm_substream *substream)
1013{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07001014 pr_debug("%s(): substream = %s stream = %d\n", __func__,
1015 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001016 return 0;
1017}
1018
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001019static int msm8960_auxpcm_startup(struct snd_pcm_substream *substream)
1020{
1021 int ret = 0;
1022
1023 pr_debug("%s(): substream = %s\n", __func__, substream->name);
1024 ret = msm8960_aux_pcm_get_gpios();
1025 if (ret < 0) {
1026 pr_err("%s: Aux PCM GPIO request failed\n", __func__);
1027 return -EINVAL;
1028 }
1029 return 0;
1030}
1031
1032static void msm8960_auxpcm_shutdown(struct snd_pcm_substream *substream)
1033{
1034
1035 pr_debug("%s(): substream = %s\n", __func__, substream->name);
1036 msm8960_aux_pcm_free_gpios();
1037}
1038
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001039static void msm8960_shutdown(struct snd_pcm_substream *substream)
1040{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07001041 pr_debug("%s(): substream = %s stream = %d\n", __func__,
1042 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001043}
1044
1045static struct snd_soc_ops msm8960_be_ops = {
1046 .startup = msm8960_startup,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08001047 .hw_params = msm8960_hw_params,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001048 .shutdown = msm8960_shutdown,
1049};
1050
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001051static struct snd_soc_ops msm8960_auxpcm_be_ops = {
1052 .startup = msm8960_auxpcm_startup,
1053 .shutdown = msm8960_auxpcm_shutdown,
1054};
1055
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001056/* Digital audio interface glue - connects codec <---> CPU */
Kuirong Wangb25838e2012-01-16 23:37:23 -08001057static struct snd_soc_dai_link msm8960_dai_common[] = {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001058 /* FrontEnd DAI Links */
1059 {
1060 .name = "MSM8960 Media1",
1061 .stream_name = "MultiMedia1",
1062 .cpu_dai_name = "MultiMedia1",
1063 .platform_name = "msm-pcm-dsp",
1064 .dynamic = 1,
1065 .dsp_link = &fe_media,
1066 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA1
1067 },
1068 {
1069 .name = "MSM8960 Media2",
1070 .stream_name = "MultiMedia2",
1071 .cpu_dai_name = "MultiMedia2",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001072 .platform_name = "msm-multi-ch-pcm-dsp",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001073 .dynamic = 1,
1074 .dsp_link = &fe_media,
1075 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA2,
1076 },
1077 {
1078 .name = "Circuit-Switch Voice",
1079 .stream_name = "CS-Voice",
1080 .cpu_dai_name = "CS-VOICE",
1081 .platform_name = "msm-pcm-voice",
1082 .dynamic = 1,
1083 .dsp_link = &fe_media,
1084 .be_id = MSM_FRONTEND_DAI_CS_VOICE,
1085 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001086 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001087 },
1088 {
1089 .name = "MSM VoIP",
1090 .stream_name = "VoIP",
1091 .cpu_dai_name = "VoIP",
1092 .platform_name = "msm-voip-dsp",
1093 .dynamic = 1,
1094 .dsp_link = &fe_media,
1095 .be_id = MSM_FRONTEND_DAI_VOIP,
1096 },
1097 {
1098 .name = "MSM8960 LPA",
1099 .stream_name = "LPA",
1100 .cpu_dai_name = "MultiMedia3",
1101 .platform_name = "msm-pcm-lpa",
1102 .dynamic = 1,
1103 .dsp_link = &lpa_fe_media,
1104 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA3,
1105 },
1106 /* Hostless PMC purpose */
1107 {
1108 .name = "SLIMBUS_0 Hostless",
1109 .stream_name = "SLIMBUS_0 Hostless",
1110 .cpu_dai_name = "SLIMBUS0_HOSTLESS",
1111 .platform_name = "msm-pcm-hostless",
1112 .dynamic = 1,
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001113 .dsp_link = &bidir_hl_media,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001114 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001115 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001116 /* .be_id = do not care */
1117 },
1118 {
1119 .name = "INT_FM Hostless",
1120 .stream_name = "INT_FM Hostless",
1121 .cpu_dai_name = "INT_FM_HOSTLESS",
1122 .platform_name = "msm-pcm-hostless",
1123 .dynamic = 1,
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001124 .dsp_link = &bidir_hl_media,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001125 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001126 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001127 /* .be_id = do not care */
1128 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301129 {
1130 .name = "MSM AFE-PCM RX",
1131 .stream_name = "AFE-PROXY RX",
1132 .cpu_dai_name = "msm-dai-q6.241",
1133 .codec_name = "msm-stub-codec.1",
1134 .codec_dai_name = "msm-stub-rx",
1135 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001136 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301137 },
1138 {
1139 .name = "MSM AFE-PCM TX",
1140 .stream_name = "AFE-PROXY TX",
1141 .cpu_dai_name = "msm-dai-q6.240",
1142 .codec_name = "msm-stub-codec.1",
1143 .codec_dai_name = "msm-stub-tx",
1144 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001145 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301146 },
Asish Bhattacharya09f9e0a2011-11-11 13:22:47 +05301147 {
1148 .name = "MSM8960 Compr",
1149 .stream_name = "COMPR",
1150 .cpu_dai_name = "MultiMedia4",
1151 .platform_name = "msm-compr-dsp",
1152 .dynamic = 1,
1153 .dsp_link = &lpa_fe_media,
1154 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA4,
1155 },
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001156 {
1157 .name = "AUXPCM Hostless",
1158 .stream_name = "AUXPCM Hostless",
1159 .cpu_dai_name = "AUXPCM_HOSTLESS",
1160 .platform_name = "msm-pcm-hostless",
1161 .dynamic = 1,
1162 .dsp_link = &bidir_hl_media,
1163 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1164 .ignore_suspend = 1,
1165 },
Alex Wongb3d06a02012-01-12 10:00:41 -08001166 /* HDMI Hostless */
1167 {
1168 .name = "HDMI_RX_HOSTLESS",
1169 .stream_name = "HDMI_RX_HOSTLESS",
1170 .cpu_dai_name = "HDMI_HOSTLESS",
1171 .platform_name = "msm-pcm-hostless",
1172 .dynamic = 1,
1173 .dsp_link = &hdmi_rx_hl,
1174 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1175 .no_codec = 1,
1176 .ignore_suspend = 1,
1177 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001178 /* Backend BT/FM DAI Links */
1179 {
1180 .name = LPASS_BE_INT_BT_SCO_RX,
1181 .stream_name = "Internal BT-SCO Playback",
1182 .cpu_dai_name = "msm-dai-q6.12288",
1183 .platform_name = "msm-pcm-routing",
1184 .codec_name = "msm-stub-codec.1",
1185 .codec_dai_name = "msm-stub-rx",
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001186 .init = &msm8960_btsco_init,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001187 .no_pcm = 1,
1188 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_RX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001189 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001190 },
1191 {
1192 .name = LPASS_BE_INT_BT_SCO_TX,
1193 .stream_name = "Internal BT-SCO Capture",
1194 .cpu_dai_name = "msm-dai-q6.12289",
1195 .platform_name = "msm-pcm-routing",
1196 .codec_name = "msm-stub-codec.1",
1197 .codec_dai_name = "msm-stub-tx",
1198 .no_pcm = 1,
1199 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_TX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001200 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001201 },
1202 {
1203 .name = LPASS_BE_INT_FM_RX,
1204 .stream_name = "Internal FM Playback",
1205 .cpu_dai_name = "msm-dai-q6.12292",
1206 .platform_name = "msm-pcm-routing",
1207 .codec_name = "msm-stub-codec.1",
1208 .codec_dai_name = "msm-stub-rx",
1209 .no_pcm = 1,
1210 .be_id = MSM_BACKEND_DAI_INT_FM_RX,
1211 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1212 },
1213 {
1214 .name = LPASS_BE_INT_FM_TX,
1215 .stream_name = "Internal FM Capture",
1216 .cpu_dai_name = "msm-dai-q6.12293",
1217 .platform_name = "msm-pcm-routing",
1218 .codec_name = "msm-stub-codec.1",
1219 .codec_dai_name = "msm-stub-tx",
1220 .no_pcm = 1,
1221 .be_id = MSM_BACKEND_DAI_INT_FM_TX,
1222 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1223 },
1224 /* HDMI BACK END DAI Link */
1225 {
1226 .name = LPASS_BE_HDMI,
1227 .stream_name = "HDMI Playback",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001228 .cpu_dai_name = "msm-dai-q6-hdmi.8",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001229 .platform_name = "msm-pcm-routing",
1230 .codec_name = "msm-stub-codec.1",
1231 .codec_dai_name = "msm-stub-rx",
1232 .no_pcm = 1,
1233 .no_codec = 1,
1234 .be_id = MSM_BACKEND_DAI_HDMI_RX,
Helen Zeng73f7fc62011-11-18 16:29:59 -08001235 .be_hw_params_fixup = msm8960_hdmi_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001236 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301237 /* Backend AFE DAI Links */
1238 {
1239 .name = LPASS_BE_AFE_PCM_RX,
1240 .stream_name = "AFE Playback",
1241 .cpu_dai_name = "msm-dai-q6.224",
1242 .platform_name = "msm-pcm-routing",
1243 .codec_name = "msm-stub-codec.1",
1244 .codec_dai_name = "msm-stub-rx",
1245 .no_codec = 1,
1246 .no_pcm = 1,
1247 .be_id = MSM_BACKEND_DAI_AFE_PCM_RX,
1248 },
1249 {
1250 .name = LPASS_BE_AFE_PCM_TX,
1251 .stream_name = "AFE Capture",
1252 .cpu_dai_name = "msm-dai-q6.225",
1253 .platform_name = "msm-pcm-routing",
1254 .codec_name = "msm-stub-codec.1",
1255 .codec_dai_name = "msm-stub-tx",
1256 .no_codec = 1,
1257 .no_pcm = 1,
1258 .be_id = MSM_BACKEND_DAI_AFE_PCM_TX,
1259 },
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001260 /* AUX PCM Backend DAI Links */
1261 {
1262 .name = LPASS_BE_AUXPCM_RX,
1263 .stream_name = "AUX PCM Playback",
1264 .cpu_dai_name = "msm-dai-q6.2",
1265 .platform_name = "msm-pcm-routing",
1266 .codec_name = "msm-stub-codec.1",
1267 .codec_dai_name = "msm-stub-rx",
1268 .no_pcm = 1,
1269 .be_id = MSM_BACKEND_DAI_AUXPCM_RX,
1270 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1271 .ops = &msm8960_auxpcm_be_ops,
1272 },
1273 {
1274 .name = LPASS_BE_AUXPCM_TX,
1275 .stream_name = "AUX PCM Capture",
1276 .cpu_dai_name = "msm-dai-q6.3",
1277 .platform_name = "msm-pcm-routing",
1278 .codec_name = "msm-stub-codec.1",
1279 .codec_dai_name = "msm-stub-tx",
1280 .no_pcm = 1,
1281 .be_id = MSM_BACKEND_DAI_AUXPCM_TX,
1282 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1283 },
Helen Zeng0705a5f2011-10-14 15:29:52 -07001284 /* Incall Music BACK END DAI Link */
1285 {
1286 .name = LPASS_BE_VOICE_PLAYBACK_TX,
1287 .stream_name = "Voice Farend Playback",
1288 .cpu_dai_name = "msm-dai-q6.32773",
1289 .platform_name = "msm-pcm-routing",
1290 .codec_name = "msm-stub-codec.1",
1291 .codec_dai_name = "msm-stub-rx",
1292 .no_pcm = 1,
1293 .no_codec = 1,
1294 .be_id = MSM_BACKEND_DAI_VOICE_PLAYBACK_TX,
1295 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1296 },
Helen Zenge3d716a2011-10-14 16:32:16 -07001297 /* Incall Record Uplink BACK END DAI Link */
1298 {
1299 .name = LPASS_BE_INCALL_RECORD_TX,
1300 .stream_name = "Voice Uplink Capture",
1301 .cpu_dai_name = "msm-dai-q6.32772",
1302 .platform_name = "msm-pcm-routing",
1303 .codec_name = "msm-stub-codec.1",
1304 .codec_dai_name = "msm-stub-tx",
1305 .no_pcm = 1,
1306 .no_codec = 1,
1307 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_TX,
1308 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1309 },
1310 /* Incall Record Downlink BACK END DAI Link */
1311 {
1312 .name = LPASS_BE_INCALL_RECORD_RX,
1313 .stream_name = "Voice Downlink Capture",
1314 .cpu_dai_name = "msm-dai-q6.32771",
1315 .platform_name = "msm-pcm-routing",
1316 .codec_name = "msm-stub-codec.1",
1317 .codec_dai_name = "msm-stub-tx",
1318 .no_pcm = 1,
1319 .no_codec = 1,
1320 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_RX,
1321 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1322 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001323};
1324
Kuirong Wangb25838e2012-01-16 23:37:23 -08001325static struct snd_soc_dai_link msm8960_dai_delta_tabla1x[] = {
1326 /* Backend DAI Links */
1327 {
1328 .name = LPASS_BE_SLIMBUS_0_RX,
1329 .stream_name = "Slimbus Playback",
1330 .cpu_dai_name = "msm-dai-q6.16384",
1331 .platform_name = "msm-pcm-routing",
1332 .codec_name = "tabla1x_codec",
1333 .codec_dai_name = "tabla_rx1",
1334 .no_pcm = 1,
1335 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1336 .init = &msm8960_audrx_init,
1337 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1338 .ops = &msm8960_be_ops,
1339 },
1340 {
1341 .name = LPASS_BE_SLIMBUS_0_TX,
1342 .stream_name = "Slimbus Capture",
1343 .cpu_dai_name = "msm-dai-q6.16385",
1344 .platform_name = "msm-pcm-routing",
1345 .codec_name = "tabla1x_codec",
1346 .codec_dai_name = "tabla_tx1",
1347 .no_pcm = 1,
1348 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1349 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1350 .ops = &msm8960_be_ops,
1351 },
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001352 {
1353 .name = "SLIMBUS_2 Hostless",
1354 .stream_name = "SLIMBUS_2 Hostless",
1355 .cpu_dai_name = "msm-dai-q6.16389",
1356 .platform_name = "msm-pcm-hostless",
1357 .codec_name = "tabla1x_codec",
1358 .codec_dai_name = "tabla_tx2",
1359 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1360 .ops = &msm8960_be_ops,
1361 },
Kuirong Wangb25838e2012-01-16 23:37:23 -08001362};
1363
1364
1365static struct snd_soc_dai_link msm8960_dai_delta_tabla2x[] = {
1366 /* Backend DAI Links */
1367 {
1368 .name = LPASS_BE_SLIMBUS_0_RX,
1369 .stream_name = "Slimbus Playback",
1370 .cpu_dai_name = "msm-dai-q6.16384",
1371 .platform_name = "msm-pcm-routing",
1372 .codec_name = "tabla_codec",
1373 .codec_dai_name = "tabla_rx1",
1374 .no_pcm = 1,
1375 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1376 .init = &msm8960_audrx_init,
1377 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1378 .ops = &msm8960_be_ops,
1379 },
1380 {
1381 .name = LPASS_BE_SLIMBUS_0_TX,
1382 .stream_name = "Slimbus Capture",
1383 .cpu_dai_name = "msm-dai-q6.16385",
1384 .platform_name = "msm-pcm-routing",
1385 .codec_name = "tabla_codec",
1386 .codec_dai_name = "tabla_tx1",
1387 .no_pcm = 1,
1388 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1389 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1390 .ops = &msm8960_be_ops,
1391 },
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001392 {
1393 .name = "SLIMBUS_2 Hostless",
1394 .stream_name = "SLIMBUS_2 Hostless",
1395 .cpu_dai_name = "msm-dai-q6.16389",
1396 .platform_name = "msm-pcm-hostless",
1397 .codec_name = "tabla_codec",
1398 .codec_dai_name = "tabla_tx2",
1399 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1400 .ops = &msm8960_be_ops,
1401 },
Kuirong Wangb25838e2012-01-16 23:37:23 -08001402};
1403
1404static struct snd_soc_dai_link msm8960_tabla1x_dai[
1405 ARRAY_SIZE(msm8960_dai_common) +
1406 ARRAY_SIZE(msm8960_dai_delta_tabla1x)];
1407
1408
1409static struct snd_soc_dai_link msm8960_dai[
1410 ARRAY_SIZE(msm8960_dai_common) +
1411 ARRAY_SIZE(msm8960_dai_delta_tabla2x)];
1412
1413static struct snd_soc_card snd_soc_tabla1x_card_msm8960 = {
1414 .name = "msm8960-tabla1x-snd-card",
1415 .dai_link = msm8960_tabla1x_dai,
1416 .num_links = ARRAY_SIZE(msm8960_tabla1x_dai),
1417};
1418
1419static struct snd_soc_card snd_soc_card_msm8960 = {
1420 .name = "msm8960-snd-card",
1421 .dai_link = msm8960_dai,
1422 .num_links = ARRAY_SIZE(msm8960_dai),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001423};
1424
1425static struct platform_device *msm8960_snd_device;
Kuirong Wangb25838e2012-01-16 23:37:23 -08001426static struct platform_device *msm8960_snd_tabla1x_device;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001427
1428static int msm8960_configure_headset_mic_gpios(void)
1429{
1430 int ret;
1431 struct pm_gpio param = {
1432 .direction = PM_GPIO_DIR_OUT,
1433 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
1434 .output_value = 1,
1435 .pull = PM_GPIO_PULL_NO,
1436 .vin_sel = PM_GPIO_VIN_S4,
1437 .out_strength = PM_GPIO_STRENGTH_MED,
1438 .function = PM_GPIO_FUNC_NORMAL,
1439 };
1440
1441 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(23), "AV_SWITCH");
1442 if (ret) {
1443 pr_err("%s: Failed to request gpio %d\n", __func__,
1444 PM8921_GPIO_PM_TO_SYS(23));
1445 return ret;
1446 }
1447
1448 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(23), &param);
1449 if (ret)
1450 pr_err("%s: Failed to configure gpio %d\n", __func__,
1451 PM8921_GPIO_PM_TO_SYS(23));
1452 else
1453 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(23), 0);
1454
1455 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(35), "US_EURO_SWITCH");
1456 if (ret) {
1457 pr_err("%s: Failed to request gpio %d\n", __func__,
1458 PM8921_GPIO_PM_TO_SYS(35));
1459 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1460 return ret;
1461 }
1462 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(35), &param);
1463 if (ret)
1464 pr_err("%s: Failed to configure gpio %d\n", __func__,
1465 PM8921_GPIO_PM_TO_SYS(35));
1466 else
Patrick Lai2ae0a002011-09-26 14:51:09 -07001467 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(35), 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001468
1469 return 0;
1470}
1471static void msm8960_free_headset_mic_gpios(void)
1472{
1473 if (msm8960_headset_gpios_configured) {
1474 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1475 gpio_free(PM8921_GPIO_PM_TO_SYS(35));
1476 }
1477}
1478
1479static int __init msm8960_audio_init(void)
1480{
1481 int ret;
1482
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301483 if (!cpu_is_msm8960()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001484 pr_err("%s: Not the right machine type\n", __func__);
1485 return -ENODEV ;
1486 }
Joonwoo Park0976d012011-12-22 11:48:18 -08001487
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001488 mbhc_cfg.calibration = def_tabla_mbhc_cal();
1489 if (!mbhc_cfg.calibration) {
Joonwoo Park0976d012011-12-22 11:48:18 -08001490 pr_err("Calibration data allocation failed\n");
1491 return -ENOMEM;
1492 }
1493
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001494 msm8960_snd_device = platform_device_alloc("soc-audio", 0);
1495 if (!msm8960_snd_device) {
1496 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001497 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001498 return -ENOMEM;
1499 }
1500
Kuirong Wangb25838e2012-01-16 23:37:23 -08001501 memcpy(msm8960_dai, msm8960_dai_common, sizeof(msm8960_dai_common));
1502 memcpy(msm8960_dai + ARRAY_SIZE(msm8960_dai_common),
1503 msm8960_dai_delta_tabla2x, sizeof(msm8960_dai_delta_tabla2x));
1504
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001505 platform_set_drvdata(msm8960_snd_device, &snd_soc_card_msm8960);
1506 ret = platform_device_add(msm8960_snd_device);
1507 if (ret) {
1508 platform_device_put(msm8960_snd_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001509 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001510 return ret;
1511 }
1512
Kuirong Wangb25838e2012-01-16 23:37:23 -08001513 msm8960_snd_tabla1x_device = platform_device_alloc("soc-audio", 1);
1514 if (!msm8960_snd_tabla1x_device) {
1515 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001516 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001517 return -ENOMEM;
1518 }
1519
1520 memcpy(msm8960_tabla1x_dai, msm8960_dai_common,
1521 sizeof(msm8960_dai_common));
1522 memcpy(msm8960_tabla1x_dai + ARRAY_SIZE(msm8960_dai_common),
1523 msm8960_dai_delta_tabla1x, sizeof(msm8960_dai_delta_tabla1x));
1524
1525 platform_set_drvdata(msm8960_snd_tabla1x_device,
1526 &snd_soc_tabla1x_card_msm8960);
1527 ret = platform_device_add(msm8960_snd_tabla1x_device);
1528 if (ret) {
1529 platform_device_put(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001530 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001531 return ret;
1532 }
1533
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001534 if (msm8960_configure_headset_mic_gpios()) {
1535 pr_err("%s Fail to configure headset mic gpios\n", __func__);
1536 msm8960_headset_gpios_configured = 0;
1537 } else
1538 msm8960_headset_gpios_configured = 1;
1539
Joonwoo Park28f49c82012-03-16 12:29:21 -07001540 mutex_init(&cdc_mclk_mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001541 return ret;
1542
1543}
1544module_init(msm8960_audio_init);
1545
1546static void __exit msm8960_audio_exit(void)
1547{
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301548 if (!cpu_is_msm8960()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001549 pr_err("%s: Not the right machine type\n", __func__);
1550 return ;
1551 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001552 msm8960_free_headset_mic_gpios();
1553 platform_device_unregister(msm8960_snd_device);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001554 platform_device_unregister(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001555 kfree(mbhc_cfg.calibration);
Joonwoo Park28f49c82012-03-16 12:29:21 -07001556 mutex_destroy(&cdc_mclk_mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001557}
1558module_exit(msm8960_audio_exit);
1559
1560MODULE_DESCRIPTION("ALSA SoC MSM8960");
1561MODULE_LICENSE("GPL v2");