blob: 98cfa6d506ee8ccb2038ca3e37b0cdc8f0c30f95 [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>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070023#include <sound/pcm.h>
24#include <sound/jack.h>
Bradley Rubin229c6a52011-07-12 16:18:48 -070025#include <asm/mach-types.h>
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -080026#include <mach/socinfo.h>
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070027#include <linux/mfd/wcd9xxx/core.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070028#include "msm-pcm-routing.h"
Bhalchandra Gajare0b9d4d52011-10-21 16:17:47 -070029#include "../codecs/wcd9310.h"
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070030
31/* 8960 machine driver */
32
33#define PM8921_GPIO_BASE NR_GPIO_IRQS
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070034#define PM8921_IRQ_BASE (NR_MSM_IRQS + NR_GPIO_IRQS)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070035#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
Kuirong Wang054e1832012-05-11 15:43:30 -070043#define SAMPLE_RATE_8KHZ 8000
44#define SAMPLE_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
Kiran Kandi97fe19d2012-05-20 22:34:04 -070050#define TOP_SPK_AMP 0x10
Kiran Kandi462acea2011-08-31 23:53:14 -070051
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 Park2cc13f02012-05-09 12:44:25 -070064#define JACK_US_EURO_SEL_GPIO 35
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070065
Harmandeep Singh0dd82412011-11-11 09:46:17 -080066static u32 top_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(18);
67static u32 bottom_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(19);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070068static int msm8960_spk_control;
Kiran Kandi462acea2011-08-31 23:53:14 -070069static int msm8960_ext_bottom_spk_pamp;
70static int msm8960_ext_top_spk_pamp;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070071static int msm8960_slim_0_rx_ch = 1;
72static int msm8960_slim_0_tx_ch = 1;
73
Kuirong Wang054e1832012-05-11 15:43:30 -070074static int msm8960_btsco_rate = SAMPLE_RATE_8KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -070075static int msm8960_btsco_ch = 1;
76
Kuirong Wang054e1832012-05-11 15:43:30 -070077static int msm8960_auxpcm_rate = SAMPLE_RATE_8KHZ;
78
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070079static struct clk *codec_clk;
80static int clk_users;
81
82static int msm8960_headset_gpios_configured;
83
84static struct snd_soc_jack hs_jack;
Bradley Rubincb1e2732011-06-23 16:49:20 -070085static struct snd_soc_jack button_jack;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070086
Joonwoo Park9115ade2012-04-18 13:14:10 -070087static bool hs_detect_use_gpio;
88module_param(hs_detect_use_gpio, bool, 0444);
89MODULE_PARM_DESC(hs_detect_use_gpio, "Use GPIO for headset detection");
90
91static bool hs_detect_use_firmware;
92module_param(hs_detect_use_firmware, bool, 0444);
93MODULE_PARM_DESC(hs_detect_use_firmware, "Use firmware for headset detection");
94
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070095static int msm8960_enable_codec_ext_clk(struct snd_soc_codec *codec, int enable,
96 bool dapm);
Joonwoo Park2cc13f02012-05-09 12:44:25 -070097static bool msm8960_swap_gnd_mic(struct snd_soc_codec *codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070098
99static struct tabla_mbhc_config mbhc_cfg = {
100 .headset_jack = &hs_jack,
101 .button_jack = &button_jack,
102 .read_fw_bin = false,
103 .calibration = NULL,
104 .micbias = TABLA_MICBIAS2,
105 .mclk_cb_fn = msm8960_enable_codec_ext_clk,
106 .mclk_rate = TABLA_EXT_CLK_RATE,
107 .gpio = 0,
108 .gpio_irq = 0,
109 .gpio_level_insert = 1,
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700110 .swap_gnd_mic = NULL,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700111};
Joonwoo Park0976d012011-12-22 11:48:18 -0800112
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700113static u32 us_euro_sel_gpio = PM8921_GPIO_PM_TO_SYS(JACK_US_EURO_SEL_GPIO);
114
Joonwoo Park28f49c82012-03-16 12:29:21 -0700115static struct mutex cdc_mclk_mutex;
116
Kiran Kandi462acea2011-08-31 23:53:14 -0700117static void msm8960_enable_ext_spk_amp_gpio(u32 spk_amp_gpio)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700118{
119 int ret = 0;
120
121 struct pm_gpio param = {
122 .direction = PM_GPIO_DIR_OUT,
123 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
124 .output_value = 1,
125 .pull = PM_GPIO_PULL_NO,
126 .vin_sel = PM_GPIO_VIN_S4,
127 .out_strength = PM_GPIO_STRENGTH_MED,
Kiran Kandi462acea2011-08-31 23:53:14 -0700128 .
129 function = PM_GPIO_FUNC_NORMAL,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700130 };
131
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800132 if (spk_amp_gpio == bottom_spk_pamp_gpio) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700133
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800134 ret = gpio_request(bottom_spk_pamp_gpio, "BOTTOM_SPK_AMP");
Kiran Kandi462acea2011-08-31 23:53:14 -0700135 if (ret) {
136 pr_err("%s: Error requesting BOTTOM SPK AMP GPIO %u\n",
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800137 __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700138 return;
139 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800140 ret = pm8xxx_gpio_config(bottom_spk_pamp_gpio, &param);
Kiran Kandi462acea2011-08-31 23:53:14 -0700141 if (ret)
142 pr_err("%s: Failed to configure Bottom Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800143 " gpio %u\n", __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700144 else {
145 pr_debug("%s: enable Bottom spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800146 gpio_direction_output(bottom_spk_pamp_gpio, 1);
Kiran Kandi462acea2011-08-31 23:53:14 -0700147 }
148
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800149 } else if (spk_amp_gpio == top_spk_pamp_gpio) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700150
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800151 ret = gpio_request(top_spk_pamp_gpio, "TOP_SPK_AMP");
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700152 if (ret) {
153 pr_err("%s: Error requesting GPIO %d\n", __func__,
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800154 top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700155 return;
156 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800157 ret = pm8xxx_gpio_config(top_spk_pamp_gpio, &param);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700158 if (ret)
Kiran Kandi462acea2011-08-31 23:53:14 -0700159 pr_err("%s: Failed to configure Top Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800160 " gpio %u\n", __func__, top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700161 else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700162 pr_debug("%s: enable Top spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800163 gpio_direction_output(top_spk_pamp_gpio, 1);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700164 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700165 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700166 pr_err("%s: ERROR : Invalid External Speaker Ampl GPIO."
167 " gpio = %u\n", __func__, spk_amp_gpio);
168 return;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700169 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700170}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700171
Kiran Kandi462acea2011-08-31 23:53:14 -0700172static void msm8960_ext_spk_power_amp_on(u32 spk)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700173{
Kiran Kandi462acea2011-08-31 23:53:14 -0700174 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
175
176 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
177 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
178
179 pr_debug("%s() External Bottom Speaker Ampl already "
180 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700181 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700182 }
183
184 msm8960_ext_bottom_spk_pamp |= spk;
185
186 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
187 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
188
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800189 msm8960_enable_ext_spk_amp_gpio(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700190 pr_debug("%s: slepping 4 ms after turning on external "
191 " Bottom Speaker Ampl\n", __func__);
192 usleep_range(4000, 4000);
193 }
194
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700195 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG | TOP_SPK_AMP)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700196
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700197 pr_debug("%s: top_spk_amp_state = 0x%x spk_event = 0x%x\n",
198 __func__, 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 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700203
204 pr_debug("%s() External Top Speaker Ampl already"
205 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700206 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700207 }
208
209 msm8960_ext_top_spk_pamp |= spk;
210
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700211 if (((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
212 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) ||
213 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700214
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800215 msm8960_enable_ext_spk_amp_gpio(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700216 pr_debug("%s: sleeping 4 ms after turning on "
217 " external Top Speaker Ampl\n", __func__);
218 usleep_range(4000, 4000);
219 }
220 } else {
221
222 pr_err("%s: ERROR : Invalid External Speaker Ampl. spk = 0x%08x\n",
223 __func__, spk);
224 return;
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700225 }
Kiran Kandi462acea2011-08-31 23:53:14 -0700226}
227
228static void msm8960_ext_spk_power_amp_off(u32 spk)
229{
230 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
231
232 if (!msm8960_ext_bottom_spk_pamp)
233 return;
234
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800235 gpio_direction_output(bottom_spk_pamp_gpio, 0);
236 gpio_free(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700237 msm8960_ext_bottom_spk_pamp = 0;
238
239 pr_debug("%s: sleeping 4 ms after turning off external Bottom"
240 " Speaker Ampl\n", __func__);
241
242 usleep_range(4000, 4000);
243
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700244 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG | TOP_SPK_AMP)) {
245
246 pr_debug("%s: top_spk_amp_state = 0x%x spk_event = 0x%x\n",
247 __func__, msm8960_ext_top_spk_pamp, spk);
Kiran Kandi462acea2011-08-31 23:53:14 -0700248
249 if (!msm8960_ext_top_spk_pamp)
250 return;
251
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700252 if ((spk & TOP_SPK_AMP_POS) || (spk & TOP_SPK_AMP_NEG)) {
253
254 msm8960_ext_top_spk_pamp &= (~(TOP_SPK_AMP_POS |
255 TOP_SPK_AMP_NEG));
256 } else if (spk & TOP_SPK_AMP) {
257 msm8960_ext_top_spk_pamp &= ~TOP_SPK_AMP;
258 }
259
260 if (msm8960_ext_top_spk_pamp)
261 return;
262
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800263 gpio_direction_output(top_spk_pamp_gpio, 0);
264 gpio_free(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700265 msm8960_ext_top_spk_pamp = 0;
266
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700267 pr_debug("%s: sleeping 4 ms after ext Top Spek Ampl is off\n",
268 __func__);
Kiran Kandi462acea2011-08-31 23:53:14 -0700269
270 usleep_range(4000, 4000);
271 } else {
272
273 pr_err("%s: ERROR : Invalid Ext Spk Ampl. spk = 0x%08x\n",
274 __func__, spk);
275 return;
276 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700277}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700278
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700279static void msm8960_ext_control(struct snd_soc_codec *codec)
280{
281 struct snd_soc_dapm_context *dapm = &codec->dapm;
282
Jayasena Sangaraboina247d36e2012-04-17 10:16:18 -0700283 mutex_lock(&dapm->codec->mutex);
284
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700285 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700286 if (msm8960_spk_control == MSM8960_SPK_ON) {
287 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
288 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
289 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
290 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
291 } else {
292 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Pos");
293 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Neg");
294 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Pos");
295 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Neg");
296 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700297
298 snd_soc_dapm_sync(dapm);
Jayasena Sangaraboina247d36e2012-04-17 10:16:18 -0700299 mutex_unlock(&dapm->codec->mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700300}
301
302static int msm8960_get_spk(struct snd_kcontrol *kcontrol,
303 struct snd_ctl_elem_value *ucontrol)
304{
305 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
306 ucontrol->value.integer.value[0] = msm8960_spk_control;
307 return 0;
308}
309static int msm8960_set_spk(struct snd_kcontrol *kcontrol,
310 struct snd_ctl_elem_value *ucontrol)
311{
312 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
313
314 pr_debug("%s()\n", __func__);
315 if (msm8960_spk_control == ucontrol->value.integer.value[0])
316 return 0;
317
318 msm8960_spk_control = ucontrol->value.integer.value[0];
319 msm8960_ext_control(codec);
320 return 1;
321}
322static int msm8960_spkramp_event(struct snd_soc_dapm_widget *w,
323 struct snd_kcontrol *k, int event)
324{
325 pr_debug("%s() %x\n", __func__, SND_SOC_DAPM_EVENT_ON(event));
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700326
327 if (SND_SOC_DAPM_EVENT_ON(event)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700328 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
329 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_POS);
330 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
331 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_NEG);
332 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
333 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_POS);
334 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
335 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_NEG);
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700336 else if (!strncmp(w->name, "Ext Spk Top", 12))
337 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP);
Kiran Kandi462acea2011-08-31 23:53:14 -0700338 else {
339 pr_err("%s() Invalid Speaker Widget = %s\n",
340 __func__, w->name);
341 return -EINVAL;
342 }
343
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700344 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700345 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
346 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_POS);
347 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
348 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_NEG);
349 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
350 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_POS);
351 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
352 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_NEG);
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700353 else if (!strncmp(w->name, "Ext Spk Top", 12))
354 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP);
Kiran Kandi462acea2011-08-31 23:53:14 -0700355 else {
356 pr_err("%s() Invalid Speaker Widget = %s\n",
357 __func__, w->name);
358 return -EINVAL;
359 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700360 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700361 return 0;
362}
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700363
364static int msm8960_enable_codec_ext_clk(struct snd_soc_codec *codec, int enable,
365 bool dapm)
Joonwoo Park0976d012011-12-22 11:48:18 -0800366{
Joonwoo Park28f49c82012-03-16 12:29:21 -0700367 int r = 0;
Joonwoo Park0976d012011-12-22 11:48:18 -0800368 pr_debug("%s: enable = %d\n", __func__, enable);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700369
370 mutex_lock(&cdc_mclk_mutex);
Joonwoo Park0976d012011-12-22 11:48:18 -0800371 if (enable) {
372 clk_users++;
373 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700374 if (clk_users == 1) {
375 if (codec_clk) {
376 clk_set_rate(codec_clk, TABLA_EXT_CLK_RATE);
Asish Bhattacharya4776d962012-04-25 12:08:09 +0530377 clk_prepare_enable(codec_clk);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700378 tabla_mclk_enable(codec, 1, dapm);
379 } else {
380 pr_err("%s: Error setting Tabla MCLK\n",
381 __func__);
382 clk_users--;
383 r = -EINVAL;
384 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800385 }
386 } else {
Joonwoo Park28f49c82012-03-16 12:29:21 -0700387 if (clk_users > 0) {
388 clk_users--;
389 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
390 if (clk_users == 0) {
391 pr_debug("%s: disabling MCLK. clk_users = %d\n",
Joonwoo Park0976d012011-12-22 11:48:18 -0800392 __func__, clk_users);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700393 tabla_mclk_enable(codec, 0, dapm);
Asish Bhattacharya4776d962012-04-25 12:08:09 +0530394 clk_disable_unprepare(codec_clk);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700395 }
396 } else {
397 pr_err("%s: Error releasing Tabla MCLK\n", __func__);
398 r = -EINVAL;
Joonwoo Park0976d012011-12-22 11:48:18 -0800399 }
400 }
Joonwoo Park28f49c82012-03-16 12:29:21 -0700401 mutex_unlock(&cdc_mclk_mutex);
402 return r;
Joonwoo Park0976d012011-12-22 11:48:18 -0800403}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700404
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700405static bool msm8960_swap_gnd_mic(struct snd_soc_codec *codec)
406{
407 int value = gpio_get_value_cansleep(us_euro_sel_gpio);
408 pr_debug("%s: US EURO select switch %d to %d\n", __func__, value,
409 !value);
410 gpio_set_value_cansleep(us_euro_sel_gpio, !value);
411 return true;
412}
413
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700414static int msm8960_mclk_event(struct snd_soc_dapm_widget *w,
415 struct snd_kcontrol *kcontrol, int event)
416{
417 pr_debug("%s: event = %d\n", __func__, event);
418
419 switch (event) {
420 case SND_SOC_DAPM_PRE_PMU:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700421 return msm8960_enable_codec_ext_clk(w->codec, 1, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700422 case SND_SOC_DAPM_POST_PMD:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700423 return msm8960_enable_codec_ext_clk(w->codec, 0, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700424 }
425 return 0;
426}
427
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700428static const struct snd_soc_dapm_widget msm8960_dapm_widgets[] = {
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700429
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700430 SND_SOC_DAPM_SUPPLY("MCLK", SND_SOC_NOPM, 0, 0,
431 msm8960_mclk_event, SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
432
Kiran Kandi462acea2011-08-31 23:53:14 -0700433 SND_SOC_DAPM_SPK("Ext Spk Bottom Pos", msm8960_spkramp_event),
434 SND_SOC_DAPM_SPK("Ext Spk Bottom Neg", msm8960_spkramp_event),
435
436 SND_SOC_DAPM_SPK("Ext Spk Top Pos", msm8960_spkramp_event),
437 SND_SOC_DAPM_SPK("Ext Spk Top Neg", msm8960_spkramp_event),
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700438 SND_SOC_DAPM_SPK("Ext Spk Top", msm8960_spkramp_event),
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700439
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700440 SND_SOC_DAPM_MIC("Handset Mic", NULL),
441 SND_SOC_DAPM_MIC("Headset Mic", NULL),
442 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700443 SND_SOC_DAPM_MIC("ANCRight Headset Mic", NULL),
444 SND_SOC_DAPM_MIC("ANCLeft Headset Mic", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700445
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700446 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700447 SND_SOC_DAPM_MIC("Digital Mic2", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700448 SND_SOC_DAPM_MIC("Digital Mic3", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700449 SND_SOC_DAPM_MIC("Digital Mic4", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700450 SND_SOC_DAPM_MIC("Digital Mic5", NULL),
451 SND_SOC_DAPM_MIC("Digital Mic6", NULL),
452
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700453};
454
Bradley Rubin229c6a52011-07-12 16:18:48 -0700455static const struct snd_soc_dapm_route common_audio_map[] = {
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700456
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700457 {"RX_BIAS", NULL, "MCLK"},
458 {"LDO_H", NULL, "MCLK"},
459
460 /* Speaker path */
Kiran Kandi462acea2011-08-31 23:53:14 -0700461 {"Ext Spk Bottom Pos", NULL, "LINEOUT1"},
462 {"Ext Spk Bottom Neg", NULL, "LINEOUT3"},
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700463
Kiran Kandi462acea2011-08-31 23:53:14 -0700464 {"Ext Spk Top Pos", NULL, "LINEOUT2"},
465 {"Ext Spk Top Neg", NULL, "LINEOUT4"},
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700466 {"Ext Spk Top", NULL, "LINEOUT5"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700467
468 /* Microphone path */
Bradley Rubin229c6a52011-07-12 16:18:48 -0700469 {"AMIC1", NULL, "MIC BIAS1 Internal1"},
470 {"MIC BIAS1 Internal1", NULL, "Handset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700471
472 {"AMIC2", NULL, "MIC BIAS2 External"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700473 {"MIC BIAS2 External", NULL, "Headset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700474
Patrick Laie519dd62011-09-28 12:37:11 -0700475 /**
476 * AMIC3 and AMIC4 inputs are connected to ANC microphones
477 * These mics are biased differently on CDP and FLUID
478 * routing entries below are based on bias arrangement
479 * on FLUID.
480 */
481 {"AMIC3", NULL, "MIC BIAS3 Internal1"},
482 {"MIC BIAS3 Internal1", NULL, "ANCRight Headset Mic"},
483
484 {"AMIC4", NULL, "MIC BIAS1 Internal2"},
485 {"MIC BIAS1 Internal2", NULL, "ANCLeft Headset Mic"},
486
Kiran Kandie9bf86a2011-07-21 16:50:41 -0700487 {"HEADPHONE", NULL, "LDO_H"},
488
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700489 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700490 * The digital Mic routes are setup considering
491 * fluid as default device.
492 */
493
494 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700495 * Digital Mic1. Front Bottom left Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700496 * Digital Mic GM5 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700497 * Conncted to DMIC2 Input on Tabla codec.
498 */
499 {"DMIC2", NULL, "MIC BIAS1 External"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700500 {"MIC BIAS1 External", NULL, "Digital Mic1"},
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700501
502 /**
503 * Digital Mic2. Front Bottom right Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700504 * Digital Mic GM6 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700505 * Conncted to DMIC1 Input on Tabla codec.
506 */
507 {"DMIC1", NULL, "MIC BIAS1 External"},
508 {"MIC BIAS1 External", NULL, "Digital Mic2"},
509
510 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700511 * Digital Mic3. Back Bottom Digital Mic on Fluid.
512 * Digital Mic GM1 on CDP mainboard.
513 * Conncted to DMIC4 Input on Tabla codec.
514 */
515 {"DMIC4", NULL, "MIC BIAS3 External"},
516 {"MIC BIAS3 External", NULL, "Digital Mic3"},
517
518 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700519 * Digital Mic4. Back top Digital Mic on Fluid.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700520 * Digital Mic GM2 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700521 * Conncted to DMIC3 Input on Tabla codec.
522 */
523 {"DMIC3", NULL, "MIC BIAS3 External"},
524 {"MIC BIAS3 External", NULL, "Digital Mic4"},
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700525
526 /**
527 * Digital Mic5. Front top Digital Mic on Fluid.
528 * Digital Mic GM3 on CDP mainboard.
529 * Conncted to DMIC5 Input on Tabla codec.
530 */
531 {"DMIC5", NULL, "MIC BIAS4 External"},
532 {"MIC BIAS4 External", NULL, "Digital Mic5"},
533
Patrick Laicb7802b2011-10-04 12:39:18 -0700534 /* Tabla digital Mic6 - back bottom digital Mic on Liquid and
535 * bottom mic on CDP. FLUID/MTP do not have dmic6 installed.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700536 */
537 {"DMIC6", NULL, "MIC BIAS4 External"},
538 {"MIC BIAS4 External", NULL, "Digital Mic6"},
Bradley Rubin229c6a52011-07-12 16:18:48 -0700539};
540
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700541static const char *spk_function[] = {"Off", "On"};
Patrick Lai9f4b4292011-07-16 22:11:09 -0700542static const char *slim0_rx_ch_text[] = {"One", "Two"};
543static const char *slim0_tx_ch_text[] = {"One", "Two", "Three", "Four"};
544
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700545static const struct soc_enum msm8960_enum[] = {
546 SOC_ENUM_SINGLE_EXT(2, spk_function),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700547 SOC_ENUM_SINGLE_EXT(2, slim0_rx_ch_text),
548 SOC_ENUM_SINGLE_EXT(4, slim0_tx_ch_text),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700549};
550
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700551static const char *btsco_rate_text[] = {"8000", "16000"};
552static const struct soc_enum msm8960_btsco_enum[] = {
553 SOC_ENUM_SINGLE_EXT(2, btsco_rate_text),
554};
555
Kuirong Wang054e1832012-05-11 15:43:30 -0700556static const char *auxpcm_rate_text[] = {"rate_8000", "rate_16000"};
557static const struct soc_enum msm8960_auxpcm_enum[] = {
558 SOC_ENUM_SINGLE_EXT(2, auxpcm_rate_text),
559};
560
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700561static int msm8960_slim_0_rx_ch_get(struct snd_kcontrol *kcontrol,
562 struct snd_ctl_elem_value *ucontrol)
563{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700564 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800565 msm8960_slim_0_rx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700566 ucontrol->value.integer.value[0] = msm8960_slim_0_rx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700567 return 0;
568}
569
570static int msm8960_slim_0_rx_ch_put(struct snd_kcontrol *kcontrol,
571 struct snd_ctl_elem_value *ucontrol)
572{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700573 msm8960_slim_0_rx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700574
Patrick Lai9f4b4292011-07-16 22:11:09 -0700575 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800576 msm8960_slim_0_rx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700577 return 1;
578}
579
580static int msm8960_slim_0_tx_ch_get(struct snd_kcontrol *kcontrol,
581 struct snd_ctl_elem_value *ucontrol)
582{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700583 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800584 msm8960_slim_0_tx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700585 ucontrol->value.integer.value[0] = msm8960_slim_0_tx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700586 return 0;
587}
588
589static int msm8960_slim_0_tx_ch_put(struct snd_kcontrol *kcontrol,
590 struct snd_ctl_elem_value *ucontrol)
591{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700592 msm8960_slim_0_tx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700593
594 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800595 msm8960_slim_0_tx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700596 return 1;
597}
598
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700599static int msm8960_btsco_rate_get(struct snd_kcontrol *kcontrol,
600 struct snd_ctl_elem_value *ucontrol)
601{
Joonwoo Park0976d012011-12-22 11:48:18 -0800602 pr_debug("%s: msm8960_btsco_rate = %d", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700603 ucontrol->value.integer.value[0] = msm8960_btsco_rate;
604 return 0;
605}
606
607static int msm8960_btsco_rate_put(struct snd_kcontrol *kcontrol,
608 struct snd_ctl_elem_value *ucontrol)
609{
610 switch (ucontrol->value.integer.value[0]) {
611 case 0:
Kuirong Wang054e1832012-05-11 15:43:30 -0700612 msm8960_btsco_rate = SAMPLE_RATE_8KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700613 break;
614 case 1:
Kuirong Wang054e1832012-05-11 15:43:30 -0700615 msm8960_btsco_rate = SAMPLE_RATE_16KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700616 break;
617 default:
Kuirong Wang054e1832012-05-11 15:43:30 -0700618 msm8960_btsco_rate = SAMPLE_RATE_8KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700619 break;
620 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800621 pr_debug("%s: msm8960_btsco_rate = %d\n", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700622 return 0;
623}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700624
Kuirong Wang054e1832012-05-11 15:43:30 -0700625static int msm8960_auxpcm_rate_get(struct snd_kcontrol *kcontrol,
626 struct snd_ctl_elem_value *ucontrol)
627{
628 pr_debug("%s: msm8960_auxpcm_rate = %d", __func__,
629 msm8960_auxpcm_rate);
630 ucontrol->value.integer.value[0] = msm8960_auxpcm_rate;
631 return 0;
632}
633
634static int msm8960_auxpcm_rate_put(struct snd_kcontrol *kcontrol,
635 struct snd_ctl_elem_value *ucontrol)
636{
637 switch (ucontrol->value.integer.value[0]) {
638 case 0:
639 msm8960_auxpcm_rate = SAMPLE_RATE_8KHZ;
640 break;
641 case 1:
642 msm8960_auxpcm_rate = SAMPLE_RATE_16KHZ;
643 break;
644 default:
645 msm8960_auxpcm_rate = SAMPLE_RATE_8KHZ;
646 break;
647 }
648 pr_debug("%s: msm8960_auxpcm_rate = %d"
649 "ucontrol->value.integer.value[0] = %d\n", __func__,
650 msm8960_auxpcm_rate,
651 (int)ucontrol->value.integer.value[0]);
652 return 0;
653}
654
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700655static const struct snd_kcontrol_new tabla_msm8960_controls[] = {
656 SOC_ENUM_EXT("Speaker Function", msm8960_enum[0], msm8960_get_spk,
657 msm8960_set_spk),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700658 SOC_ENUM_EXT("SLIM_0_RX Channels", msm8960_enum[1],
659 msm8960_slim_0_rx_ch_get, msm8960_slim_0_rx_ch_put),
660 SOC_ENUM_EXT("SLIM_0_TX Channels", msm8960_enum[2],
661 msm8960_slim_0_tx_ch_get, msm8960_slim_0_tx_ch_put),
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700662 SOC_ENUM_EXT("Internal BTSCO SampleRate", msm8960_btsco_enum[0],
663 msm8960_btsco_rate_get, msm8960_btsco_rate_put),
Kuirong Wang054e1832012-05-11 15:43:30 -0700664 SOC_ENUM_EXT("AUX PCM SampleRate", msm8960_auxpcm_enum[0],
665 msm8960_auxpcm_rate_get, msm8960_auxpcm_rate_put),
666};
667
Joonwoo Park0976d012011-12-22 11:48:18 -0800668static void *def_tabla_mbhc_cal(void)
669{
670 void *tabla_cal;
671 struct tabla_mbhc_btn_detect_cfg *btn_cfg;
672 u16 *btn_low, *btn_high;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800673 u8 *n_ready, *n_cic, *gain;
Joonwoo Park0976d012011-12-22 11:48:18 -0800674
675 tabla_cal = kzalloc(TABLA_MBHC_CAL_SIZE(TABLA_MBHC_DEF_BUTTONS,
676 TABLA_MBHC_DEF_RLOADS),
677 GFP_KERNEL);
678 if (!tabla_cal) {
679 pr_err("%s: out of memory\n", __func__);
680 return NULL;
681 }
682
683#define S(X, Y) ((TABLA_MBHC_CAL_GENERAL_PTR(tabla_cal)->X) = (Y))
684 S(t_ldoh, 100);
685 S(t_bg_fast_settle, 100);
686 S(t_shutdown_plug_rem, 255);
687 S(mbhc_nsa, 4);
688 S(mbhc_navg, 4);
689#undef S
690#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_DET_PTR(tabla_cal)->X) = (Y))
691 S(mic_current, TABLA_PID_MIC_5_UA);
692 S(hph_current, TABLA_PID_MIC_5_UA);
693 S(t_mic_pid, 100);
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800694 S(t_ins_complete, 250);
Joonwoo Park0976d012011-12-22 11:48:18 -0800695 S(t_ins_retry, 200);
696#undef S
697#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla_cal)->X) = (Y))
698 S(v_no_mic, 30);
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700699 S(v_hs_max, 2400);
Joonwoo Park0976d012011-12-22 11:48:18 -0800700#undef S
701#define S(X, Y) ((TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal)->X) = (Y))
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800702 S(c[0], 62);
703 S(c[1], 124);
704 S(nc, 1);
705 S(n_meas, 3);
Joonwoo Park0976d012011-12-22 11:48:18 -0800706 S(mbhc_nsc, 11);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800707 S(n_btn_meas, 1);
708 S(n_btn_con, 2);
Joonwoo Park0976d012011-12-22 11:48:18 -0800709 S(num_btn, TABLA_MBHC_DEF_BUTTONS);
710 S(v_btn_press_delta_sta, 100);
711 S(v_btn_press_delta_cic, 50);
712#undef S
713 btn_cfg = TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal);
714 btn_low = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_LOW);
715 btn_high = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_HIGH);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800716 btn_low[0] = -50;
Joonwoo Parkfee17432012-04-16 16:33:55 -0700717 btn_high[0] = 20;
718 btn_low[1] = 21;
719 btn_high[1] = 62;
720 btn_low[2] = 63;
721 btn_high[2] = 104;
722 btn_low[3] = 105;
723 btn_high[3] = 143;
724 btn_low[4] = 144;
725 btn_high[4] = 181;
726 btn_low[5] = 182;
727 btn_high[5] = 218;
728 btn_low[6] = 219;
729 btn_high[6] = 254;
730 btn_low[7] = 255;
731 btn_high[7] = 330;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800732 n_ready = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_READY);
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700733 n_ready[0] = 80;
734 n_ready[1] = 68;
Joonwoo Park0976d012011-12-22 11:48:18 -0800735 n_cic = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_CIC);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800736 n_cic[0] = 60;
737 n_cic[1] = 47;
Joonwoo Park0976d012011-12-22 11:48:18 -0800738 gain = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_GAIN);
739 gain[0] = 11;
740 gain[1] = 9;
741
742 return tabla_cal;
743}
744
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800745static int msm8960_hw_params(struct snd_pcm_substream *substream,
746 struct snd_pcm_hw_params *params)
747{
748 struct snd_soc_pcm_runtime *rtd = substream->private_data;
749 struct snd_soc_dai *codec_dai = rtd->codec_dai;
750 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
751 int ret = 0;
752 unsigned int rx_ch[SLIM_MAX_RX_PORTS], tx_ch[SLIM_MAX_TX_PORTS];
753 unsigned int rx_ch_cnt = 0, tx_ch_cnt = 0;
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700754
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800755 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700756
757 pr_debug("%s: rx_0_ch=%d\n", __func__, msm8960_slim_0_rx_ch);
758
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800759 ret = snd_soc_dai_get_channel_map(codec_dai,
760 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
761 if (ret < 0) {
762 pr_err("%s: failed to get codec chan map\n", __func__);
763 goto end;
764 }
765
766 ret = snd_soc_dai_set_channel_map(cpu_dai, 0, 0,
767 msm8960_slim_0_rx_ch, rx_ch);
768 if (ret < 0) {
769 pr_err("%s: failed to set cpu chan map\n", __func__);
770 goto end;
771 }
772 ret = snd_soc_dai_set_channel_map(codec_dai, 0, 0,
773 msm8960_slim_0_rx_ch, rx_ch);
774 if (ret < 0) {
775 pr_err("%s: failed to set codec channel map\n",
776 __func__);
777 goto end;
778 }
779 } else {
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700780
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700781 pr_debug("%s: %s tx_dai_id = %d num_ch = %d\n", __func__,
782 codec_dai->name, codec_dai->id, msm8960_slim_0_tx_ch);
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700783
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800784 ret = snd_soc_dai_get_channel_map(codec_dai,
785 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
786 if (ret < 0) {
787 pr_err("%s: failed to get codec chan map\n", __func__);
788 goto end;
789 }
790 ret = snd_soc_dai_set_channel_map(cpu_dai,
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700791 msm8960_slim_0_tx_ch, tx_ch, 0 , 0);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800792 if (ret < 0) {
793 pr_err("%s: failed to set cpu chan map\n", __func__);
794 goto end;
795 }
796 ret = snd_soc_dai_set_channel_map(codec_dai,
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700797 msm8960_slim_0_tx_ch, tx_ch, 0, 0);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800798 if (ret < 0) {
799 pr_err("%s: failed to set codec channel map\n",
800 __func__);
801 goto end;
802 }
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700803 }
804end:
805 return ret;
806}
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800807
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700808static int msm8960_slimbus_2_hw_params(struct snd_pcm_substream *substream,
809 struct snd_pcm_hw_params *params)
810{
811 struct snd_soc_pcm_runtime *rtd = substream->private_data;
812 struct snd_soc_dai *codec_dai = rtd->codec_dai;
813 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
814 int ret = 0;
815 unsigned int rx_ch[SLIM_MAX_RX_PORTS], tx_ch[SLIM_MAX_TX_PORTS];
816 unsigned int rx_ch_cnt = 0, tx_ch_cnt = 0;
817 unsigned int num_tx_ch = 0;
818 unsigned int num_rx_ch = 0;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800819
Kiran Kandi97fe19d2012-05-20 22:34:04 -0700820 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
821
822 num_rx_ch = params_channels(params);
823
824 pr_debug("%s: %s rx_dai_id = %d num_ch = %d\n", __func__,
825 codec_dai->name, codec_dai->id, num_rx_ch);
826
827 ret = snd_soc_dai_get_channel_map(codec_dai,
828 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
829 if (ret < 0) {
830 pr_err("%s: failed to get codec chan map\n", __func__);
831 goto end;
832 }
833
834 ret = snd_soc_dai_set_channel_map(cpu_dai, 0, 0,
835 num_rx_ch, rx_ch);
836 if (ret < 0) {
837 pr_err("%s: failed to set cpu chan map\n", __func__);
838 goto end;
839 }
840 ret = snd_soc_dai_set_channel_map(codec_dai, 0, 0,
841 num_rx_ch, rx_ch);
842 if (ret < 0) {
843 pr_err("%s: failed to set codec channel map\n",
844 __func__);
845 goto end;
846 }
847 } else {
848 num_tx_ch = params_channels(params);
849
850 pr_debug("%s: %s tx_dai_id = %d num_ch = %d\n", __func__,
851 codec_dai->name, codec_dai->id, num_tx_ch);
852
853 ret = snd_soc_dai_get_channel_map(codec_dai,
854 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
855 if (ret < 0) {
856 pr_err("%s: failed to get codec chan map\n", __func__);
857 goto end;
858 }
859
860 ret = snd_soc_dai_set_channel_map(cpu_dai,
861 num_tx_ch, tx_ch, 0 , 0);
862 if (ret < 0) {
863 pr_err("%s: failed to set cpu chan map\n", __func__);
864 goto end;
865 }
866 ret = snd_soc_dai_set_channel_map(codec_dai,
867 num_tx_ch, tx_ch, 0, 0);
868 if (ret < 0) {
869 pr_err("%s: failed to set codec channel map\n",
870 __func__);
871 goto end;
872 }
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800873 }
874end:
875 return ret;
876}
877
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700878static int msm8960_audrx_init(struct snd_soc_pcm_runtime *rtd)
879{
880 int err;
881 struct snd_soc_codec *codec = rtd->codec;
882 struct snd_soc_dapm_context *dapm = &codec->dapm;
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800883 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700884 struct pm_gpio jack_gpio_cfg = {
885 .direction = PM_GPIO_DIR_IN,
886 .pull = PM_GPIO_PULL_UP_1P5,
887 .function = PM_GPIO_FUNC_NORMAL,
888 .vin_sel = 2,
889 .inv_int_pol = 0,
890 };
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700891
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800892 pr_debug("%s(), dev_name%s\n", __func__, dev_name(cpu_dai->dev));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700893
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800894 if (machine_is_msm8960_liquid()) {
895 top_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(19));
896 bottom_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(18));
897 }
898
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700899 snd_soc_dapm_new_controls(dapm, msm8960_dapm_widgets,
900 ARRAY_SIZE(msm8960_dapm_widgets));
901
Bradley Rubin229c6a52011-07-12 16:18:48 -0700902 snd_soc_dapm_add_routes(dapm, common_audio_map,
903 ARRAY_SIZE(common_audio_map));
904
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700905 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
906 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
907 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
908 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700909
910 snd_soc_dapm_sync(dapm);
911
912 err = snd_soc_jack_new(codec, "Headset Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800913 (SND_JACK_HEADSET | SND_JACK_OC_HPHL |
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700914 SND_JACK_OC_HPHR | SND_JACK_UNSUPPORTED),
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800915 &hs_jack);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700916 if (err) {
917 pr_err("failed to create new jack\n");
918 return err;
919 }
Bradley Rubincb1e2732011-06-23 16:49:20 -0700920
921 err = snd_soc_jack_new(codec, "Button Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800922 TABLA_JACK_BUTTON_MASK, &button_jack);
Bradley Rubincb1e2732011-06-23 16:49:20 -0700923 if (err) {
924 pr_err("failed to create new jack\n");
925 return err;
926 }
927
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800928 codec_clk = clk_get(cpu_dai->dev, "osr_clk");
929
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700930 if (machine_is_msm8960_cdp())
931 mbhc_cfg.swap_gnd_mic = msm8960_swap_gnd_mic;
932
Joonwoo Park9115ade2012-04-18 13:14:10 -0700933 if (hs_detect_use_gpio) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700934 mbhc_cfg.gpio = PM8921_GPIO_PM_TO_SYS(JACK_DETECT_GPIO);
935 mbhc_cfg.gpio_irq = JACK_DETECT_INT;
936 }
937
938 if (mbhc_cfg.gpio) {
939 err = pm8xxx_gpio_config(mbhc_cfg.gpio, &jack_gpio_cfg);
940 if (err) {
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700941 pr_err("%s: pm8xxx_gpio_config JACK_DETECT failed %d\n",
942 __func__, err);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700943 return err;
944 }
945 }
Joonwoo Park9115ade2012-04-18 13:14:10 -0700946
947 mbhc_cfg.read_fw_bin = hs_detect_use_firmware;
948
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700949 err = tabla_hs_detect(codec, &mbhc_cfg);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700950
Joonwoo Park03324832012-03-19 19:36:16 -0700951 return err;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700952}
953
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700954static int msm8960_slim_0_rx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
955 struct snd_pcm_hw_params *params)
956{
957 struct snd_interval *rate = hw_param_interval(params,
958 SNDRV_PCM_HW_PARAM_RATE);
959
960 struct snd_interval *channels = hw_param_interval(params,
961 SNDRV_PCM_HW_PARAM_CHANNELS);
962
963 pr_debug("%s()\n", __func__);
964 rate->min = rate->max = 48000;
965 channels->min = channels->max = msm8960_slim_0_rx_ch;
966
967 return 0;
968}
969
970static int msm8960_slim_0_tx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
971 struct snd_pcm_hw_params *params)
972{
973 struct snd_interval *rate = hw_param_interval(params,
974 SNDRV_PCM_HW_PARAM_RATE);
975
976 struct snd_interval *channels = hw_param_interval(params,
977 SNDRV_PCM_HW_PARAM_CHANNELS);
978
979 pr_debug("%s()\n", __func__);
980 rate->min = rate->max = 48000;
981 channels->min = channels->max = msm8960_slim_0_tx_ch;
982
983 return 0;
984}
985
986static int msm8960_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
987 struct snd_pcm_hw_params *params)
988{
989 struct snd_interval *rate = hw_param_interval(params,
990 SNDRV_PCM_HW_PARAM_RATE);
991
992 pr_debug("%s()\n", __func__);
993 rate->min = rate->max = 48000;
994
995 return 0;
996}
997
Helen Zeng73f7fc62011-11-18 16:29:59 -0800998static int msm8960_hdmi_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
999 struct snd_pcm_hw_params *params)
1000{
1001 struct snd_interval *rate = hw_param_interval(params,
1002 SNDRV_PCM_HW_PARAM_RATE);
1003
1004 struct snd_interval *channels = hw_param_interval(params,
1005 SNDRV_PCM_HW_PARAM_CHANNELS);
1006
Kiran Kandi5e809b02012-01-31 00:24:33 -08001007 pr_debug("%s channels->min %u channels->max %u ()\n", __func__,
1008 channels->min, channels->max);
1009
Helen Zeng73f7fc62011-11-18 16:29:59 -08001010 rate->min = rate->max = 48000;
Helen Zeng73f7fc62011-11-18 16:29:59 -08001011
1012 return 0;
1013}
1014
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001015static int msm8960_btsco_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
1016 struct snd_pcm_hw_params *params)
1017{
1018 struct snd_interval *rate = hw_param_interval(params,
1019 SNDRV_PCM_HW_PARAM_RATE);
1020
1021 struct snd_interval *channels = hw_param_interval(params,
1022 SNDRV_PCM_HW_PARAM_CHANNELS);
1023
1024 rate->min = rate->max = msm8960_btsco_rate;
1025 channels->min = channels->max = msm8960_btsco_ch;
1026
1027 return 0;
1028}
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001029static int msm8960_auxpcm_be_params_fixup(struct snd_soc_pcm_runtime *rtd,
1030 struct snd_pcm_hw_params *params)
1031{
1032 struct snd_interval *rate = hw_param_interval(params,
1033 SNDRV_PCM_HW_PARAM_RATE);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001034
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001035 struct snd_interval *channels = hw_param_interval(params,
1036 SNDRV_PCM_HW_PARAM_CHANNELS);
1037
Kuirong Wang054e1832012-05-11 15:43:30 -07001038 rate->min = rate->max = msm8960_auxpcm_rate;
1039 /* PCM only supports mono output */
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001040 channels->min = channels->max = 1;
1041
1042 return 0;
1043}
1044static int msm8960_aux_pcm_get_gpios(void)
1045{
1046 int ret = 0;
1047
1048 pr_debug("%s\n", __func__);
1049
1050 ret = gpio_request(GPIO_AUX_PCM_DOUT, "AUX PCM DOUT");
1051 if (ret < 0) {
1052 pr_err("%s: Failed to request gpio(%d): AUX PCM DOUT",
1053 __func__, GPIO_AUX_PCM_DOUT);
1054 goto fail_dout;
1055 }
1056
1057 ret = gpio_request(GPIO_AUX_PCM_DIN, "AUX PCM DIN");
1058 if (ret < 0) {
1059 pr_err("%s: Failed to request gpio(%d): AUX PCM DIN",
1060 __func__, GPIO_AUX_PCM_DIN);
1061 goto fail_din;
1062 }
1063
1064 ret = gpio_request(GPIO_AUX_PCM_SYNC, "AUX PCM SYNC");
1065 if (ret < 0) {
1066 pr_err("%s: Failed to request gpio(%d): AUX PCM SYNC",
1067 __func__, GPIO_AUX_PCM_SYNC);
1068 goto fail_sync;
1069 }
1070 ret = gpio_request(GPIO_AUX_PCM_CLK, "AUX PCM CLK");
1071 if (ret < 0) {
1072 pr_err("%s: Failed to request gpio(%d): AUX PCM CLK",
1073 __func__, GPIO_AUX_PCM_CLK);
1074 goto fail_clk;
1075 }
1076
1077 return 0;
1078
1079fail_clk:
1080 gpio_free(GPIO_AUX_PCM_SYNC);
1081fail_sync:
1082 gpio_free(GPIO_AUX_PCM_DIN);
1083fail_din:
1084 gpio_free(GPIO_AUX_PCM_DOUT);
1085fail_dout:
1086
1087 return ret;
1088}
1089
1090static int msm8960_aux_pcm_free_gpios(void)
1091{
1092 gpio_free(GPIO_AUX_PCM_DIN);
1093 gpio_free(GPIO_AUX_PCM_DOUT);
1094 gpio_free(GPIO_AUX_PCM_SYNC);
1095 gpio_free(GPIO_AUX_PCM_CLK);
1096
1097 return 0;
1098}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001099static int msm8960_startup(struct snd_pcm_substream *substream)
1100{
Kiran Kandi97fe19d2012-05-20 22:34:04 -07001101 struct snd_soc_pcm_runtime *rtd = substream->private_data;
1102
1103 pr_debug("%s(): dai_link_str_name = %s cpu_dai = %s codec_dai = %s\n",
1104 __func__, rtd->dai_link->stream_name,
1105 rtd->dai_link->cpu_dai_name, rtd->dai_link->codec_dai_name);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001106 return 0;
1107}
1108
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001109static int msm8960_auxpcm_startup(struct snd_pcm_substream *substream)
1110{
1111 int ret = 0;
1112
1113 pr_debug("%s(): substream = %s\n", __func__, substream->name);
1114 ret = msm8960_aux_pcm_get_gpios();
1115 if (ret < 0) {
1116 pr_err("%s: Aux PCM GPIO request failed\n", __func__);
1117 return -EINVAL;
1118 }
1119 return 0;
1120}
1121
1122static void msm8960_auxpcm_shutdown(struct snd_pcm_substream *substream)
1123{
1124
1125 pr_debug("%s(): substream = %s\n", __func__, substream->name);
1126 msm8960_aux_pcm_free_gpios();
1127}
1128
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001129static void msm8960_shutdown(struct snd_pcm_substream *substream)
1130{
Kiran Kandi97fe19d2012-05-20 22:34:04 -07001131 struct snd_soc_pcm_runtime *rtd = substream->private_data;
1132
1133 pr_debug("%s(): dai_link str_name = %s cpu_dai = %s codec_dai = %s\n",
1134 __func__, rtd->dai_link->stream_name,
1135 rtd->dai_link->cpu_dai_name, rtd->dai_link->codec_dai_name);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001136}
1137
1138static struct snd_soc_ops msm8960_be_ops = {
1139 .startup = msm8960_startup,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08001140 .hw_params = msm8960_hw_params,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001141 .shutdown = msm8960_shutdown,
1142};
1143
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001144static struct snd_soc_ops msm8960_auxpcm_be_ops = {
1145 .startup = msm8960_auxpcm_startup,
1146 .shutdown = msm8960_auxpcm_shutdown,
1147};
1148
Kiran Kandi97fe19d2012-05-20 22:34:04 -07001149static struct snd_soc_ops msm8960_slimbus_2_be_ops = {
1150 .startup = msm8960_startup,
1151 .hw_params = msm8960_slimbus_2_hw_params,
1152 .shutdown = msm8960_shutdown,
1153};
1154
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001155/* Digital audio interface glue - connects codec <---> CPU */
Kuirong Wangb25838e2012-01-16 23:37:23 -08001156static struct snd_soc_dai_link msm8960_dai_common[] = {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001157 /* FrontEnd DAI Links */
1158 {
1159 .name = "MSM8960 Media1",
1160 .stream_name = "MultiMedia1",
1161 .cpu_dai_name = "MultiMedia1",
1162 .platform_name = "msm-pcm-dsp",
1163 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001164 .codec_dai_name = "snd-soc-dummy-dai",
1165 .codec_name = "snd-soc-dummy",
1166 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1167 .ignore_suspend = 1,
1168 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001169 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA1
1170 },
1171 {
1172 .name = "MSM8960 Media2",
1173 .stream_name = "MultiMedia2",
1174 .cpu_dai_name = "MultiMedia2",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001175 .platform_name = "msm-multi-ch-pcm-dsp",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001176 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001177 .codec_dai_name = "snd-soc-dummy-dai",
1178 .codec_name = "snd-soc-dummy",
1179 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1180 .ignore_suspend = 1,
1181 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001182 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA2,
1183 },
1184 {
1185 .name = "Circuit-Switch Voice",
1186 .stream_name = "CS-Voice",
1187 .cpu_dai_name = "CS-VOICE",
1188 .platform_name = "msm-pcm-voice",
1189 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001190 .codec_dai_name = "snd-soc-dummy-dai",
1191 .codec_name = "snd-soc-dummy",
1192 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001193 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001194 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001195 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1196 .be_id = MSM_FRONTEND_DAI_CS_VOICE,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001197 },
1198 {
1199 .name = "MSM VoIP",
1200 .stream_name = "VoIP",
1201 .cpu_dai_name = "VoIP",
1202 .platform_name = "msm-voip-dsp",
1203 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001204 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1205 .codec_dai_name = "snd-soc-dummy-dai",
1206 .codec_name = "snd-soc-dummy",
1207 .ignore_suspend = 1,
1208 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001209 .be_id = MSM_FRONTEND_DAI_VOIP,
1210 },
1211 {
1212 .name = "MSM8960 LPA",
1213 .stream_name = "LPA",
1214 .cpu_dai_name = "MultiMedia3",
1215 .platform_name = "msm-pcm-lpa",
1216 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001217 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1218 .codec_dai_name = "snd-soc-dummy-dai",
1219 .codec_name = "snd-soc-dummy",
1220 .ignore_suspend = 1,
1221 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001222 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA3,
1223 },
1224 /* Hostless PMC purpose */
1225 {
1226 .name = "SLIMBUS_0 Hostless",
1227 .stream_name = "SLIMBUS_0 Hostless",
1228 .cpu_dai_name = "SLIMBUS0_HOSTLESS",
1229 .platform_name = "msm-pcm-hostless",
1230 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001231 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001232 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001233 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001234 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1235 .codec_dai_name = "snd-soc-dummy-dai",
1236 .codec_name = "snd-soc-dummy",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001237 },
1238 {
1239 .name = "INT_FM Hostless",
1240 .stream_name = "INT_FM Hostless",
1241 .cpu_dai_name = "INT_FM_HOSTLESS",
1242 .platform_name = "msm-pcm-hostless",
1243 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001244 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001245 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001246 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001247 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1248 .codec_dai_name = "snd-soc-dummy-dai",
1249 .codec_name = "snd-soc-dummy",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001250 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301251 {
1252 .name = "MSM AFE-PCM RX",
1253 .stream_name = "AFE-PROXY RX",
1254 .cpu_dai_name = "msm-dai-q6.241",
1255 .codec_name = "msm-stub-codec.1",
1256 .codec_dai_name = "msm-stub-rx",
1257 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001258 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001259 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301260 },
1261 {
1262 .name = "MSM AFE-PCM TX",
1263 .stream_name = "AFE-PROXY TX",
1264 .cpu_dai_name = "msm-dai-q6.240",
1265 .codec_name = "msm-stub-codec.1",
1266 .codec_dai_name = "msm-stub-tx",
1267 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001268 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301269 },
Asish Bhattacharya09f9e0a2011-11-11 13:22:47 +05301270 {
1271 .name = "MSM8960 Compr",
1272 .stream_name = "COMPR",
1273 .cpu_dai_name = "MultiMedia4",
1274 .platform_name = "msm-compr-dsp",
1275 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001276 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1277 .codec_dai_name = "snd-soc-dummy-dai",
1278 .codec_name = "snd-soc-dummy",
1279 .ignore_suspend = 1,
1280 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Asish Bhattacharya09f9e0a2011-11-11 13:22:47 +05301281 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA4,
1282 },
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001283 {
1284 .name = "AUXPCM Hostless",
1285 .stream_name = "AUXPCM Hostless",
1286 .cpu_dai_name = "AUXPCM_HOSTLESS",
1287 .platform_name = "msm-pcm-hostless",
1288 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001289 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001290 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1291 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001292 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1293 .codec_dai_name = "snd-soc-dummy-dai",
1294 .codec_name = "snd-soc-dummy",
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001295 },
Alex Wongb3d06a02012-01-12 10:00:41 -08001296 /* HDMI Hostless */
1297 {
1298 .name = "HDMI_RX_HOSTLESS",
1299 .stream_name = "HDMI_RX_HOSTLESS",
1300 .cpu_dai_name = "HDMI_HOSTLESS",
1301 .platform_name = "msm-pcm-hostless",
1302 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001303 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Alex Wongb3d06a02012-01-12 10:00:41 -08001304 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb3d06a02012-01-12 10:00:41 -08001305 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001306 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1307 .codec_dai_name = "snd-soc-dummy-dai",
1308 .codec_name = "snd-soc-dummy",
Alex Wongb3d06a02012-01-12 10:00:41 -08001309 },
Venkat Sudhir1c79c3b2012-04-09 23:42:28 -07001310 {
1311 .name = "VoLTE",
1312 .stream_name = "VoLTE",
1313 .cpu_dai_name = "VoLTE",
1314 .platform_name = "msm-pcm-voice",
1315 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001316 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Venkat Sudhir1c79c3b2012-04-09 23:42:28 -07001317 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1318 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001319 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1320 .codec_dai_name = "snd-soc-dummy-dai",
1321 .codec_name = "snd-soc-dummy",
1322 .be_id = MSM_FRONTEND_DAI_VOLTE,
Venkat Sudhir1c79c3b2012-04-09 23:42:28 -07001323 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001324 /* Backend BT/FM DAI Links */
1325 {
1326 .name = LPASS_BE_INT_BT_SCO_RX,
1327 .stream_name = "Internal BT-SCO Playback",
1328 .cpu_dai_name = "msm-dai-q6.12288",
1329 .platform_name = "msm-pcm-routing",
1330 .codec_name = "msm-stub-codec.1",
1331 .codec_dai_name = "msm-stub-rx",
1332 .no_pcm = 1,
1333 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_RX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001334 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001335 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001336 },
1337 {
1338 .name = LPASS_BE_INT_BT_SCO_TX,
1339 .stream_name = "Internal BT-SCO Capture",
1340 .cpu_dai_name = "msm-dai-q6.12289",
1341 .platform_name = "msm-pcm-routing",
1342 .codec_name = "msm-stub-codec.1",
1343 .codec_dai_name = "msm-stub-tx",
1344 .no_pcm = 1,
1345 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_TX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001346 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001347 },
1348 {
1349 .name = LPASS_BE_INT_FM_RX,
1350 .stream_name = "Internal FM Playback",
1351 .cpu_dai_name = "msm-dai-q6.12292",
1352 .platform_name = "msm-pcm-routing",
1353 .codec_name = "msm-stub-codec.1",
1354 .codec_dai_name = "msm-stub-rx",
1355 .no_pcm = 1,
1356 .be_id = MSM_BACKEND_DAI_INT_FM_RX,
1357 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001358 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001359 },
1360 {
1361 .name = LPASS_BE_INT_FM_TX,
1362 .stream_name = "Internal FM Capture",
1363 .cpu_dai_name = "msm-dai-q6.12293",
1364 .platform_name = "msm-pcm-routing",
1365 .codec_name = "msm-stub-codec.1",
1366 .codec_dai_name = "msm-stub-tx",
1367 .no_pcm = 1,
1368 .be_id = MSM_BACKEND_DAI_INT_FM_TX,
1369 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1370 },
1371 /* HDMI BACK END DAI Link */
1372 {
1373 .name = LPASS_BE_HDMI,
1374 .stream_name = "HDMI Playback",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001375 .cpu_dai_name = "msm-dai-q6-hdmi.8",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001376 .platform_name = "msm-pcm-routing",
1377 .codec_name = "msm-stub-codec.1",
1378 .codec_dai_name = "msm-stub-rx",
1379 .no_pcm = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001380 .be_id = MSM_BACKEND_DAI_HDMI_RX,
Helen Zeng73f7fc62011-11-18 16:29:59 -08001381 .be_hw_params_fixup = msm8960_hdmi_be_hw_params_fixup,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001382 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001383 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301384 /* Backend AFE DAI Links */
1385 {
1386 .name = LPASS_BE_AFE_PCM_RX,
1387 .stream_name = "AFE Playback",
1388 .cpu_dai_name = "msm-dai-q6.224",
1389 .platform_name = "msm-pcm-routing",
1390 .codec_name = "msm-stub-codec.1",
1391 .codec_dai_name = "msm-stub-rx",
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301392 .no_pcm = 1,
1393 .be_id = MSM_BACKEND_DAI_AFE_PCM_RX,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001394 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301395 },
1396 {
1397 .name = LPASS_BE_AFE_PCM_TX,
1398 .stream_name = "AFE Capture",
1399 .cpu_dai_name = "msm-dai-q6.225",
1400 .platform_name = "msm-pcm-routing",
1401 .codec_name = "msm-stub-codec.1",
1402 .codec_dai_name = "msm-stub-tx",
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301403 .no_pcm = 1,
1404 .be_id = MSM_BACKEND_DAI_AFE_PCM_TX,
1405 },
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001406 /* AUX PCM Backend DAI Links */
1407 {
1408 .name = LPASS_BE_AUXPCM_RX,
1409 .stream_name = "AUX PCM Playback",
1410 .cpu_dai_name = "msm-dai-q6.2",
1411 .platform_name = "msm-pcm-routing",
1412 .codec_name = "msm-stub-codec.1",
1413 .codec_dai_name = "msm-stub-rx",
1414 .no_pcm = 1,
1415 .be_id = MSM_BACKEND_DAI_AUXPCM_RX,
1416 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1417 .ops = &msm8960_auxpcm_be_ops,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001418 .ignore_pmdown_time = 1,
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001419 },
1420 {
1421 .name = LPASS_BE_AUXPCM_TX,
1422 .stream_name = "AUX PCM Capture",
1423 .cpu_dai_name = "msm-dai-q6.3",
1424 .platform_name = "msm-pcm-routing",
1425 .codec_name = "msm-stub-codec.1",
1426 .codec_dai_name = "msm-stub-tx",
1427 .no_pcm = 1,
1428 .be_id = MSM_BACKEND_DAI_AUXPCM_TX,
1429 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1430 },
Helen Zeng0705a5f2011-10-14 15:29:52 -07001431 /* Incall Music BACK END DAI Link */
1432 {
1433 .name = LPASS_BE_VOICE_PLAYBACK_TX,
1434 .stream_name = "Voice Farend Playback",
1435 .cpu_dai_name = "msm-dai-q6.32773",
1436 .platform_name = "msm-pcm-routing",
1437 .codec_name = "msm-stub-codec.1",
1438 .codec_dai_name = "msm-stub-rx",
1439 .no_pcm = 1,
Helen Zeng0705a5f2011-10-14 15:29:52 -07001440 .be_id = MSM_BACKEND_DAI_VOICE_PLAYBACK_TX,
1441 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1442 },
Helen Zenge3d716a2011-10-14 16:32:16 -07001443 /* Incall Record Uplink BACK END DAI Link */
1444 {
1445 .name = LPASS_BE_INCALL_RECORD_TX,
1446 .stream_name = "Voice Uplink Capture",
1447 .cpu_dai_name = "msm-dai-q6.32772",
1448 .platform_name = "msm-pcm-routing",
1449 .codec_name = "msm-stub-codec.1",
1450 .codec_dai_name = "msm-stub-tx",
1451 .no_pcm = 1,
Helen Zenge3d716a2011-10-14 16:32:16 -07001452 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_TX,
1453 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1454 },
1455 /* Incall Record Downlink BACK END DAI Link */
1456 {
1457 .name = LPASS_BE_INCALL_RECORD_RX,
1458 .stream_name = "Voice Downlink Capture",
1459 .cpu_dai_name = "msm-dai-q6.32771",
1460 .platform_name = "msm-pcm-routing",
1461 .codec_name = "msm-stub-codec.1",
1462 .codec_dai_name = "msm-stub-tx",
1463 .no_pcm = 1,
Helen Zenge3d716a2011-10-14 16:32:16 -07001464 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_RX,
1465 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001466 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Helen Zenge3d716a2011-10-14 16:32:16 -07001467 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001468};
1469
Kuirong Wangb25838e2012-01-16 23:37:23 -08001470static struct snd_soc_dai_link msm8960_dai_delta_tabla1x[] = {
1471 /* Backend DAI Links */
1472 {
1473 .name = LPASS_BE_SLIMBUS_0_RX,
1474 .stream_name = "Slimbus Playback",
1475 .cpu_dai_name = "msm-dai-q6.16384",
1476 .platform_name = "msm-pcm-routing",
1477 .codec_name = "tabla1x_codec",
1478 .codec_dai_name = "tabla_rx1",
1479 .no_pcm = 1,
1480 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1481 .init = &msm8960_audrx_init,
1482 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1483 .ops = &msm8960_be_ops,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001484 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Kuirong Wangb25838e2012-01-16 23:37:23 -08001485 },
1486 {
1487 .name = LPASS_BE_SLIMBUS_0_TX,
1488 .stream_name = "Slimbus Capture",
1489 .cpu_dai_name = "msm-dai-q6.16385",
1490 .platform_name = "msm-pcm-routing",
1491 .codec_name = "tabla1x_codec",
1492 .codec_dai_name = "tabla_tx1",
1493 .no_pcm = 1,
1494 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1495 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1496 .ops = &msm8960_be_ops,
1497 },
Kiran Kandi97fe19d2012-05-20 22:34:04 -07001498 /* Ultrasound TX Back End DAI Link */
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001499 {
Kiran Kandi97fe19d2012-05-20 22:34:04 -07001500 .name = "SLIMBUS_2 Hostless Capture",
1501 .stream_name = "SLIMBUS_2 Hostless Capture",
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001502 .cpu_dai_name = "msm-dai-q6.16389",
1503 .platform_name = "msm-pcm-hostless",
1504 .codec_name = "tabla1x_codec",
1505 .codec_dai_name = "tabla_tx2",
Swaminathan Sathappan630c5a42012-05-10 17:33:01 -07001506 .ignore_suspend = 1,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001507 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Kiran Kandi97fe19d2012-05-20 22:34:04 -07001508 .ops = &msm8960_slimbus_2_be_ops,
1509 },
1510 /* Ultrasound RX Back End DAI Link */
1511 {
1512 .name = "SLIMBUS_2 Hostless Playback",
1513 .stream_name = "SLIMBUS_2 Hostless Playback",
1514 .cpu_dai_name = "msm-dai-q6.16388",
1515 .platform_name = "msm-pcm-hostless",
1516 .codec_name = "tabla1x_codec",
1517 .codec_dai_name = "tabla_rx3",
1518 .ignore_suspend = 1,
1519 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1520 .ops = &msm8960_slimbus_2_be_ops,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001521 },
Kuirong Wangb25838e2012-01-16 23:37:23 -08001522};
1523
1524
1525static struct snd_soc_dai_link msm8960_dai_delta_tabla2x[] = {
1526 /* Backend DAI Links */
1527 {
1528 .name = LPASS_BE_SLIMBUS_0_RX,
1529 .stream_name = "Slimbus Playback",
1530 .cpu_dai_name = "msm-dai-q6.16384",
1531 .platform_name = "msm-pcm-routing",
1532 .codec_name = "tabla_codec",
1533 .codec_dai_name = "tabla_rx1",
1534 .no_pcm = 1,
1535 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1536 .init = &msm8960_audrx_init,
1537 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1538 .ops = &msm8960_be_ops,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001539 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Kuirong Wangb25838e2012-01-16 23:37:23 -08001540 },
1541 {
1542 .name = LPASS_BE_SLIMBUS_0_TX,
1543 .stream_name = "Slimbus Capture",
1544 .cpu_dai_name = "msm-dai-q6.16385",
1545 .platform_name = "msm-pcm-routing",
1546 .codec_name = "tabla_codec",
1547 .codec_dai_name = "tabla_tx1",
1548 .no_pcm = 1,
1549 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1550 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1551 .ops = &msm8960_be_ops,
1552 },
Kiran Kandi97fe19d2012-05-20 22:34:04 -07001553 /* Ultrasound TX Back End DAI Link */
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001554 {
Kiran Kandi97fe19d2012-05-20 22:34:04 -07001555 .name = "SLIMBUS_2 Hostless Capture",
1556 .stream_name = "SLIMBUS_2 Hostless Capture",
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001557 .cpu_dai_name = "msm-dai-q6.16389",
1558 .platform_name = "msm-pcm-hostless",
1559 .codec_name = "tabla_codec",
1560 .codec_dai_name = "tabla_tx2",
Swaminathan Sathappan630c5a42012-05-10 17:33:01 -07001561 .ignore_suspend = 1,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001562 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Kiran Kandi97fe19d2012-05-20 22:34:04 -07001563 .ops = &msm8960_slimbus_2_be_ops,
1564 },
1565 /* Ultrasound RX Back End DAI Link */
1566 {
1567 .name = "SLIMBUS_2 Hostless Playback",
1568 .stream_name = "SLIMBUS_2 Hostless Playback",
1569 .cpu_dai_name = "msm-dai-q6.16388",
1570 .platform_name = "msm-pcm-hostless",
1571 .codec_name = "tabla_codec",
1572 .codec_dai_name = "tabla_rx3",
1573 .ignore_suspend = 1,
1574 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1575 .ops = &msm8960_slimbus_2_be_ops,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001576 },
Kuirong Wangb25838e2012-01-16 23:37:23 -08001577};
1578
1579static struct snd_soc_dai_link msm8960_tabla1x_dai[
1580 ARRAY_SIZE(msm8960_dai_common) +
1581 ARRAY_SIZE(msm8960_dai_delta_tabla1x)];
1582
1583
1584static struct snd_soc_dai_link msm8960_dai[
1585 ARRAY_SIZE(msm8960_dai_common) +
1586 ARRAY_SIZE(msm8960_dai_delta_tabla2x)];
1587
1588static struct snd_soc_card snd_soc_tabla1x_card_msm8960 = {
1589 .name = "msm8960-tabla1x-snd-card",
1590 .dai_link = msm8960_tabla1x_dai,
1591 .num_links = ARRAY_SIZE(msm8960_tabla1x_dai),
Steve Mucklef132c6c2012-06-06 18:30:57 -07001592 .controls = tabla_msm8960_controls,
1593 .num_controls = ARRAY_SIZE(tabla_msm8960_controls),
Kuirong Wangb25838e2012-01-16 23:37:23 -08001594};
1595
1596static struct snd_soc_card snd_soc_card_msm8960 = {
1597 .name = "msm8960-snd-card",
1598 .dai_link = msm8960_dai,
1599 .num_links = ARRAY_SIZE(msm8960_dai),
Steve Mucklef132c6c2012-06-06 18:30:57 -07001600 .controls = tabla_msm8960_controls,
1601 .num_controls = ARRAY_SIZE(tabla_msm8960_controls),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001602};
1603
1604static struct platform_device *msm8960_snd_device;
Kuirong Wangb25838e2012-01-16 23:37:23 -08001605static struct platform_device *msm8960_snd_tabla1x_device;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001606
1607static int msm8960_configure_headset_mic_gpios(void)
1608{
1609 int ret;
1610 struct pm_gpio param = {
1611 .direction = PM_GPIO_DIR_OUT,
1612 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
1613 .output_value = 1,
1614 .pull = PM_GPIO_PULL_NO,
1615 .vin_sel = PM_GPIO_VIN_S4,
1616 .out_strength = PM_GPIO_STRENGTH_MED,
1617 .function = PM_GPIO_FUNC_NORMAL,
1618 };
1619
1620 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(23), "AV_SWITCH");
1621 if (ret) {
1622 pr_err("%s: Failed to request gpio %d\n", __func__,
1623 PM8921_GPIO_PM_TO_SYS(23));
1624 return ret;
1625 }
1626
1627 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(23), &param);
1628 if (ret)
1629 pr_err("%s: Failed to configure gpio %d\n", __func__,
1630 PM8921_GPIO_PM_TO_SYS(23));
1631 else
1632 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(23), 0);
1633
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001634 ret = gpio_request(us_euro_sel_gpio, "US_EURO_SWITCH");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001635 if (ret) {
1636 pr_err("%s: Failed to request gpio %d\n", __func__,
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001637 us_euro_sel_gpio);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001638 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1639 return ret;
1640 }
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001641 ret = pm8xxx_gpio_config(us_euro_sel_gpio, &param);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001642 if (ret)
1643 pr_err("%s: Failed to configure gpio %d\n", __func__,
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001644 us_euro_sel_gpio);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001645 else
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001646 gpio_direction_output(us_euro_sel_gpio, 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001647
1648 return 0;
1649}
1650static void msm8960_free_headset_mic_gpios(void)
1651{
1652 if (msm8960_headset_gpios_configured) {
1653 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001654 gpio_free(us_euro_sel_gpio);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001655 }
1656}
1657
1658static int __init msm8960_audio_init(void)
1659{
1660 int ret;
1661
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301662 if (!cpu_is_msm8960()) {
Steve Mucklef132c6c2012-06-06 18:30:57 -07001663 pr_debug("%s: Not the right machine type\n", __func__);
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001664 return -ENODEV ;
1665 }
Joonwoo Park0976d012011-12-22 11:48:18 -08001666
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001667 mbhc_cfg.calibration = def_tabla_mbhc_cal();
1668 if (!mbhc_cfg.calibration) {
Joonwoo Park0976d012011-12-22 11:48:18 -08001669 pr_err("Calibration data allocation failed\n");
1670 return -ENOMEM;
1671 }
1672
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001673 msm8960_snd_device = platform_device_alloc("soc-audio", 0);
1674 if (!msm8960_snd_device) {
1675 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001676 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001677 return -ENOMEM;
1678 }
1679
Kuirong Wangb25838e2012-01-16 23:37:23 -08001680 memcpy(msm8960_dai, msm8960_dai_common, sizeof(msm8960_dai_common));
1681 memcpy(msm8960_dai + ARRAY_SIZE(msm8960_dai_common),
1682 msm8960_dai_delta_tabla2x, sizeof(msm8960_dai_delta_tabla2x));
1683
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001684 platform_set_drvdata(msm8960_snd_device, &snd_soc_card_msm8960);
1685 ret = platform_device_add(msm8960_snd_device);
1686 if (ret) {
1687 platform_device_put(msm8960_snd_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001688 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001689 return ret;
1690 }
1691
Kuirong Wangb25838e2012-01-16 23:37:23 -08001692 msm8960_snd_tabla1x_device = platform_device_alloc("soc-audio", 1);
1693 if (!msm8960_snd_tabla1x_device) {
1694 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001695 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001696 return -ENOMEM;
1697 }
1698
1699 memcpy(msm8960_tabla1x_dai, msm8960_dai_common,
1700 sizeof(msm8960_dai_common));
1701 memcpy(msm8960_tabla1x_dai + ARRAY_SIZE(msm8960_dai_common),
1702 msm8960_dai_delta_tabla1x, sizeof(msm8960_dai_delta_tabla1x));
1703
1704 platform_set_drvdata(msm8960_snd_tabla1x_device,
1705 &snd_soc_tabla1x_card_msm8960);
1706 ret = platform_device_add(msm8960_snd_tabla1x_device);
1707 if (ret) {
1708 platform_device_put(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001709 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001710 return ret;
1711 }
1712
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001713 if (msm8960_configure_headset_mic_gpios()) {
1714 pr_err("%s Fail to configure headset mic gpios\n", __func__);
1715 msm8960_headset_gpios_configured = 0;
1716 } else
1717 msm8960_headset_gpios_configured = 1;
1718
Joonwoo Park28f49c82012-03-16 12:29:21 -07001719 mutex_init(&cdc_mclk_mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001720 return ret;
1721
1722}
1723module_init(msm8960_audio_init);
1724
1725static void __exit msm8960_audio_exit(void)
1726{
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301727 if (!cpu_is_msm8960()) {
Steve Mucklef132c6c2012-06-06 18:30:57 -07001728 pr_debug("%s: Not the right machine type\n", __func__);
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001729 return ;
1730 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001731 msm8960_free_headset_mic_gpios();
1732 platform_device_unregister(msm8960_snd_device);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001733 platform_device_unregister(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001734 kfree(mbhc_cfg.calibration);
Joonwoo Park28f49c82012-03-16 12:29:21 -07001735 mutex_destroy(&cdc_mclk_mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001736}
1737module_exit(msm8960_audio_exit);
1738
1739MODULE_DESCRIPTION("ALSA SoC MSM8960");
1740MODULE_LICENSE("GPL v2");