blob: e3424695c198c9a0d0e4e3cb79b930e83d096cf2 [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)
64#define GPIO_DETECT_USED false
65
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
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -070074static int msm8960_btsco_rate = BTSCO_RATE_8KHZ;
75static int msm8960_btsco_ch = 1;
76
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070077static struct clk *codec_clk;
78static int clk_users;
79
80static int msm8960_headset_gpios_configured;
81
82static struct snd_soc_jack hs_jack;
Bradley Rubincb1e2732011-06-23 16:49:20 -070083static struct snd_soc_jack button_jack;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070084
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070085static int msm8960_enable_codec_ext_clk(struct snd_soc_codec *codec, int enable,
86 bool dapm);
87
88static struct tabla_mbhc_config mbhc_cfg = {
89 .headset_jack = &hs_jack,
90 .button_jack = &button_jack,
91 .read_fw_bin = false,
92 .calibration = NULL,
93 .micbias = TABLA_MICBIAS2,
94 .mclk_cb_fn = msm8960_enable_codec_ext_clk,
95 .mclk_rate = TABLA_EXT_CLK_RATE,
96 .gpio = 0,
97 .gpio_irq = 0,
98 .gpio_level_insert = 1,
99};
Joonwoo Park0976d012011-12-22 11:48:18 -0800100
Kiran Kandi462acea2011-08-31 23:53:14 -0700101static void msm8960_enable_ext_spk_amp_gpio(u32 spk_amp_gpio)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700102{
103 int ret = 0;
104
105 struct pm_gpio param = {
106 .direction = PM_GPIO_DIR_OUT,
107 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
108 .output_value = 1,
109 .pull = PM_GPIO_PULL_NO,
110 .vin_sel = PM_GPIO_VIN_S4,
111 .out_strength = PM_GPIO_STRENGTH_MED,
Kiran Kandi462acea2011-08-31 23:53:14 -0700112 .
113 function = PM_GPIO_FUNC_NORMAL,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700114 };
115
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800116 if (spk_amp_gpio == bottom_spk_pamp_gpio) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700117
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800118 ret = gpio_request(bottom_spk_pamp_gpio, "BOTTOM_SPK_AMP");
Kiran Kandi462acea2011-08-31 23:53:14 -0700119 if (ret) {
120 pr_err("%s: Error requesting BOTTOM SPK AMP GPIO %u\n",
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800121 __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700122 return;
123 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800124 ret = pm8xxx_gpio_config(bottom_spk_pamp_gpio, &param);
Kiran Kandi462acea2011-08-31 23:53:14 -0700125 if (ret)
126 pr_err("%s: Failed to configure Bottom Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800127 " gpio %u\n", __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700128 else {
129 pr_debug("%s: enable Bottom spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800130 gpio_direction_output(bottom_spk_pamp_gpio, 1);
Kiran Kandi462acea2011-08-31 23:53:14 -0700131 }
132
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800133 } else if (spk_amp_gpio == top_spk_pamp_gpio) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700134
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800135 ret = gpio_request(top_spk_pamp_gpio, "TOP_SPK_AMP");
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700136 if (ret) {
137 pr_err("%s: Error requesting GPIO %d\n", __func__,
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800138 top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700139 return;
140 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800141 ret = pm8xxx_gpio_config(top_spk_pamp_gpio, &param);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700142 if (ret)
Kiran Kandi462acea2011-08-31 23:53:14 -0700143 pr_err("%s: Failed to configure Top Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800144 " gpio %u\n", __func__, top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700145 else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700146 pr_debug("%s: enable Top spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800147 gpio_direction_output(top_spk_pamp_gpio, 1);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700148 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700149 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700150 pr_err("%s: ERROR : Invalid External Speaker Ampl GPIO."
151 " gpio = %u\n", __func__, spk_amp_gpio);
152 return;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700153 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700154}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700155
Kiran Kandi462acea2011-08-31 23:53:14 -0700156static void msm8960_ext_spk_power_amp_on(u32 spk)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700157{
Kiran Kandi462acea2011-08-31 23:53:14 -0700158 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
159
160 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
161 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
162
163 pr_debug("%s() External Bottom Speaker Ampl already "
164 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700165 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700166 }
167
168 msm8960_ext_bottom_spk_pamp |= spk;
169
170 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
171 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
172
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800173 msm8960_enable_ext_spk_amp_gpio(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700174 pr_debug("%s: slepping 4 ms after turning on external "
175 " Bottom Speaker Ampl\n", __func__);
176 usleep_range(4000, 4000);
177 }
178
179 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
180
181 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
182 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
183
184 pr_debug("%s() External Top Speaker Ampl already"
185 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700186 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700187 }
188
189 msm8960_ext_top_spk_pamp |= spk;
190
191 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
192 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
193
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800194 msm8960_enable_ext_spk_amp_gpio(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700195 pr_debug("%s: sleeping 4 ms after turning on "
196 " external Top Speaker Ampl\n", __func__);
197 usleep_range(4000, 4000);
198 }
199 } else {
200
201 pr_err("%s: ERROR : Invalid External Speaker Ampl. spk = 0x%08x\n",
202 __func__, spk);
203 return;
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700204 }
Kiran Kandi462acea2011-08-31 23:53:14 -0700205}
206
207static void msm8960_ext_spk_power_amp_off(u32 spk)
208{
209 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
210
211 if (!msm8960_ext_bottom_spk_pamp)
212 return;
213
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800214 gpio_direction_output(bottom_spk_pamp_gpio, 0);
215 gpio_free(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700216 msm8960_ext_bottom_spk_pamp = 0;
217
218 pr_debug("%s: sleeping 4 ms after turning off external Bottom"
219 " Speaker Ampl\n", __func__);
220
221 usleep_range(4000, 4000);
222
223 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
224
225 if (!msm8960_ext_top_spk_pamp)
226 return;
227
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800228 gpio_direction_output(top_spk_pamp_gpio, 0);
229 gpio_free(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700230 msm8960_ext_top_spk_pamp = 0;
231
232 pr_debug("%s: sleeping 4 ms after turning off external Top"
233 " Spkaker Ampl\n", __func__);
234
235 usleep_range(4000, 4000);
236 } else {
237
238 pr_err("%s: ERROR : Invalid Ext Spk Ampl. spk = 0x%08x\n",
239 __func__, spk);
240 return;
241 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700242}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700243
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700244static void msm8960_ext_control(struct snd_soc_codec *codec)
245{
246 struct snd_soc_dapm_context *dapm = &codec->dapm;
247
248 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700249 if (msm8960_spk_control == MSM8960_SPK_ON) {
250 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
251 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
252 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
253 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
254 } else {
255 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Pos");
256 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Neg");
257 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Pos");
258 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Neg");
259 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700260
261 snd_soc_dapm_sync(dapm);
262}
263
264static int msm8960_get_spk(struct snd_kcontrol *kcontrol,
265 struct snd_ctl_elem_value *ucontrol)
266{
267 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
268 ucontrol->value.integer.value[0] = msm8960_spk_control;
269 return 0;
270}
271static int msm8960_set_spk(struct snd_kcontrol *kcontrol,
272 struct snd_ctl_elem_value *ucontrol)
273{
274 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
275
276 pr_debug("%s()\n", __func__);
277 if (msm8960_spk_control == ucontrol->value.integer.value[0])
278 return 0;
279
280 msm8960_spk_control = ucontrol->value.integer.value[0];
281 msm8960_ext_control(codec);
282 return 1;
283}
284static int msm8960_spkramp_event(struct snd_soc_dapm_widget *w,
285 struct snd_kcontrol *k, int event)
286{
287 pr_debug("%s() %x\n", __func__, SND_SOC_DAPM_EVENT_ON(event));
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700288
289 if (SND_SOC_DAPM_EVENT_ON(event)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700290 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
291 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_POS);
292 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
293 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_NEG);
294 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
295 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_POS);
296 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
297 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_NEG);
298 else {
299 pr_err("%s() Invalid Speaker Widget = %s\n",
300 __func__, w->name);
301 return -EINVAL;
302 }
303
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700304 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700305 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
306 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_POS);
307 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
308 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_NEG);
309 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
310 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_POS);
311 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
312 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_NEG);
313 else {
314 pr_err("%s() Invalid Speaker Widget = %s\n",
315 __func__, w->name);
316 return -EINVAL;
317 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700318 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700319 return 0;
320}
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700321
322static int msm8960_enable_codec_ext_clk(struct snd_soc_codec *codec, int enable,
323 bool dapm)
Joonwoo Park0976d012011-12-22 11:48:18 -0800324{
325 pr_debug("%s: enable = %d\n", __func__, enable);
326 if (enable) {
327 clk_users++;
328 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
329 if (clk_users != 1)
330 return 0;
331
Joonwoo Park0976d012011-12-22 11:48:18 -0800332 if (codec_clk) {
333 clk_set_rate(codec_clk, TABLA_EXT_CLK_RATE);
334 clk_enable(codec_clk);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700335 tabla_mclk_enable(codec, 1, dapm);
Joonwoo Park0976d012011-12-22 11:48:18 -0800336 } else {
337 pr_err("%s: Error setting Tabla MCLK\n", __func__);
338 clk_users--;
339 return -EINVAL;
340 }
341 } else {
342 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
343 if (clk_users == 0)
344 return 0;
345 clk_users--;
346 if (!clk_users) {
347 pr_debug("%s: disabling MCLK. clk_users = %d\n",
348 __func__, clk_users);
349 clk_disable(codec_clk);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700350 tabla_mclk_enable(codec, 0, dapm);
Joonwoo Park0976d012011-12-22 11:48:18 -0800351 }
352 }
353 return 0;
354}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700355
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700356static int msm8960_mclk_event(struct snd_soc_dapm_widget *w,
357 struct snd_kcontrol *kcontrol, int event)
358{
359 pr_debug("%s: event = %d\n", __func__, event);
360
361 switch (event) {
362 case SND_SOC_DAPM_PRE_PMU:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700363 return msm8960_enable_codec_ext_clk(w->codec, 1, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700364 case SND_SOC_DAPM_POST_PMD:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700365 return msm8960_enable_codec_ext_clk(w->codec, 0, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700366 }
367 return 0;
368}
369
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700370static const struct snd_soc_dapm_widget msm8960_dapm_widgets[] = {
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700371
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700372 SND_SOC_DAPM_SUPPLY("MCLK", SND_SOC_NOPM, 0, 0,
373 msm8960_mclk_event, SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
374
Kiran Kandi462acea2011-08-31 23:53:14 -0700375 SND_SOC_DAPM_SPK("Ext Spk Bottom Pos", msm8960_spkramp_event),
376 SND_SOC_DAPM_SPK("Ext Spk Bottom Neg", msm8960_spkramp_event),
377
378 SND_SOC_DAPM_SPK("Ext Spk Top Pos", msm8960_spkramp_event),
379 SND_SOC_DAPM_SPK("Ext Spk Top Neg", msm8960_spkramp_event),
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700380
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700381 SND_SOC_DAPM_MIC("Handset Mic", NULL),
382 SND_SOC_DAPM_MIC("Headset Mic", NULL),
383 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700384 SND_SOC_DAPM_MIC("ANCRight Headset Mic", NULL),
385 SND_SOC_DAPM_MIC("ANCLeft Headset Mic", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700386
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700387 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700388 SND_SOC_DAPM_MIC("Digital Mic2", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700389 SND_SOC_DAPM_MIC("Digital Mic3", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700390 SND_SOC_DAPM_MIC("Digital Mic4", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700391 SND_SOC_DAPM_MIC("Digital Mic5", NULL),
392 SND_SOC_DAPM_MIC("Digital Mic6", NULL),
393
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700394};
395
Bradley Rubin229c6a52011-07-12 16:18:48 -0700396static const struct snd_soc_dapm_route common_audio_map[] = {
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700397
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700398 {"RX_BIAS", NULL, "MCLK"},
399 {"LDO_H", NULL, "MCLK"},
400
401 /* Speaker path */
Kiran Kandi462acea2011-08-31 23:53:14 -0700402 {"Ext Spk Bottom Pos", NULL, "LINEOUT1"},
403 {"Ext Spk Bottom Neg", NULL, "LINEOUT3"},
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700404
Kiran Kandi462acea2011-08-31 23:53:14 -0700405 {"Ext Spk Top Pos", NULL, "LINEOUT2"},
406 {"Ext Spk Top Neg", NULL, "LINEOUT4"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700407
408 /* Microphone path */
Bradley Rubin229c6a52011-07-12 16:18:48 -0700409 {"AMIC1", NULL, "MIC BIAS1 Internal1"},
410 {"MIC BIAS1 Internal1", NULL, "Handset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700411
412 {"AMIC2", NULL, "MIC BIAS2 External"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700413 {"MIC BIAS2 External", NULL, "Headset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700414
Patrick Laie519dd62011-09-28 12:37:11 -0700415 /**
416 * AMIC3 and AMIC4 inputs are connected to ANC microphones
417 * These mics are biased differently on CDP and FLUID
418 * routing entries below are based on bias arrangement
419 * on FLUID.
420 */
421 {"AMIC3", NULL, "MIC BIAS3 Internal1"},
422 {"MIC BIAS3 Internal1", NULL, "ANCRight Headset Mic"},
423
424 {"AMIC4", NULL, "MIC BIAS1 Internal2"},
425 {"MIC BIAS1 Internal2", NULL, "ANCLeft Headset Mic"},
426
Kiran Kandie9bf86a2011-07-21 16:50:41 -0700427 {"HEADPHONE", NULL, "LDO_H"},
428
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700429 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700430 * The digital Mic routes are setup considering
431 * fluid as default device.
432 */
433
434 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700435 * Digital Mic1. Front Bottom left Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700436 * Digital Mic GM5 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700437 * Conncted to DMIC2 Input on Tabla codec.
438 */
439 {"DMIC2", NULL, "MIC BIAS1 External"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700440 {"MIC BIAS1 External", NULL, "Digital Mic1"},
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700441
442 /**
443 * Digital Mic2. Front Bottom right Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700444 * Digital Mic GM6 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700445 * Conncted to DMIC1 Input on Tabla codec.
446 */
447 {"DMIC1", NULL, "MIC BIAS1 External"},
448 {"MIC BIAS1 External", NULL, "Digital Mic2"},
449
450 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700451 * Digital Mic3. Back Bottom Digital Mic on Fluid.
452 * Digital Mic GM1 on CDP mainboard.
453 * Conncted to DMIC4 Input on Tabla codec.
454 */
455 {"DMIC4", NULL, "MIC BIAS3 External"},
456 {"MIC BIAS3 External", NULL, "Digital Mic3"},
457
458 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700459 * Digital Mic4. Back top Digital Mic on Fluid.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700460 * Digital Mic GM2 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700461 * Conncted to DMIC3 Input on Tabla codec.
462 */
463 {"DMIC3", NULL, "MIC BIAS3 External"},
464 {"MIC BIAS3 External", NULL, "Digital Mic4"},
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700465
466 /**
467 * Digital Mic5. Front top Digital Mic on Fluid.
468 * Digital Mic GM3 on CDP mainboard.
469 * Conncted to DMIC5 Input on Tabla codec.
470 */
471 {"DMIC5", NULL, "MIC BIAS4 External"},
472 {"MIC BIAS4 External", NULL, "Digital Mic5"},
473
Patrick Laicb7802b2011-10-04 12:39:18 -0700474 /* Tabla digital Mic6 - back bottom digital Mic on Liquid and
475 * bottom mic on CDP. FLUID/MTP do not have dmic6 installed.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700476 */
477 {"DMIC6", NULL, "MIC BIAS4 External"},
478 {"MIC BIAS4 External", NULL, "Digital Mic6"},
Bradley Rubin229c6a52011-07-12 16:18:48 -0700479};
480
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700481static const char *spk_function[] = {"Off", "On"};
Patrick Lai9f4b4292011-07-16 22:11:09 -0700482static const char *slim0_rx_ch_text[] = {"One", "Two"};
483static const char *slim0_tx_ch_text[] = {"One", "Two", "Three", "Four"};
484
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700485static const struct soc_enum msm8960_enum[] = {
486 SOC_ENUM_SINGLE_EXT(2, spk_function),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700487 SOC_ENUM_SINGLE_EXT(2, slim0_rx_ch_text),
488 SOC_ENUM_SINGLE_EXT(4, slim0_tx_ch_text),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700489};
490
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700491static const char *btsco_rate_text[] = {"8000", "16000"};
492static const struct soc_enum msm8960_btsco_enum[] = {
493 SOC_ENUM_SINGLE_EXT(2, btsco_rate_text),
494};
495
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700496static int msm8960_slim_0_rx_ch_get(struct snd_kcontrol *kcontrol,
497 struct snd_ctl_elem_value *ucontrol)
498{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700499 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800500 msm8960_slim_0_rx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700501 ucontrol->value.integer.value[0] = msm8960_slim_0_rx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700502 return 0;
503}
504
505static int msm8960_slim_0_rx_ch_put(struct snd_kcontrol *kcontrol,
506 struct snd_ctl_elem_value *ucontrol)
507{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700508 msm8960_slim_0_rx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700509
Patrick Lai9f4b4292011-07-16 22:11:09 -0700510 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800511 msm8960_slim_0_rx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700512 return 1;
513}
514
515static int msm8960_slim_0_tx_ch_get(struct snd_kcontrol *kcontrol,
516 struct snd_ctl_elem_value *ucontrol)
517{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700518 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800519 msm8960_slim_0_tx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700520 ucontrol->value.integer.value[0] = msm8960_slim_0_tx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700521 return 0;
522}
523
524static int msm8960_slim_0_tx_ch_put(struct snd_kcontrol *kcontrol,
525 struct snd_ctl_elem_value *ucontrol)
526{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700527 msm8960_slim_0_tx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700528
529 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800530 msm8960_slim_0_tx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700531 return 1;
532}
533
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700534static int msm8960_btsco_rate_get(struct snd_kcontrol *kcontrol,
535 struct snd_ctl_elem_value *ucontrol)
536{
Joonwoo Park0976d012011-12-22 11:48:18 -0800537 pr_debug("%s: msm8960_btsco_rate = %d", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700538 ucontrol->value.integer.value[0] = msm8960_btsco_rate;
539 return 0;
540}
541
542static int msm8960_btsco_rate_put(struct snd_kcontrol *kcontrol,
543 struct snd_ctl_elem_value *ucontrol)
544{
545 switch (ucontrol->value.integer.value[0]) {
546 case 0:
547 msm8960_btsco_rate = BTSCO_RATE_8KHZ;
548 break;
549 case 1:
550 msm8960_btsco_rate = BTSCO_RATE_16KHZ;
551 break;
552 default:
553 msm8960_btsco_rate = BTSCO_RATE_8KHZ;
554 break;
555 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800556 pr_debug("%s: msm8960_btsco_rate = %d\n", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700557 return 0;
558}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700559
560static const struct snd_kcontrol_new tabla_msm8960_controls[] = {
561 SOC_ENUM_EXT("Speaker Function", msm8960_enum[0], msm8960_get_spk,
562 msm8960_set_spk),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700563 SOC_ENUM_EXT("SLIM_0_RX Channels", msm8960_enum[1],
564 msm8960_slim_0_rx_ch_get, msm8960_slim_0_rx_ch_put),
565 SOC_ENUM_EXT("SLIM_0_TX Channels", msm8960_enum[2],
566 msm8960_slim_0_tx_ch_get, msm8960_slim_0_tx_ch_put),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700567};
568
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700569static const struct snd_kcontrol_new int_btsco_rate_mixer_controls[] = {
570 SOC_ENUM_EXT("Internal BTSCO SampleRate", msm8960_btsco_enum[0],
571 msm8960_btsco_rate_get, msm8960_btsco_rate_put),
572};
573
574static int msm8960_btsco_init(struct snd_soc_pcm_runtime *rtd)
575{
576 int err = 0;
577 struct snd_soc_platform *platform = rtd->platform;
578
579 err = snd_soc_add_platform_controls(platform,
580 int_btsco_rate_mixer_controls,
581 ARRAY_SIZE(int_btsco_rate_mixer_controls));
582 if (err < 0)
583 return err;
584 return 0;
585}
586
Joonwoo Park0976d012011-12-22 11:48:18 -0800587static void *def_tabla_mbhc_cal(void)
588{
589 void *tabla_cal;
590 struct tabla_mbhc_btn_detect_cfg *btn_cfg;
591 u16 *btn_low, *btn_high;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800592 u8 *n_ready, *n_cic, *gain;
Joonwoo Park0976d012011-12-22 11:48:18 -0800593
594 tabla_cal = kzalloc(TABLA_MBHC_CAL_SIZE(TABLA_MBHC_DEF_BUTTONS,
595 TABLA_MBHC_DEF_RLOADS),
596 GFP_KERNEL);
597 if (!tabla_cal) {
598 pr_err("%s: out of memory\n", __func__);
599 return NULL;
600 }
601
602#define S(X, Y) ((TABLA_MBHC_CAL_GENERAL_PTR(tabla_cal)->X) = (Y))
603 S(t_ldoh, 100);
604 S(t_bg_fast_settle, 100);
605 S(t_shutdown_plug_rem, 255);
606 S(mbhc_nsa, 4);
607 S(mbhc_navg, 4);
608#undef S
609#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_DET_PTR(tabla_cal)->X) = (Y))
610 S(mic_current, TABLA_PID_MIC_5_UA);
611 S(hph_current, TABLA_PID_MIC_5_UA);
612 S(t_mic_pid, 100);
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800613 S(t_ins_complete, 250);
Joonwoo Park0976d012011-12-22 11:48:18 -0800614 S(t_ins_retry, 200);
615#undef S
616#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla_cal)->X) = (Y))
617 S(v_no_mic, 30);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800618 S(v_hs_max, 1550);
Joonwoo Park0976d012011-12-22 11:48:18 -0800619#undef S
620#define S(X, Y) ((TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal)->X) = (Y))
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800621 S(c[0], 62);
622 S(c[1], 124);
623 S(nc, 1);
624 S(n_meas, 3);
Joonwoo Park0976d012011-12-22 11:48:18 -0800625 S(mbhc_nsc, 11);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800626 S(n_btn_meas, 1);
627 S(n_btn_con, 2);
Joonwoo Park0976d012011-12-22 11:48:18 -0800628 S(num_btn, TABLA_MBHC_DEF_BUTTONS);
629 S(v_btn_press_delta_sta, 100);
630 S(v_btn_press_delta_cic, 50);
631#undef S
632 btn_cfg = TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal);
633 btn_low = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_LOW);
634 btn_high = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_HIGH);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800635 btn_low[0] = -50;
636 btn_high[0] = 10;
637 btn_low[1] = 11;
638 btn_high[1] = 38;
639 btn_low[2] = 39;
640 btn_high[2] = 64;
641 btn_low[3] = 65;
642 btn_high[3] = 91;
643 btn_low[4] = 92;
644 btn_high[4] = 115;
645 btn_low[5] = 116;
646 btn_high[5] = 141;
647 btn_low[6] = 142;
648 btn_high[6] = 163;
649 btn_low[7] = 164;
650 btn_high[7] = 250;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800651 n_ready = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_READY);
652 n_ready[0] = 48;
653 n_ready[1] = 38;
Joonwoo Park0976d012011-12-22 11:48:18 -0800654 n_cic = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_CIC);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800655 n_cic[0] = 60;
656 n_cic[1] = 47;
Joonwoo Park0976d012011-12-22 11:48:18 -0800657 gain = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_GAIN);
658 gain[0] = 11;
659 gain[1] = 9;
660
661 return tabla_cal;
662}
663
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800664static int msm8960_hw_params(struct snd_pcm_substream *substream,
665 struct snd_pcm_hw_params *params)
666{
667 struct snd_soc_pcm_runtime *rtd = substream->private_data;
668 struct snd_soc_dai *codec_dai = rtd->codec_dai;
669 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
670 int ret = 0;
671 unsigned int rx_ch[SLIM_MAX_RX_PORTS], tx_ch[SLIM_MAX_TX_PORTS];
672 unsigned int rx_ch_cnt = 0, tx_ch_cnt = 0;
673
674 pr_debug("%s: ch=%d\n", __func__,
675 msm8960_slim_0_rx_ch);
676 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
677 ret = snd_soc_dai_get_channel_map(codec_dai,
678 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
679 if (ret < 0) {
680 pr_err("%s: failed to get codec chan map\n", __func__);
681 goto end;
682 }
683
684 ret = snd_soc_dai_set_channel_map(cpu_dai, 0, 0,
685 msm8960_slim_0_rx_ch, rx_ch);
686 if (ret < 0) {
687 pr_err("%s: failed to set cpu chan map\n", __func__);
688 goto end;
689 }
690 ret = snd_soc_dai_set_channel_map(codec_dai, 0, 0,
691 msm8960_slim_0_rx_ch, rx_ch);
692 if (ret < 0) {
693 pr_err("%s: failed to set codec channel map\n",
694 __func__);
695 goto end;
696 }
697 } else {
698 ret = snd_soc_dai_get_channel_map(codec_dai,
699 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
700 if (ret < 0) {
701 pr_err("%s: failed to get codec chan map\n", __func__);
702 goto end;
703 }
704 ret = snd_soc_dai_set_channel_map(cpu_dai,
705 msm8960_slim_0_tx_ch, tx_ch, 0 , 0);
706 if (ret < 0) {
707 pr_err("%s: failed to set cpu chan map\n", __func__);
708 goto end;
709 }
710 ret = snd_soc_dai_set_channel_map(codec_dai,
711 msm8960_slim_0_tx_ch, tx_ch, 0, 0);
712 if (ret < 0) {
713 pr_err("%s: failed to set codec channel map\n",
714 __func__);
715 goto end;
716 }
717
718
719 }
720end:
721 return ret;
722}
723
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700724static int msm8960_audrx_init(struct snd_soc_pcm_runtime *rtd)
725{
726 int err;
727 struct snd_soc_codec *codec = rtd->codec;
728 struct snd_soc_dapm_context *dapm = &codec->dapm;
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800729 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700730 struct pm_gpio jack_gpio_cfg = {
731 .direction = PM_GPIO_DIR_IN,
732 .pull = PM_GPIO_PULL_UP_1P5,
733 .function = PM_GPIO_FUNC_NORMAL,
734 .vin_sel = 2,
735 .inv_int_pol = 0,
736 };
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700737
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800738 pr_debug("%s(), dev_name%s\n", __func__, dev_name(cpu_dai->dev));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700739
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800740 if (machine_is_msm8960_liquid()) {
741 top_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(19));
742 bottom_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(18));
743 }
744
Kiran Kandi28ef14e2011-09-16 15:45:47 -0700745 rtd->pmdown_time = 0;
746
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700747 err = snd_soc_add_controls(codec, tabla_msm8960_controls,
748 ARRAY_SIZE(tabla_msm8960_controls));
749 if (err < 0)
750 return err;
751
752 snd_soc_dapm_new_controls(dapm, msm8960_dapm_widgets,
753 ARRAY_SIZE(msm8960_dapm_widgets));
754
Bradley Rubin229c6a52011-07-12 16:18:48 -0700755 snd_soc_dapm_add_routes(dapm, common_audio_map,
756 ARRAY_SIZE(common_audio_map));
757
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700758 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
759 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
760 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
761 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700762
763 snd_soc_dapm_sync(dapm);
764
765 err = snd_soc_jack_new(codec, "Headset Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800766 (SND_JACK_HEADSET | SND_JACK_OC_HPHL |
767 SND_JACK_OC_HPHR),
768 &hs_jack);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700769 if (err) {
770 pr_err("failed to create new jack\n");
771 return err;
772 }
Bradley Rubincb1e2732011-06-23 16:49:20 -0700773
774 err = snd_soc_jack_new(codec, "Button Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800775 TABLA_JACK_BUTTON_MASK, &button_jack);
Bradley Rubincb1e2732011-06-23 16:49:20 -0700776 if (err) {
777 pr_err("failed to create new jack\n");
778 return err;
779 }
780
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800781 codec_clk = clk_get(cpu_dai->dev, "osr_clk");
782
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700783 if (GPIO_DETECT_USED) {
784 mbhc_cfg.gpio = PM8921_GPIO_PM_TO_SYS(JACK_DETECT_GPIO);
785 mbhc_cfg.gpio_irq = JACK_DETECT_INT;
786 }
787
788 if (mbhc_cfg.gpio) {
789 err = pm8xxx_gpio_config(mbhc_cfg.gpio, &jack_gpio_cfg);
790 if (err) {
791 pr_err("%s: pm8xxx_gpio_config failed %d\n", __func__,
792 err);
793 return err;
794 }
795 }
796 err = tabla_hs_detect(codec, &mbhc_cfg);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700797
Joonwoo Park03324832012-03-19 19:36:16 -0700798 return err;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700799}
800
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700801static struct snd_soc_dsp_link lpa_fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700802 .playback = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700803 .trigger = {
804 SND_SOC_DSP_TRIGGER_POST,
805 SND_SOC_DSP_TRIGGER_POST
806 },
807};
808
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700809static struct snd_soc_dsp_link fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700810 .playback = true,
811 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700812 .trigger = {
813 SND_SOC_DSP_TRIGGER_POST,
814 SND_SOC_DSP_TRIGGER_POST
815 },
816};
817
Patrick Lai9b56e1d2012-01-22 22:14:15 -0800818/* bi-directional media definition for hostless PCM device */
819static struct snd_soc_dsp_link bidir_hl_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700820 .playback = true,
821 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700822 .trigger = {
823 SND_SOC_DSP_TRIGGER_POST,
824 SND_SOC_DSP_TRIGGER_POST
825 },
826};
827
Alex Wongb3d06a02012-01-12 10:00:41 -0800828static struct snd_soc_dsp_link hdmi_rx_hl = {
829 .playback = true,
830 .trigger = {
831 SND_SOC_DSP_TRIGGER_POST,
832 SND_SOC_DSP_TRIGGER_POST
833 },
834};
835
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700836static int msm8960_slim_0_rx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
837 struct snd_pcm_hw_params *params)
838{
839 struct snd_interval *rate = hw_param_interval(params,
840 SNDRV_PCM_HW_PARAM_RATE);
841
842 struct snd_interval *channels = hw_param_interval(params,
843 SNDRV_PCM_HW_PARAM_CHANNELS);
844
845 pr_debug("%s()\n", __func__);
846 rate->min = rate->max = 48000;
847 channels->min = channels->max = msm8960_slim_0_rx_ch;
848
849 return 0;
850}
851
852static int msm8960_slim_0_tx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
853 struct snd_pcm_hw_params *params)
854{
855 struct snd_interval *rate = hw_param_interval(params,
856 SNDRV_PCM_HW_PARAM_RATE);
857
858 struct snd_interval *channels = hw_param_interval(params,
859 SNDRV_PCM_HW_PARAM_CHANNELS);
860
861 pr_debug("%s()\n", __func__);
862 rate->min = rate->max = 48000;
863 channels->min = channels->max = msm8960_slim_0_tx_ch;
864
865 return 0;
866}
867
868static int msm8960_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
869 struct snd_pcm_hw_params *params)
870{
871 struct snd_interval *rate = hw_param_interval(params,
872 SNDRV_PCM_HW_PARAM_RATE);
873
874 pr_debug("%s()\n", __func__);
875 rate->min = rate->max = 48000;
876
877 return 0;
878}
879
Helen Zeng73f7fc62011-11-18 16:29:59 -0800880static int msm8960_hdmi_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
881 struct snd_pcm_hw_params *params)
882{
883 struct snd_interval *rate = hw_param_interval(params,
884 SNDRV_PCM_HW_PARAM_RATE);
885
886 struct snd_interval *channels = hw_param_interval(params,
887 SNDRV_PCM_HW_PARAM_CHANNELS);
888
Kiran Kandi5e809b02012-01-31 00:24:33 -0800889 pr_debug("%s channels->min %u channels->max %u ()\n", __func__,
890 channels->min, channels->max);
891
Helen Zeng73f7fc62011-11-18 16:29:59 -0800892 rate->min = rate->max = 48000;
Helen Zeng73f7fc62011-11-18 16:29:59 -0800893
894 return 0;
895}
896
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700897static int msm8960_btsco_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
898 struct snd_pcm_hw_params *params)
899{
900 struct snd_interval *rate = hw_param_interval(params,
901 SNDRV_PCM_HW_PARAM_RATE);
902
903 struct snd_interval *channels = hw_param_interval(params,
904 SNDRV_PCM_HW_PARAM_CHANNELS);
905
906 rate->min = rate->max = msm8960_btsco_rate;
907 channels->min = channels->max = msm8960_btsco_ch;
908
909 return 0;
910}
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700911static int msm8960_auxpcm_be_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);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700916
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700917 struct snd_interval *channels = hw_param_interval(params,
918 SNDRV_PCM_HW_PARAM_CHANNELS);
919
920 /* PCM only supports mono output with 8khz sample rate */
921 rate->min = rate->max = 8000;
922 channels->min = channels->max = 1;
923
924 return 0;
925}
926static int msm8960_aux_pcm_get_gpios(void)
927{
928 int ret = 0;
929
930 pr_debug("%s\n", __func__);
931
932 ret = gpio_request(GPIO_AUX_PCM_DOUT, "AUX PCM DOUT");
933 if (ret < 0) {
934 pr_err("%s: Failed to request gpio(%d): AUX PCM DOUT",
935 __func__, GPIO_AUX_PCM_DOUT);
936 goto fail_dout;
937 }
938
939 ret = gpio_request(GPIO_AUX_PCM_DIN, "AUX PCM DIN");
940 if (ret < 0) {
941 pr_err("%s: Failed to request gpio(%d): AUX PCM DIN",
942 __func__, GPIO_AUX_PCM_DIN);
943 goto fail_din;
944 }
945
946 ret = gpio_request(GPIO_AUX_PCM_SYNC, "AUX PCM SYNC");
947 if (ret < 0) {
948 pr_err("%s: Failed to request gpio(%d): AUX PCM SYNC",
949 __func__, GPIO_AUX_PCM_SYNC);
950 goto fail_sync;
951 }
952 ret = gpio_request(GPIO_AUX_PCM_CLK, "AUX PCM CLK");
953 if (ret < 0) {
954 pr_err("%s: Failed to request gpio(%d): AUX PCM CLK",
955 __func__, GPIO_AUX_PCM_CLK);
956 goto fail_clk;
957 }
958
959 return 0;
960
961fail_clk:
962 gpio_free(GPIO_AUX_PCM_SYNC);
963fail_sync:
964 gpio_free(GPIO_AUX_PCM_DIN);
965fail_din:
966 gpio_free(GPIO_AUX_PCM_DOUT);
967fail_dout:
968
969 return ret;
970}
971
972static int msm8960_aux_pcm_free_gpios(void)
973{
974 gpio_free(GPIO_AUX_PCM_DIN);
975 gpio_free(GPIO_AUX_PCM_DOUT);
976 gpio_free(GPIO_AUX_PCM_SYNC);
977 gpio_free(GPIO_AUX_PCM_CLK);
978
979 return 0;
980}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700981static int msm8960_startup(struct snd_pcm_substream *substream)
982{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700983 pr_debug("%s(): substream = %s stream = %d\n", __func__,
984 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700985 return 0;
986}
987
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700988static int msm8960_auxpcm_startup(struct snd_pcm_substream *substream)
989{
990 int ret = 0;
991
992 pr_debug("%s(): substream = %s\n", __func__, substream->name);
993 ret = msm8960_aux_pcm_get_gpios();
994 if (ret < 0) {
995 pr_err("%s: Aux PCM GPIO request failed\n", __func__);
996 return -EINVAL;
997 }
998 return 0;
999}
1000
1001static void msm8960_auxpcm_shutdown(struct snd_pcm_substream *substream)
1002{
1003
1004 pr_debug("%s(): substream = %s\n", __func__, substream->name);
1005 msm8960_aux_pcm_free_gpios();
1006}
1007
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001008static void msm8960_shutdown(struct snd_pcm_substream *substream)
1009{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07001010 pr_debug("%s(): substream = %s stream = %d\n", __func__,
1011 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001012}
1013
1014static struct snd_soc_ops msm8960_be_ops = {
1015 .startup = msm8960_startup,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08001016 .hw_params = msm8960_hw_params,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001017 .shutdown = msm8960_shutdown,
1018};
1019
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001020static struct snd_soc_ops msm8960_auxpcm_be_ops = {
1021 .startup = msm8960_auxpcm_startup,
1022 .shutdown = msm8960_auxpcm_shutdown,
1023};
1024
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001025/* Digital audio interface glue - connects codec <---> CPU */
Kuirong Wangb25838e2012-01-16 23:37:23 -08001026static struct snd_soc_dai_link msm8960_dai_common[] = {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001027 /* FrontEnd DAI Links */
1028 {
1029 .name = "MSM8960 Media1",
1030 .stream_name = "MultiMedia1",
1031 .cpu_dai_name = "MultiMedia1",
1032 .platform_name = "msm-pcm-dsp",
1033 .dynamic = 1,
1034 .dsp_link = &fe_media,
1035 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA1
1036 },
1037 {
1038 .name = "MSM8960 Media2",
1039 .stream_name = "MultiMedia2",
1040 .cpu_dai_name = "MultiMedia2",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001041 .platform_name = "msm-multi-ch-pcm-dsp",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001042 .dynamic = 1,
1043 .dsp_link = &fe_media,
1044 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA2,
1045 },
1046 {
1047 .name = "Circuit-Switch Voice",
1048 .stream_name = "CS-Voice",
1049 .cpu_dai_name = "CS-VOICE",
1050 .platform_name = "msm-pcm-voice",
1051 .dynamic = 1,
1052 .dsp_link = &fe_media,
1053 .be_id = MSM_FRONTEND_DAI_CS_VOICE,
1054 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001055 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001056 },
1057 {
1058 .name = "MSM VoIP",
1059 .stream_name = "VoIP",
1060 .cpu_dai_name = "VoIP",
1061 .platform_name = "msm-voip-dsp",
1062 .dynamic = 1,
1063 .dsp_link = &fe_media,
1064 .be_id = MSM_FRONTEND_DAI_VOIP,
1065 },
1066 {
1067 .name = "MSM8960 LPA",
1068 .stream_name = "LPA",
1069 .cpu_dai_name = "MultiMedia3",
1070 .platform_name = "msm-pcm-lpa",
1071 .dynamic = 1,
1072 .dsp_link = &lpa_fe_media,
1073 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA3,
1074 },
1075 /* Hostless PMC purpose */
1076 {
1077 .name = "SLIMBUS_0 Hostless",
1078 .stream_name = "SLIMBUS_0 Hostless",
1079 .cpu_dai_name = "SLIMBUS0_HOSTLESS",
1080 .platform_name = "msm-pcm-hostless",
1081 .dynamic = 1,
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001082 .dsp_link = &bidir_hl_media,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001083 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001084 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001085 /* .be_id = do not care */
1086 },
1087 {
1088 .name = "INT_FM Hostless",
1089 .stream_name = "INT_FM Hostless",
1090 .cpu_dai_name = "INT_FM_HOSTLESS",
1091 .platform_name = "msm-pcm-hostless",
1092 .dynamic = 1,
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001093 .dsp_link = &bidir_hl_media,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001094 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001095 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001096 /* .be_id = do not care */
1097 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301098 {
1099 .name = "MSM AFE-PCM RX",
1100 .stream_name = "AFE-PROXY RX",
1101 .cpu_dai_name = "msm-dai-q6.241",
1102 .codec_name = "msm-stub-codec.1",
1103 .codec_dai_name = "msm-stub-rx",
1104 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001105 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301106 },
1107 {
1108 .name = "MSM AFE-PCM TX",
1109 .stream_name = "AFE-PROXY TX",
1110 .cpu_dai_name = "msm-dai-q6.240",
1111 .codec_name = "msm-stub-codec.1",
1112 .codec_dai_name = "msm-stub-tx",
1113 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001114 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301115 },
Asish Bhattacharya09f9e0a2011-11-11 13:22:47 +05301116 {
1117 .name = "MSM8960 Compr",
1118 .stream_name = "COMPR",
1119 .cpu_dai_name = "MultiMedia4",
1120 .platform_name = "msm-compr-dsp",
1121 .dynamic = 1,
1122 .dsp_link = &lpa_fe_media,
1123 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA4,
1124 },
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001125 {
1126 .name = "AUXPCM Hostless",
1127 .stream_name = "AUXPCM Hostless",
1128 .cpu_dai_name = "AUXPCM_HOSTLESS",
1129 .platform_name = "msm-pcm-hostless",
1130 .dynamic = 1,
1131 .dsp_link = &bidir_hl_media,
1132 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1133 .ignore_suspend = 1,
1134 },
Alex Wongb3d06a02012-01-12 10:00:41 -08001135 /* HDMI Hostless */
1136 {
1137 .name = "HDMI_RX_HOSTLESS",
1138 .stream_name = "HDMI_RX_HOSTLESS",
1139 .cpu_dai_name = "HDMI_HOSTLESS",
1140 .platform_name = "msm-pcm-hostless",
1141 .dynamic = 1,
1142 .dsp_link = &hdmi_rx_hl,
1143 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1144 .no_codec = 1,
1145 .ignore_suspend = 1,
1146 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001147 /* Backend BT/FM DAI Links */
1148 {
1149 .name = LPASS_BE_INT_BT_SCO_RX,
1150 .stream_name = "Internal BT-SCO Playback",
1151 .cpu_dai_name = "msm-dai-q6.12288",
1152 .platform_name = "msm-pcm-routing",
1153 .codec_name = "msm-stub-codec.1",
1154 .codec_dai_name = "msm-stub-rx",
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001155 .init = &msm8960_btsco_init,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001156 .no_pcm = 1,
1157 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_RX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001158 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001159 },
1160 {
1161 .name = LPASS_BE_INT_BT_SCO_TX,
1162 .stream_name = "Internal BT-SCO Capture",
1163 .cpu_dai_name = "msm-dai-q6.12289",
1164 .platform_name = "msm-pcm-routing",
1165 .codec_name = "msm-stub-codec.1",
1166 .codec_dai_name = "msm-stub-tx",
1167 .no_pcm = 1,
1168 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_TX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001169 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001170 },
1171 {
1172 .name = LPASS_BE_INT_FM_RX,
1173 .stream_name = "Internal FM Playback",
1174 .cpu_dai_name = "msm-dai-q6.12292",
1175 .platform_name = "msm-pcm-routing",
1176 .codec_name = "msm-stub-codec.1",
1177 .codec_dai_name = "msm-stub-rx",
1178 .no_pcm = 1,
1179 .be_id = MSM_BACKEND_DAI_INT_FM_RX,
1180 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1181 },
1182 {
1183 .name = LPASS_BE_INT_FM_TX,
1184 .stream_name = "Internal FM Capture",
1185 .cpu_dai_name = "msm-dai-q6.12293",
1186 .platform_name = "msm-pcm-routing",
1187 .codec_name = "msm-stub-codec.1",
1188 .codec_dai_name = "msm-stub-tx",
1189 .no_pcm = 1,
1190 .be_id = MSM_BACKEND_DAI_INT_FM_TX,
1191 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1192 },
1193 /* HDMI BACK END DAI Link */
1194 {
1195 .name = LPASS_BE_HDMI,
1196 .stream_name = "HDMI Playback",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001197 .cpu_dai_name = "msm-dai-q6-hdmi.8",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001198 .platform_name = "msm-pcm-routing",
1199 .codec_name = "msm-stub-codec.1",
1200 .codec_dai_name = "msm-stub-rx",
1201 .no_pcm = 1,
1202 .no_codec = 1,
1203 .be_id = MSM_BACKEND_DAI_HDMI_RX,
Helen Zeng73f7fc62011-11-18 16:29:59 -08001204 .be_hw_params_fixup = msm8960_hdmi_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001205 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301206 /* Backend AFE DAI Links */
1207 {
1208 .name = LPASS_BE_AFE_PCM_RX,
1209 .stream_name = "AFE Playback",
1210 .cpu_dai_name = "msm-dai-q6.224",
1211 .platform_name = "msm-pcm-routing",
1212 .codec_name = "msm-stub-codec.1",
1213 .codec_dai_name = "msm-stub-rx",
1214 .no_codec = 1,
1215 .no_pcm = 1,
1216 .be_id = MSM_BACKEND_DAI_AFE_PCM_RX,
1217 },
1218 {
1219 .name = LPASS_BE_AFE_PCM_TX,
1220 .stream_name = "AFE Capture",
1221 .cpu_dai_name = "msm-dai-q6.225",
1222 .platform_name = "msm-pcm-routing",
1223 .codec_name = "msm-stub-codec.1",
1224 .codec_dai_name = "msm-stub-tx",
1225 .no_codec = 1,
1226 .no_pcm = 1,
1227 .be_id = MSM_BACKEND_DAI_AFE_PCM_TX,
1228 },
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001229 /* AUX PCM Backend DAI Links */
1230 {
1231 .name = LPASS_BE_AUXPCM_RX,
1232 .stream_name = "AUX PCM Playback",
1233 .cpu_dai_name = "msm-dai-q6.2",
1234 .platform_name = "msm-pcm-routing",
1235 .codec_name = "msm-stub-codec.1",
1236 .codec_dai_name = "msm-stub-rx",
1237 .no_pcm = 1,
1238 .be_id = MSM_BACKEND_DAI_AUXPCM_RX,
1239 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1240 .ops = &msm8960_auxpcm_be_ops,
1241 },
1242 {
1243 .name = LPASS_BE_AUXPCM_TX,
1244 .stream_name = "AUX PCM Capture",
1245 .cpu_dai_name = "msm-dai-q6.3",
1246 .platform_name = "msm-pcm-routing",
1247 .codec_name = "msm-stub-codec.1",
1248 .codec_dai_name = "msm-stub-tx",
1249 .no_pcm = 1,
1250 .be_id = MSM_BACKEND_DAI_AUXPCM_TX,
1251 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1252 },
Helen Zeng0705a5f2011-10-14 15:29:52 -07001253 /* Incall Music BACK END DAI Link */
1254 {
1255 .name = LPASS_BE_VOICE_PLAYBACK_TX,
1256 .stream_name = "Voice Farend Playback",
1257 .cpu_dai_name = "msm-dai-q6.32773",
1258 .platform_name = "msm-pcm-routing",
1259 .codec_name = "msm-stub-codec.1",
1260 .codec_dai_name = "msm-stub-rx",
1261 .no_pcm = 1,
1262 .no_codec = 1,
1263 .be_id = MSM_BACKEND_DAI_VOICE_PLAYBACK_TX,
1264 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1265 },
Helen Zenge3d716a2011-10-14 16:32:16 -07001266 /* Incall Record Uplink BACK END DAI Link */
1267 {
1268 .name = LPASS_BE_INCALL_RECORD_TX,
1269 .stream_name = "Voice Uplink Capture",
1270 .cpu_dai_name = "msm-dai-q6.32772",
1271 .platform_name = "msm-pcm-routing",
1272 .codec_name = "msm-stub-codec.1",
1273 .codec_dai_name = "msm-stub-tx",
1274 .no_pcm = 1,
1275 .no_codec = 1,
1276 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_TX,
1277 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1278 },
1279 /* Incall Record Downlink BACK END DAI Link */
1280 {
1281 .name = LPASS_BE_INCALL_RECORD_RX,
1282 .stream_name = "Voice Downlink Capture",
1283 .cpu_dai_name = "msm-dai-q6.32771",
1284 .platform_name = "msm-pcm-routing",
1285 .codec_name = "msm-stub-codec.1",
1286 .codec_dai_name = "msm-stub-tx",
1287 .no_pcm = 1,
1288 .no_codec = 1,
1289 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_RX,
1290 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1291 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001292};
1293
Kuirong Wangb25838e2012-01-16 23:37:23 -08001294static struct snd_soc_dai_link msm8960_dai_delta_tabla1x[] = {
1295 /* Backend DAI Links */
1296 {
1297 .name = LPASS_BE_SLIMBUS_0_RX,
1298 .stream_name = "Slimbus Playback",
1299 .cpu_dai_name = "msm-dai-q6.16384",
1300 .platform_name = "msm-pcm-routing",
1301 .codec_name = "tabla1x_codec",
1302 .codec_dai_name = "tabla_rx1",
1303 .no_pcm = 1,
1304 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1305 .init = &msm8960_audrx_init,
1306 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1307 .ops = &msm8960_be_ops,
1308 },
1309 {
1310 .name = LPASS_BE_SLIMBUS_0_TX,
1311 .stream_name = "Slimbus Capture",
1312 .cpu_dai_name = "msm-dai-q6.16385",
1313 .platform_name = "msm-pcm-routing",
1314 .codec_name = "tabla1x_codec",
1315 .codec_dai_name = "tabla_tx1",
1316 .no_pcm = 1,
1317 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1318 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1319 .ops = &msm8960_be_ops,
1320 },
1321};
1322
1323
1324static struct snd_soc_dai_link msm8960_dai_delta_tabla2x[] = {
1325 /* Backend DAI Links */
1326 {
1327 .name = LPASS_BE_SLIMBUS_0_RX,
1328 .stream_name = "Slimbus Playback",
1329 .cpu_dai_name = "msm-dai-q6.16384",
1330 .platform_name = "msm-pcm-routing",
1331 .codec_name = "tabla_codec",
1332 .codec_dai_name = "tabla_rx1",
1333 .no_pcm = 1,
1334 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1335 .init = &msm8960_audrx_init,
1336 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1337 .ops = &msm8960_be_ops,
1338 },
1339 {
1340 .name = LPASS_BE_SLIMBUS_0_TX,
1341 .stream_name = "Slimbus Capture",
1342 .cpu_dai_name = "msm-dai-q6.16385",
1343 .platform_name = "msm-pcm-routing",
1344 .codec_name = "tabla_codec",
1345 .codec_dai_name = "tabla_tx1",
1346 .no_pcm = 1,
1347 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1348 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1349 .ops = &msm8960_be_ops,
1350 },
1351};
1352
1353static struct snd_soc_dai_link msm8960_tabla1x_dai[
1354 ARRAY_SIZE(msm8960_dai_common) +
1355 ARRAY_SIZE(msm8960_dai_delta_tabla1x)];
1356
1357
1358static struct snd_soc_dai_link msm8960_dai[
1359 ARRAY_SIZE(msm8960_dai_common) +
1360 ARRAY_SIZE(msm8960_dai_delta_tabla2x)];
1361
1362static struct snd_soc_card snd_soc_tabla1x_card_msm8960 = {
1363 .name = "msm8960-tabla1x-snd-card",
1364 .dai_link = msm8960_tabla1x_dai,
1365 .num_links = ARRAY_SIZE(msm8960_tabla1x_dai),
1366};
1367
1368static struct snd_soc_card snd_soc_card_msm8960 = {
1369 .name = "msm8960-snd-card",
1370 .dai_link = msm8960_dai,
1371 .num_links = ARRAY_SIZE(msm8960_dai),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001372};
1373
1374static struct platform_device *msm8960_snd_device;
Kuirong Wangb25838e2012-01-16 23:37:23 -08001375static struct platform_device *msm8960_snd_tabla1x_device;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001376
1377static int msm8960_configure_headset_mic_gpios(void)
1378{
1379 int ret;
1380 struct pm_gpio param = {
1381 .direction = PM_GPIO_DIR_OUT,
1382 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
1383 .output_value = 1,
1384 .pull = PM_GPIO_PULL_NO,
1385 .vin_sel = PM_GPIO_VIN_S4,
1386 .out_strength = PM_GPIO_STRENGTH_MED,
1387 .function = PM_GPIO_FUNC_NORMAL,
1388 };
1389
1390 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(23), "AV_SWITCH");
1391 if (ret) {
1392 pr_err("%s: Failed to request gpio %d\n", __func__,
1393 PM8921_GPIO_PM_TO_SYS(23));
1394 return ret;
1395 }
1396
1397 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(23), &param);
1398 if (ret)
1399 pr_err("%s: Failed to configure gpio %d\n", __func__,
1400 PM8921_GPIO_PM_TO_SYS(23));
1401 else
1402 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(23), 0);
1403
1404 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(35), "US_EURO_SWITCH");
1405 if (ret) {
1406 pr_err("%s: Failed to request gpio %d\n", __func__,
1407 PM8921_GPIO_PM_TO_SYS(35));
1408 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1409 return ret;
1410 }
1411 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(35), &param);
1412 if (ret)
1413 pr_err("%s: Failed to configure gpio %d\n", __func__,
1414 PM8921_GPIO_PM_TO_SYS(35));
1415 else
Patrick Lai2ae0a002011-09-26 14:51:09 -07001416 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(35), 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001417
1418 return 0;
1419}
1420static void msm8960_free_headset_mic_gpios(void)
1421{
1422 if (msm8960_headset_gpios_configured) {
1423 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1424 gpio_free(PM8921_GPIO_PM_TO_SYS(35));
1425 }
1426}
1427
1428static int __init msm8960_audio_init(void)
1429{
1430 int ret;
1431
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301432 if (!cpu_is_msm8960()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001433 pr_err("%s: Not the right machine type\n", __func__);
1434 return -ENODEV ;
1435 }
Joonwoo Park0976d012011-12-22 11:48:18 -08001436
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001437 mbhc_cfg.calibration = def_tabla_mbhc_cal();
1438 if (!mbhc_cfg.calibration) {
Joonwoo Park0976d012011-12-22 11:48:18 -08001439 pr_err("Calibration data allocation failed\n");
1440 return -ENOMEM;
1441 }
1442
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001443 msm8960_snd_device = platform_device_alloc("soc-audio", 0);
1444 if (!msm8960_snd_device) {
1445 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001446 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001447 return -ENOMEM;
1448 }
1449
Kuirong Wangb25838e2012-01-16 23:37:23 -08001450 memcpy(msm8960_dai, msm8960_dai_common, sizeof(msm8960_dai_common));
1451 memcpy(msm8960_dai + ARRAY_SIZE(msm8960_dai_common),
1452 msm8960_dai_delta_tabla2x, sizeof(msm8960_dai_delta_tabla2x));
1453
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001454 platform_set_drvdata(msm8960_snd_device, &snd_soc_card_msm8960);
1455 ret = platform_device_add(msm8960_snd_device);
1456 if (ret) {
1457 platform_device_put(msm8960_snd_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001458 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001459 return ret;
1460 }
1461
Kuirong Wangb25838e2012-01-16 23:37:23 -08001462 msm8960_snd_tabla1x_device = platform_device_alloc("soc-audio", 1);
1463 if (!msm8960_snd_tabla1x_device) {
1464 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001465 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001466 return -ENOMEM;
1467 }
1468
1469 memcpy(msm8960_tabla1x_dai, msm8960_dai_common,
1470 sizeof(msm8960_dai_common));
1471 memcpy(msm8960_tabla1x_dai + ARRAY_SIZE(msm8960_dai_common),
1472 msm8960_dai_delta_tabla1x, sizeof(msm8960_dai_delta_tabla1x));
1473
1474 platform_set_drvdata(msm8960_snd_tabla1x_device,
1475 &snd_soc_tabla1x_card_msm8960);
1476 ret = platform_device_add(msm8960_snd_tabla1x_device);
1477 if (ret) {
1478 platform_device_put(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001479 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001480 return ret;
1481 }
1482
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001483 if (msm8960_configure_headset_mic_gpios()) {
1484 pr_err("%s Fail to configure headset mic gpios\n", __func__);
1485 msm8960_headset_gpios_configured = 0;
1486 } else
1487 msm8960_headset_gpios_configured = 1;
1488
1489 return ret;
1490
1491}
1492module_init(msm8960_audio_init);
1493
1494static void __exit msm8960_audio_exit(void)
1495{
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301496 if (!cpu_is_msm8960()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001497 pr_err("%s: Not the right machine type\n", __func__);
1498 return ;
1499 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001500 msm8960_free_headset_mic_gpios();
1501 platform_device_unregister(msm8960_snd_device);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001502 platform_device_unregister(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001503 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001504}
1505module_exit(msm8960_audio_exit);
1506
1507MODULE_DESCRIPTION("ALSA SoC MSM8960");
1508MODULE_LICENSE("GPL v2");