blob: 9dfef92e6acf0b0fcf64bb0e98848baff92702f6 [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
Kuirong Wang054e1832012-05-11 15:43:30 -070044#define SAMPLE_RATE_8KHZ 8000
45#define SAMPLE_RATE_16KHZ 16000
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070046
Kiran Kandi462acea2011-08-31 23:53:14 -070047#define BOTTOM_SPK_AMP_POS 0x1
48#define BOTTOM_SPK_AMP_NEG 0x2
49#define TOP_SPK_AMP_POS 0x4
50#define TOP_SPK_AMP_NEG 0x8
51
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -070052#define GPIO_AUX_PCM_DOUT 63
53#define GPIO_AUX_PCM_DIN 64
54#define GPIO_AUX_PCM_SYNC 65
55#define GPIO_AUX_PCM_CLK 66
56
Joonwoo Park0976d012011-12-22 11:48:18 -080057#define TABLA_EXT_CLK_RATE 12288000
58
Joonwoo Park6b9b03f2012-01-23 18:48:54 -080059#define TABLA_MBHC_DEF_BUTTONS 8
Joonwoo Park0976d012011-12-22 11:48:18 -080060#define TABLA_MBHC_DEF_RLOADS 5
61
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070062#define JACK_DETECT_GPIO 38
63#define JACK_DETECT_INT PM8921_GPIO_IRQ(PM8921_IRQ_BASE, JACK_DETECT_GPIO)
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070064
Harmandeep Singh0dd82412011-11-11 09:46:17 -080065static u32 top_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(18);
66static u32 bottom_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(19);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070067static int msm8960_spk_control;
Kiran Kandi462acea2011-08-31 23:53:14 -070068static int msm8960_ext_bottom_spk_pamp;
69static int msm8960_ext_top_spk_pamp;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070070static int msm8960_slim_0_rx_ch = 1;
71static int msm8960_slim_0_tx_ch = 1;
72
Kuirong Wang054e1832012-05-11 15:43:30 -070073static int msm8960_btsco_rate = SAMPLE_RATE_8KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -070074static int msm8960_btsco_ch = 1;
75
Kuirong Wang054e1832012-05-11 15:43:30 -070076static int msm8960_auxpcm_rate = SAMPLE_RATE_8KHZ;
77
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070078static struct clk *codec_clk;
79static int clk_users;
80
81static int msm8960_headset_gpios_configured;
82
83static struct snd_soc_jack hs_jack;
Bradley Rubincb1e2732011-06-23 16:49:20 -070084static struct snd_soc_jack button_jack;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070085
Joonwoo Park9115ade2012-04-18 13:14:10 -070086static bool hs_detect_use_gpio;
87module_param(hs_detect_use_gpio, bool, 0444);
88MODULE_PARM_DESC(hs_detect_use_gpio, "Use GPIO for headset detection");
89
90static bool hs_detect_use_firmware;
91module_param(hs_detect_use_firmware, bool, 0444);
92MODULE_PARM_DESC(hs_detect_use_firmware, "Use firmware for headset detection");
93
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070094static int msm8960_enable_codec_ext_clk(struct snd_soc_codec *codec, int enable,
95 bool dapm);
96
97static struct tabla_mbhc_config mbhc_cfg = {
98 .headset_jack = &hs_jack,
99 .button_jack = &button_jack,
100 .read_fw_bin = false,
101 .calibration = NULL,
102 .micbias = TABLA_MICBIAS2,
103 .mclk_cb_fn = msm8960_enable_codec_ext_clk,
104 .mclk_rate = TABLA_EXT_CLK_RATE,
105 .gpio = 0,
106 .gpio_irq = 0,
107 .gpio_level_insert = 1,
108};
Joonwoo Park0976d012011-12-22 11:48:18 -0800109
Joonwoo Park28f49c82012-03-16 12:29:21 -0700110static struct mutex cdc_mclk_mutex;
111
Kiran Kandi462acea2011-08-31 23:53:14 -0700112static void msm8960_enable_ext_spk_amp_gpio(u32 spk_amp_gpio)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700113{
114 int ret = 0;
115
116 struct pm_gpio param = {
117 .direction = PM_GPIO_DIR_OUT,
118 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
119 .output_value = 1,
120 .pull = PM_GPIO_PULL_NO,
121 .vin_sel = PM_GPIO_VIN_S4,
122 .out_strength = PM_GPIO_STRENGTH_MED,
Kiran Kandi462acea2011-08-31 23:53:14 -0700123 .
124 function = PM_GPIO_FUNC_NORMAL,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700125 };
126
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800127 if (spk_amp_gpio == bottom_spk_pamp_gpio) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700128
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800129 ret = gpio_request(bottom_spk_pamp_gpio, "BOTTOM_SPK_AMP");
Kiran Kandi462acea2011-08-31 23:53:14 -0700130 if (ret) {
131 pr_err("%s: Error requesting BOTTOM SPK AMP GPIO %u\n",
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800132 __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700133 return;
134 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800135 ret = pm8xxx_gpio_config(bottom_spk_pamp_gpio, &param);
Kiran Kandi462acea2011-08-31 23:53:14 -0700136 if (ret)
137 pr_err("%s: Failed to configure Bottom Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800138 " gpio %u\n", __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700139 else {
140 pr_debug("%s: enable Bottom spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800141 gpio_direction_output(bottom_spk_pamp_gpio, 1);
Kiran Kandi462acea2011-08-31 23:53:14 -0700142 }
143
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800144 } else if (spk_amp_gpio == top_spk_pamp_gpio) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700145
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800146 ret = gpio_request(top_spk_pamp_gpio, "TOP_SPK_AMP");
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700147 if (ret) {
148 pr_err("%s: Error requesting GPIO %d\n", __func__,
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800149 top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700150 return;
151 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800152 ret = pm8xxx_gpio_config(top_spk_pamp_gpio, &param);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700153 if (ret)
Kiran Kandi462acea2011-08-31 23:53:14 -0700154 pr_err("%s: Failed to configure Top Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800155 " gpio %u\n", __func__, top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700156 else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700157 pr_debug("%s: enable Top spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800158 gpio_direction_output(top_spk_pamp_gpio, 1);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700159 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700160 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700161 pr_err("%s: ERROR : Invalid External Speaker Ampl GPIO."
162 " gpio = %u\n", __func__, spk_amp_gpio);
163 return;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700164 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700165}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700166
Kiran Kandi462acea2011-08-31 23:53:14 -0700167static void msm8960_ext_spk_power_amp_on(u32 spk)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700168{
Kiran Kandi462acea2011-08-31 23:53:14 -0700169 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
170
171 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
172 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
173
174 pr_debug("%s() External Bottom Speaker Ampl already "
175 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700176 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700177 }
178
179 msm8960_ext_bottom_spk_pamp |= spk;
180
181 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
182 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
183
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800184 msm8960_enable_ext_spk_amp_gpio(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700185 pr_debug("%s: slepping 4 ms after turning on external "
186 " Bottom Speaker Ampl\n", __func__);
187 usleep_range(4000, 4000);
188 }
189
190 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
191
192 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
193 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
194
195 pr_debug("%s() External Top Speaker Ampl already"
196 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700197 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700198 }
199
200 msm8960_ext_top_spk_pamp |= spk;
201
202 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
203 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
204
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800205 msm8960_enable_ext_spk_amp_gpio(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700206 pr_debug("%s: sleeping 4 ms after turning on "
207 " external Top Speaker Ampl\n", __func__);
208 usleep_range(4000, 4000);
209 }
210 } else {
211
212 pr_err("%s: ERROR : Invalid External Speaker Ampl. spk = 0x%08x\n",
213 __func__, spk);
214 return;
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700215 }
Kiran Kandi462acea2011-08-31 23:53:14 -0700216}
217
218static void msm8960_ext_spk_power_amp_off(u32 spk)
219{
220 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
221
222 if (!msm8960_ext_bottom_spk_pamp)
223 return;
224
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800225 gpio_direction_output(bottom_spk_pamp_gpio, 0);
226 gpio_free(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700227 msm8960_ext_bottom_spk_pamp = 0;
228
229 pr_debug("%s: sleeping 4 ms after turning off external Bottom"
230 " Speaker Ampl\n", __func__);
231
232 usleep_range(4000, 4000);
233
234 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
235
236 if (!msm8960_ext_top_spk_pamp)
237 return;
238
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800239 gpio_direction_output(top_spk_pamp_gpio, 0);
240 gpio_free(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700241 msm8960_ext_top_spk_pamp = 0;
242
243 pr_debug("%s: sleeping 4 ms after turning off external Top"
244 " Spkaker Ampl\n", __func__);
245
246 usleep_range(4000, 4000);
247 } else {
248
249 pr_err("%s: ERROR : Invalid Ext Spk Ampl. spk = 0x%08x\n",
250 __func__, spk);
251 return;
252 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700253}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700254
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700255static void msm8960_ext_control(struct snd_soc_codec *codec)
256{
257 struct snd_soc_dapm_context *dapm = &codec->dapm;
258
Jayasena Sangaraboina247d36e2012-04-17 10:16:18 -0700259 mutex_lock(&dapm->codec->mutex);
260
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700261 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700262 if (msm8960_spk_control == MSM8960_SPK_ON) {
263 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
264 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
265 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
266 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
267 } else {
268 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Pos");
269 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Neg");
270 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Pos");
271 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Neg");
272 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700273
274 snd_soc_dapm_sync(dapm);
Jayasena Sangaraboina247d36e2012-04-17 10:16:18 -0700275 mutex_unlock(&dapm->codec->mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700276}
277
278static int msm8960_get_spk(struct snd_kcontrol *kcontrol,
279 struct snd_ctl_elem_value *ucontrol)
280{
281 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
282 ucontrol->value.integer.value[0] = msm8960_spk_control;
283 return 0;
284}
285static int msm8960_set_spk(struct snd_kcontrol *kcontrol,
286 struct snd_ctl_elem_value *ucontrol)
287{
288 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
289
290 pr_debug("%s()\n", __func__);
291 if (msm8960_spk_control == ucontrol->value.integer.value[0])
292 return 0;
293
294 msm8960_spk_control = ucontrol->value.integer.value[0];
295 msm8960_ext_control(codec);
296 return 1;
297}
298static int msm8960_spkramp_event(struct snd_soc_dapm_widget *w,
299 struct snd_kcontrol *k, int event)
300{
301 pr_debug("%s() %x\n", __func__, SND_SOC_DAPM_EVENT_ON(event));
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700302
303 if (SND_SOC_DAPM_EVENT_ON(event)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700304 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
305 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_POS);
306 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
307 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_NEG);
308 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
309 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_POS);
310 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
311 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_NEG);
312 else {
313 pr_err("%s() Invalid Speaker Widget = %s\n",
314 __func__, w->name);
315 return -EINVAL;
316 }
317
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700318 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700319 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
320 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_POS);
321 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
322 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_NEG);
323 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
324 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_POS);
325 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
326 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_NEG);
327 else {
328 pr_err("%s() Invalid Speaker Widget = %s\n",
329 __func__, w->name);
330 return -EINVAL;
331 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700332 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700333 return 0;
334}
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700335
336static int msm8960_enable_codec_ext_clk(struct snd_soc_codec *codec, int enable,
337 bool dapm)
Joonwoo Park0976d012011-12-22 11:48:18 -0800338{
Joonwoo Park28f49c82012-03-16 12:29:21 -0700339 int r = 0;
Joonwoo Park0976d012011-12-22 11:48:18 -0800340 pr_debug("%s: enable = %d\n", __func__, enable);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700341
342 mutex_lock(&cdc_mclk_mutex);
Joonwoo Park0976d012011-12-22 11:48:18 -0800343 if (enable) {
344 clk_users++;
345 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700346 if (clk_users == 1) {
347 if (codec_clk) {
348 clk_set_rate(codec_clk, TABLA_EXT_CLK_RATE);
Asish Bhattacharya4776d962012-04-25 12:08:09 +0530349 clk_prepare_enable(codec_clk);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700350 tabla_mclk_enable(codec, 1, dapm);
351 } else {
352 pr_err("%s: Error setting Tabla MCLK\n",
353 __func__);
354 clk_users--;
355 r = -EINVAL;
356 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800357 }
358 } else {
Joonwoo Park28f49c82012-03-16 12:29:21 -0700359 if (clk_users > 0) {
360 clk_users--;
361 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
362 if (clk_users == 0) {
363 pr_debug("%s: disabling MCLK. clk_users = %d\n",
Joonwoo Park0976d012011-12-22 11:48:18 -0800364 __func__, clk_users);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700365 tabla_mclk_enable(codec, 0, dapm);
Asish Bhattacharya4776d962012-04-25 12:08:09 +0530366 clk_disable_unprepare(codec_clk);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700367 }
368 } else {
369 pr_err("%s: Error releasing Tabla MCLK\n", __func__);
370 r = -EINVAL;
Joonwoo Park0976d012011-12-22 11:48:18 -0800371 }
372 }
Joonwoo Park28f49c82012-03-16 12:29:21 -0700373 mutex_unlock(&cdc_mclk_mutex);
374 return r;
Joonwoo Park0976d012011-12-22 11:48:18 -0800375}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700376
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700377static int msm8960_mclk_event(struct snd_soc_dapm_widget *w,
378 struct snd_kcontrol *kcontrol, int event)
379{
380 pr_debug("%s: event = %d\n", __func__, event);
381
382 switch (event) {
383 case SND_SOC_DAPM_PRE_PMU:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700384 return msm8960_enable_codec_ext_clk(w->codec, 1, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700385 case SND_SOC_DAPM_POST_PMD:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700386 return msm8960_enable_codec_ext_clk(w->codec, 0, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700387 }
388 return 0;
389}
390
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700391static const struct snd_soc_dapm_widget msm8960_dapm_widgets[] = {
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700392
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700393 SND_SOC_DAPM_SUPPLY("MCLK", SND_SOC_NOPM, 0, 0,
394 msm8960_mclk_event, SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
395
Kiran Kandi462acea2011-08-31 23:53:14 -0700396 SND_SOC_DAPM_SPK("Ext Spk Bottom Pos", msm8960_spkramp_event),
397 SND_SOC_DAPM_SPK("Ext Spk Bottom Neg", msm8960_spkramp_event),
398
399 SND_SOC_DAPM_SPK("Ext Spk Top Pos", msm8960_spkramp_event),
400 SND_SOC_DAPM_SPK("Ext Spk Top Neg", msm8960_spkramp_event),
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700401
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700402 SND_SOC_DAPM_MIC("Handset Mic", NULL),
403 SND_SOC_DAPM_MIC("Headset Mic", NULL),
404 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700405 SND_SOC_DAPM_MIC("ANCRight Headset Mic", NULL),
406 SND_SOC_DAPM_MIC("ANCLeft Headset Mic", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700407
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700408 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700409 SND_SOC_DAPM_MIC("Digital Mic2", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700410 SND_SOC_DAPM_MIC("Digital Mic3", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700411 SND_SOC_DAPM_MIC("Digital Mic4", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700412 SND_SOC_DAPM_MIC("Digital Mic5", NULL),
413 SND_SOC_DAPM_MIC("Digital Mic6", NULL),
414
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700415};
416
Bradley Rubin229c6a52011-07-12 16:18:48 -0700417static const struct snd_soc_dapm_route common_audio_map[] = {
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700418
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700419 {"RX_BIAS", NULL, "MCLK"},
420 {"LDO_H", NULL, "MCLK"},
421
422 /* Speaker path */
Kiran Kandi462acea2011-08-31 23:53:14 -0700423 {"Ext Spk Bottom Pos", NULL, "LINEOUT1"},
424 {"Ext Spk Bottom Neg", NULL, "LINEOUT3"},
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700425
Kiran Kandi462acea2011-08-31 23:53:14 -0700426 {"Ext Spk Top Pos", NULL, "LINEOUT2"},
427 {"Ext Spk Top Neg", NULL, "LINEOUT4"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700428
429 /* Microphone path */
Bradley Rubin229c6a52011-07-12 16:18:48 -0700430 {"AMIC1", NULL, "MIC BIAS1 Internal1"},
431 {"MIC BIAS1 Internal1", NULL, "Handset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700432
433 {"AMIC2", NULL, "MIC BIAS2 External"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700434 {"MIC BIAS2 External", NULL, "Headset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700435
Patrick Laie519dd62011-09-28 12:37:11 -0700436 /**
437 * AMIC3 and AMIC4 inputs are connected to ANC microphones
438 * These mics are biased differently on CDP and FLUID
439 * routing entries below are based on bias arrangement
440 * on FLUID.
441 */
442 {"AMIC3", NULL, "MIC BIAS3 Internal1"},
443 {"MIC BIAS3 Internal1", NULL, "ANCRight Headset Mic"},
444
445 {"AMIC4", NULL, "MIC BIAS1 Internal2"},
446 {"MIC BIAS1 Internal2", NULL, "ANCLeft Headset Mic"},
447
Kiran Kandie9bf86a2011-07-21 16:50:41 -0700448 {"HEADPHONE", NULL, "LDO_H"},
449
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700450 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700451 * The digital Mic routes are setup considering
452 * fluid as default device.
453 */
454
455 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700456 * Digital Mic1. Front Bottom left Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700457 * Digital Mic GM5 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700458 * Conncted to DMIC2 Input on Tabla codec.
459 */
460 {"DMIC2", NULL, "MIC BIAS1 External"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700461 {"MIC BIAS1 External", NULL, "Digital Mic1"},
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700462
463 /**
464 * Digital Mic2. Front Bottom right Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700465 * Digital Mic GM6 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700466 * Conncted to DMIC1 Input on Tabla codec.
467 */
468 {"DMIC1", NULL, "MIC BIAS1 External"},
469 {"MIC BIAS1 External", NULL, "Digital Mic2"},
470
471 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700472 * Digital Mic3. Back Bottom Digital Mic on Fluid.
473 * Digital Mic GM1 on CDP mainboard.
474 * Conncted to DMIC4 Input on Tabla codec.
475 */
476 {"DMIC4", NULL, "MIC BIAS3 External"},
477 {"MIC BIAS3 External", NULL, "Digital Mic3"},
478
479 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700480 * Digital Mic4. Back top Digital Mic on Fluid.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700481 * Digital Mic GM2 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700482 * Conncted to DMIC3 Input on Tabla codec.
483 */
484 {"DMIC3", NULL, "MIC BIAS3 External"},
485 {"MIC BIAS3 External", NULL, "Digital Mic4"},
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700486
487 /**
488 * Digital Mic5. Front top Digital Mic on Fluid.
489 * Digital Mic GM3 on CDP mainboard.
490 * Conncted to DMIC5 Input on Tabla codec.
491 */
492 {"DMIC5", NULL, "MIC BIAS4 External"},
493 {"MIC BIAS4 External", NULL, "Digital Mic5"},
494
Patrick Laicb7802b2011-10-04 12:39:18 -0700495 /* Tabla digital Mic6 - back bottom digital Mic on Liquid and
496 * bottom mic on CDP. FLUID/MTP do not have dmic6 installed.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700497 */
498 {"DMIC6", NULL, "MIC BIAS4 External"},
499 {"MIC BIAS4 External", NULL, "Digital Mic6"},
Bradley Rubin229c6a52011-07-12 16:18:48 -0700500};
501
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700502static const char *spk_function[] = {"Off", "On"};
Patrick Lai9f4b4292011-07-16 22:11:09 -0700503static const char *slim0_rx_ch_text[] = {"One", "Two"};
504static const char *slim0_tx_ch_text[] = {"One", "Two", "Three", "Four"};
505
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700506static const struct soc_enum msm8960_enum[] = {
507 SOC_ENUM_SINGLE_EXT(2, spk_function),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700508 SOC_ENUM_SINGLE_EXT(2, slim0_rx_ch_text),
509 SOC_ENUM_SINGLE_EXT(4, slim0_tx_ch_text),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700510};
511
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700512static const char *btsco_rate_text[] = {"8000", "16000"};
513static const struct soc_enum msm8960_btsco_enum[] = {
514 SOC_ENUM_SINGLE_EXT(2, btsco_rate_text),
515};
516
Kuirong Wang054e1832012-05-11 15:43:30 -0700517static const char *auxpcm_rate_text[] = {"rate_8000", "rate_16000"};
518static const struct soc_enum msm8960_auxpcm_enum[] = {
519 SOC_ENUM_SINGLE_EXT(2, auxpcm_rate_text),
520};
521
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700522static int msm8960_slim_0_rx_ch_get(struct snd_kcontrol *kcontrol,
523 struct snd_ctl_elem_value *ucontrol)
524{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700525 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800526 msm8960_slim_0_rx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700527 ucontrol->value.integer.value[0] = msm8960_slim_0_rx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700528 return 0;
529}
530
531static int msm8960_slim_0_rx_ch_put(struct snd_kcontrol *kcontrol,
532 struct snd_ctl_elem_value *ucontrol)
533{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700534 msm8960_slim_0_rx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700535
Patrick Lai9f4b4292011-07-16 22:11:09 -0700536 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800537 msm8960_slim_0_rx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700538 return 1;
539}
540
541static int msm8960_slim_0_tx_ch_get(struct snd_kcontrol *kcontrol,
542 struct snd_ctl_elem_value *ucontrol)
543{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700544 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800545 msm8960_slim_0_tx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700546 ucontrol->value.integer.value[0] = msm8960_slim_0_tx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700547 return 0;
548}
549
550static int msm8960_slim_0_tx_ch_put(struct snd_kcontrol *kcontrol,
551 struct snd_ctl_elem_value *ucontrol)
552{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700553 msm8960_slim_0_tx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700554
555 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800556 msm8960_slim_0_tx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700557 return 1;
558}
559
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700560static int msm8960_btsco_rate_get(struct snd_kcontrol *kcontrol,
561 struct snd_ctl_elem_value *ucontrol)
562{
Joonwoo Park0976d012011-12-22 11:48:18 -0800563 pr_debug("%s: msm8960_btsco_rate = %d", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700564 ucontrol->value.integer.value[0] = msm8960_btsco_rate;
565 return 0;
566}
567
568static int msm8960_btsco_rate_put(struct snd_kcontrol *kcontrol,
569 struct snd_ctl_elem_value *ucontrol)
570{
571 switch (ucontrol->value.integer.value[0]) {
572 case 0:
Kuirong Wang054e1832012-05-11 15:43:30 -0700573 msm8960_btsco_rate = SAMPLE_RATE_8KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700574 break;
575 case 1:
Kuirong Wang054e1832012-05-11 15:43:30 -0700576 msm8960_btsco_rate = SAMPLE_RATE_16KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700577 break;
578 default:
Kuirong Wang054e1832012-05-11 15:43:30 -0700579 msm8960_btsco_rate = SAMPLE_RATE_8KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700580 break;
581 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800582 pr_debug("%s: msm8960_btsco_rate = %d\n", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700583 return 0;
584}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700585
Kuirong Wang054e1832012-05-11 15:43:30 -0700586static int msm8960_auxpcm_rate_get(struct snd_kcontrol *kcontrol,
587 struct snd_ctl_elem_value *ucontrol)
588{
589 pr_debug("%s: msm8960_auxpcm_rate = %d", __func__,
590 msm8960_auxpcm_rate);
591 ucontrol->value.integer.value[0] = msm8960_auxpcm_rate;
592 return 0;
593}
594
595static int msm8960_auxpcm_rate_put(struct snd_kcontrol *kcontrol,
596 struct snd_ctl_elem_value *ucontrol)
597{
598 switch (ucontrol->value.integer.value[0]) {
599 case 0:
600 msm8960_auxpcm_rate = SAMPLE_RATE_8KHZ;
601 break;
602 case 1:
603 msm8960_auxpcm_rate = SAMPLE_RATE_16KHZ;
604 break;
605 default:
606 msm8960_auxpcm_rate = SAMPLE_RATE_8KHZ;
607 break;
608 }
609 pr_debug("%s: msm8960_auxpcm_rate = %d"
610 "ucontrol->value.integer.value[0] = %d\n", __func__,
611 msm8960_auxpcm_rate,
612 (int)ucontrol->value.integer.value[0]);
613 return 0;
614}
615
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700616static const struct snd_kcontrol_new tabla_msm8960_controls[] = {
617 SOC_ENUM_EXT("Speaker Function", msm8960_enum[0], msm8960_get_spk,
618 msm8960_set_spk),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700619 SOC_ENUM_EXT("SLIM_0_RX Channels", msm8960_enum[1],
620 msm8960_slim_0_rx_ch_get, msm8960_slim_0_rx_ch_put),
621 SOC_ENUM_EXT("SLIM_0_TX Channels", msm8960_enum[2],
622 msm8960_slim_0_tx_ch_get, msm8960_slim_0_tx_ch_put),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700623};
624
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700625static const struct snd_kcontrol_new int_btsco_rate_mixer_controls[] = {
626 SOC_ENUM_EXT("Internal BTSCO SampleRate", msm8960_btsco_enum[0],
627 msm8960_btsco_rate_get, msm8960_btsco_rate_put),
628};
629
Kuirong Wang054e1832012-05-11 15:43:30 -0700630static const struct snd_kcontrol_new auxpcm_rate_mixer_controls[] = {
631 SOC_ENUM_EXT("AUX PCM SampleRate", msm8960_auxpcm_enum[0],
632 msm8960_auxpcm_rate_get, msm8960_auxpcm_rate_put),
633};
634
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700635static int msm8960_btsco_init(struct snd_soc_pcm_runtime *rtd)
636{
637 int err = 0;
638 struct snd_soc_platform *platform = rtd->platform;
639
640 err = snd_soc_add_platform_controls(platform,
641 int_btsco_rate_mixer_controls,
642 ARRAY_SIZE(int_btsco_rate_mixer_controls));
643 if (err < 0)
644 return err;
645 return 0;
646}
647
Kuirong Wang054e1832012-05-11 15:43:30 -0700648static int msm8960_auxpcm_init(struct snd_soc_pcm_runtime *rtd)
649{
650 int err = 0;
651 struct snd_soc_platform *platform = rtd->platform;
652
653 err = snd_soc_add_platform_controls(platform,
654 auxpcm_rate_mixer_controls,
655 ARRAY_SIZE(auxpcm_rate_mixer_controls));
656 if (err < 0)
657 return err;
658 return 0;
659}
660
Joonwoo Park0976d012011-12-22 11:48:18 -0800661static void *def_tabla_mbhc_cal(void)
662{
663 void *tabla_cal;
664 struct tabla_mbhc_btn_detect_cfg *btn_cfg;
665 u16 *btn_low, *btn_high;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800666 u8 *n_ready, *n_cic, *gain;
Joonwoo Park0976d012011-12-22 11:48:18 -0800667
668 tabla_cal = kzalloc(TABLA_MBHC_CAL_SIZE(TABLA_MBHC_DEF_BUTTONS,
669 TABLA_MBHC_DEF_RLOADS),
670 GFP_KERNEL);
671 if (!tabla_cal) {
672 pr_err("%s: out of memory\n", __func__);
673 return NULL;
674 }
675
676#define S(X, Y) ((TABLA_MBHC_CAL_GENERAL_PTR(tabla_cal)->X) = (Y))
677 S(t_ldoh, 100);
678 S(t_bg_fast_settle, 100);
679 S(t_shutdown_plug_rem, 255);
680 S(mbhc_nsa, 4);
681 S(mbhc_navg, 4);
682#undef S
683#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_DET_PTR(tabla_cal)->X) = (Y))
684 S(mic_current, TABLA_PID_MIC_5_UA);
685 S(hph_current, TABLA_PID_MIC_5_UA);
686 S(t_mic_pid, 100);
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800687 S(t_ins_complete, 250);
Joonwoo Park0976d012011-12-22 11:48:18 -0800688 S(t_ins_retry, 200);
689#undef S
690#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla_cal)->X) = (Y))
691 S(v_no_mic, 30);
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700692 S(v_hs_max, 2400);
Joonwoo Park0976d012011-12-22 11:48:18 -0800693#undef S
694#define S(X, Y) ((TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal)->X) = (Y))
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800695 S(c[0], 62);
696 S(c[1], 124);
697 S(nc, 1);
698 S(n_meas, 3);
Joonwoo Park0976d012011-12-22 11:48:18 -0800699 S(mbhc_nsc, 11);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800700 S(n_btn_meas, 1);
701 S(n_btn_con, 2);
Joonwoo Park0976d012011-12-22 11:48:18 -0800702 S(num_btn, TABLA_MBHC_DEF_BUTTONS);
703 S(v_btn_press_delta_sta, 100);
704 S(v_btn_press_delta_cic, 50);
705#undef S
706 btn_cfg = TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal);
707 btn_low = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_LOW);
708 btn_high = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_HIGH);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800709 btn_low[0] = -50;
Joonwoo Parkfee17432012-04-16 16:33:55 -0700710 btn_high[0] = 20;
711 btn_low[1] = 21;
712 btn_high[1] = 62;
713 btn_low[2] = 63;
714 btn_high[2] = 104;
715 btn_low[3] = 105;
716 btn_high[3] = 143;
717 btn_low[4] = 144;
718 btn_high[4] = 181;
719 btn_low[5] = 182;
720 btn_high[5] = 218;
721 btn_low[6] = 219;
722 btn_high[6] = 254;
723 btn_low[7] = 255;
724 btn_high[7] = 330;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800725 n_ready = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_READY);
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700726 n_ready[0] = 80;
727 n_ready[1] = 68;
Joonwoo Park0976d012011-12-22 11:48:18 -0800728 n_cic = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_CIC);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800729 n_cic[0] = 60;
730 n_cic[1] = 47;
Joonwoo Park0976d012011-12-22 11:48:18 -0800731 gain = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_GAIN);
732 gain[0] = 11;
733 gain[1] = 9;
734
735 return tabla_cal;
736}
737
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800738static int msm8960_hw_params(struct snd_pcm_substream *substream,
739 struct snd_pcm_hw_params *params)
740{
741 struct snd_soc_pcm_runtime *rtd = substream->private_data;
742 struct snd_soc_dai *codec_dai = rtd->codec_dai;
743 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
744 int ret = 0;
745 unsigned int rx_ch[SLIM_MAX_RX_PORTS], tx_ch[SLIM_MAX_TX_PORTS];
746 unsigned int rx_ch_cnt = 0, tx_ch_cnt = 0;
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700747 unsigned int user_set_tx_ch = 0;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800748
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700749
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800750 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700751
752 pr_debug("%s: rx_0_ch=%d\n", __func__, msm8960_slim_0_rx_ch);
753
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800754 ret = snd_soc_dai_get_channel_map(codec_dai,
755 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
756 if (ret < 0) {
757 pr_err("%s: failed to get codec chan map\n", __func__);
758 goto end;
759 }
760
761 ret = snd_soc_dai_set_channel_map(cpu_dai, 0, 0,
762 msm8960_slim_0_rx_ch, rx_ch);
763 if (ret < 0) {
764 pr_err("%s: failed to set cpu chan map\n", __func__);
765 goto end;
766 }
767 ret = snd_soc_dai_set_channel_map(codec_dai, 0, 0,
768 msm8960_slim_0_rx_ch, rx_ch);
769 if (ret < 0) {
770 pr_err("%s: failed to set codec channel map\n",
771 __func__);
772 goto end;
773 }
774 } else {
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700775
776 if (codec_dai->id == 2)
777 user_set_tx_ch = msm8960_slim_0_tx_ch;
778 else if (codec_dai->id == 4)
779 user_set_tx_ch = params_channels(params);
780
781 pr_debug("%s: %s_tx_dai_id_%d_ch=%d\n", __func__,
782 codec_dai->name, codec_dai->id, user_set_tx_ch);
783
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 Kandi1e6371d2012-03-29 11:48:57 -0700791 user_set_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 Kandi1e6371d2012-03-29 11:48:57 -0700797 user_set_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 }
803
804
805 }
806end:
807 return ret;
808}
809
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700810static int msm8960_audrx_init(struct snd_soc_pcm_runtime *rtd)
811{
812 int err;
813 struct snd_soc_codec *codec = rtd->codec;
814 struct snd_soc_dapm_context *dapm = &codec->dapm;
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800815 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700816 struct pm_gpio jack_gpio_cfg = {
817 .direction = PM_GPIO_DIR_IN,
818 .pull = PM_GPIO_PULL_UP_1P5,
819 .function = PM_GPIO_FUNC_NORMAL,
820 .vin_sel = 2,
821 .inv_int_pol = 0,
822 };
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700823
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800824 pr_debug("%s(), dev_name%s\n", __func__, dev_name(cpu_dai->dev));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700825
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800826 if (machine_is_msm8960_liquid()) {
827 top_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(19));
828 bottom_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(18));
829 }
830
Kiran Kandi28ef14e2011-09-16 15:45:47 -0700831 rtd->pmdown_time = 0;
832
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700833 err = snd_soc_add_controls(codec, tabla_msm8960_controls,
834 ARRAY_SIZE(tabla_msm8960_controls));
835 if (err < 0)
836 return err;
837
838 snd_soc_dapm_new_controls(dapm, msm8960_dapm_widgets,
839 ARRAY_SIZE(msm8960_dapm_widgets));
840
Bradley Rubin229c6a52011-07-12 16:18:48 -0700841 snd_soc_dapm_add_routes(dapm, common_audio_map,
842 ARRAY_SIZE(common_audio_map));
843
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700844 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
845 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
846 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
847 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700848
849 snd_soc_dapm_sync(dapm);
850
851 err = snd_soc_jack_new(codec, "Headset Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800852 (SND_JACK_HEADSET | SND_JACK_OC_HPHL |
853 SND_JACK_OC_HPHR),
854 &hs_jack);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700855 if (err) {
856 pr_err("failed to create new jack\n");
857 return err;
858 }
Bradley Rubincb1e2732011-06-23 16:49:20 -0700859
860 err = snd_soc_jack_new(codec, "Button Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800861 TABLA_JACK_BUTTON_MASK, &button_jack);
Bradley Rubincb1e2732011-06-23 16:49:20 -0700862 if (err) {
863 pr_err("failed to create new jack\n");
864 return err;
865 }
866
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800867 codec_clk = clk_get(cpu_dai->dev, "osr_clk");
868
Joonwoo Park9115ade2012-04-18 13:14:10 -0700869 if (hs_detect_use_gpio) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700870 mbhc_cfg.gpio = PM8921_GPIO_PM_TO_SYS(JACK_DETECT_GPIO);
871 mbhc_cfg.gpio_irq = JACK_DETECT_INT;
872 }
873
874 if (mbhc_cfg.gpio) {
875 err = pm8xxx_gpio_config(mbhc_cfg.gpio, &jack_gpio_cfg);
876 if (err) {
877 pr_err("%s: pm8xxx_gpio_config failed %d\n", __func__,
878 err);
879 return err;
880 }
881 }
Joonwoo Park9115ade2012-04-18 13:14:10 -0700882
883 mbhc_cfg.read_fw_bin = hs_detect_use_firmware;
884
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700885 err = tabla_hs_detect(codec, &mbhc_cfg);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700886
Joonwoo Park03324832012-03-19 19:36:16 -0700887 return err;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700888}
889
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700890static struct snd_soc_dsp_link lpa_fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700891 .playback = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700892 .trigger = {
893 SND_SOC_DSP_TRIGGER_POST,
894 SND_SOC_DSP_TRIGGER_POST
895 },
896};
897
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700898static struct snd_soc_dsp_link fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700899 .playback = true,
900 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700901 .trigger = {
902 SND_SOC_DSP_TRIGGER_POST,
903 SND_SOC_DSP_TRIGGER_POST
904 },
905};
906
Patrick Lai9b56e1d2012-01-22 22:14:15 -0800907/* bi-directional media definition for hostless PCM device */
908static struct snd_soc_dsp_link bidir_hl_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700909 .playback = true,
910 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700911 .trigger = {
912 SND_SOC_DSP_TRIGGER_POST,
913 SND_SOC_DSP_TRIGGER_POST
914 },
915};
916
Alex Wongb3d06a02012-01-12 10:00:41 -0800917static struct snd_soc_dsp_link hdmi_rx_hl = {
918 .playback = true,
919 .trigger = {
920 SND_SOC_DSP_TRIGGER_POST,
921 SND_SOC_DSP_TRIGGER_POST
922 },
923};
924
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700925static int msm8960_slim_0_rx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
926 struct snd_pcm_hw_params *params)
927{
928 struct snd_interval *rate = hw_param_interval(params,
929 SNDRV_PCM_HW_PARAM_RATE);
930
931 struct snd_interval *channels = hw_param_interval(params,
932 SNDRV_PCM_HW_PARAM_CHANNELS);
933
934 pr_debug("%s()\n", __func__);
935 rate->min = rate->max = 48000;
936 channels->min = channels->max = msm8960_slim_0_rx_ch;
937
938 return 0;
939}
940
941static int msm8960_slim_0_tx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
942 struct snd_pcm_hw_params *params)
943{
944 struct snd_interval *rate = hw_param_interval(params,
945 SNDRV_PCM_HW_PARAM_RATE);
946
947 struct snd_interval *channels = hw_param_interval(params,
948 SNDRV_PCM_HW_PARAM_CHANNELS);
949
950 pr_debug("%s()\n", __func__);
951 rate->min = rate->max = 48000;
952 channels->min = channels->max = msm8960_slim_0_tx_ch;
953
954 return 0;
955}
956
957static int msm8960_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
958 struct snd_pcm_hw_params *params)
959{
960 struct snd_interval *rate = hw_param_interval(params,
961 SNDRV_PCM_HW_PARAM_RATE);
962
963 pr_debug("%s()\n", __func__);
964 rate->min = rate->max = 48000;
965
966 return 0;
967}
968
Helen Zeng73f7fc62011-11-18 16:29:59 -0800969static int msm8960_hdmi_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
970 struct snd_pcm_hw_params *params)
971{
972 struct snd_interval *rate = hw_param_interval(params,
973 SNDRV_PCM_HW_PARAM_RATE);
974
975 struct snd_interval *channels = hw_param_interval(params,
976 SNDRV_PCM_HW_PARAM_CHANNELS);
977
Kiran Kandi5e809b02012-01-31 00:24:33 -0800978 pr_debug("%s channels->min %u channels->max %u ()\n", __func__,
979 channels->min, channels->max);
980
Helen Zeng73f7fc62011-11-18 16:29:59 -0800981 rate->min = rate->max = 48000;
Helen Zeng73f7fc62011-11-18 16:29:59 -0800982
983 return 0;
984}
985
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700986static int msm8960_btsco_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 struct snd_interval *channels = hw_param_interval(params,
993 SNDRV_PCM_HW_PARAM_CHANNELS);
994
995 rate->min = rate->max = msm8960_btsco_rate;
996 channels->min = channels->max = msm8960_btsco_ch;
997
998 return 0;
999}
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001000static int msm8960_auxpcm_be_params_fixup(struct snd_soc_pcm_runtime *rtd,
1001 struct snd_pcm_hw_params *params)
1002{
1003 struct snd_interval *rate = hw_param_interval(params,
1004 SNDRV_PCM_HW_PARAM_RATE);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001005
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001006 struct snd_interval *channels = hw_param_interval(params,
1007 SNDRV_PCM_HW_PARAM_CHANNELS);
1008
Kuirong Wang054e1832012-05-11 15:43:30 -07001009 rate->min = rate->max = msm8960_auxpcm_rate;
1010 /* PCM only supports mono output */
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001011 channels->min = channels->max = 1;
1012
1013 return 0;
1014}
1015static int msm8960_aux_pcm_get_gpios(void)
1016{
1017 int ret = 0;
1018
1019 pr_debug("%s\n", __func__);
1020
1021 ret = gpio_request(GPIO_AUX_PCM_DOUT, "AUX PCM DOUT");
1022 if (ret < 0) {
1023 pr_err("%s: Failed to request gpio(%d): AUX PCM DOUT",
1024 __func__, GPIO_AUX_PCM_DOUT);
1025 goto fail_dout;
1026 }
1027
1028 ret = gpio_request(GPIO_AUX_PCM_DIN, "AUX PCM DIN");
1029 if (ret < 0) {
1030 pr_err("%s: Failed to request gpio(%d): AUX PCM DIN",
1031 __func__, GPIO_AUX_PCM_DIN);
1032 goto fail_din;
1033 }
1034
1035 ret = gpio_request(GPIO_AUX_PCM_SYNC, "AUX PCM SYNC");
1036 if (ret < 0) {
1037 pr_err("%s: Failed to request gpio(%d): AUX PCM SYNC",
1038 __func__, GPIO_AUX_PCM_SYNC);
1039 goto fail_sync;
1040 }
1041 ret = gpio_request(GPIO_AUX_PCM_CLK, "AUX PCM CLK");
1042 if (ret < 0) {
1043 pr_err("%s: Failed to request gpio(%d): AUX PCM CLK",
1044 __func__, GPIO_AUX_PCM_CLK);
1045 goto fail_clk;
1046 }
1047
1048 return 0;
1049
1050fail_clk:
1051 gpio_free(GPIO_AUX_PCM_SYNC);
1052fail_sync:
1053 gpio_free(GPIO_AUX_PCM_DIN);
1054fail_din:
1055 gpio_free(GPIO_AUX_PCM_DOUT);
1056fail_dout:
1057
1058 return ret;
1059}
1060
1061static int msm8960_aux_pcm_free_gpios(void)
1062{
1063 gpio_free(GPIO_AUX_PCM_DIN);
1064 gpio_free(GPIO_AUX_PCM_DOUT);
1065 gpio_free(GPIO_AUX_PCM_SYNC);
1066 gpio_free(GPIO_AUX_PCM_CLK);
1067
1068 return 0;
1069}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001070static int msm8960_startup(struct snd_pcm_substream *substream)
1071{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07001072 pr_debug("%s(): substream = %s stream = %d\n", __func__,
1073 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001074 return 0;
1075}
1076
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001077static int msm8960_auxpcm_startup(struct snd_pcm_substream *substream)
1078{
1079 int ret = 0;
1080
1081 pr_debug("%s(): substream = %s\n", __func__, substream->name);
1082 ret = msm8960_aux_pcm_get_gpios();
1083 if (ret < 0) {
1084 pr_err("%s: Aux PCM GPIO request failed\n", __func__);
1085 return -EINVAL;
1086 }
1087 return 0;
1088}
1089
1090static void msm8960_auxpcm_shutdown(struct snd_pcm_substream *substream)
1091{
1092
1093 pr_debug("%s(): substream = %s\n", __func__, substream->name);
1094 msm8960_aux_pcm_free_gpios();
1095}
1096
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001097static void msm8960_shutdown(struct snd_pcm_substream *substream)
1098{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07001099 pr_debug("%s(): substream = %s stream = %d\n", __func__,
1100 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001101}
1102
1103static struct snd_soc_ops msm8960_be_ops = {
1104 .startup = msm8960_startup,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08001105 .hw_params = msm8960_hw_params,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001106 .shutdown = msm8960_shutdown,
1107};
1108
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001109static struct snd_soc_ops msm8960_auxpcm_be_ops = {
1110 .startup = msm8960_auxpcm_startup,
1111 .shutdown = msm8960_auxpcm_shutdown,
1112};
1113
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001114/* Digital audio interface glue - connects codec <---> CPU */
Kuirong Wangb25838e2012-01-16 23:37:23 -08001115static struct snd_soc_dai_link msm8960_dai_common[] = {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001116 /* FrontEnd DAI Links */
1117 {
1118 .name = "MSM8960 Media1",
1119 .stream_name = "MultiMedia1",
1120 .cpu_dai_name = "MultiMedia1",
1121 .platform_name = "msm-pcm-dsp",
1122 .dynamic = 1,
1123 .dsp_link = &fe_media,
1124 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA1
1125 },
1126 {
1127 .name = "MSM8960 Media2",
1128 .stream_name = "MultiMedia2",
1129 .cpu_dai_name = "MultiMedia2",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001130 .platform_name = "msm-multi-ch-pcm-dsp",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001131 .dynamic = 1,
1132 .dsp_link = &fe_media,
1133 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA2,
1134 },
1135 {
1136 .name = "Circuit-Switch Voice",
1137 .stream_name = "CS-Voice",
1138 .cpu_dai_name = "CS-VOICE",
1139 .platform_name = "msm-pcm-voice",
1140 .dynamic = 1,
1141 .dsp_link = &fe_media,
1142 .be_id = MSM_FRONTEND_DAI_CS_VOICE,
1143 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001144 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001145 },
1146 {
1147 .name = "MSM VoIP",
1148 .stream_name = "VoIP",
1149 .cpu_dai_name = "VoIP",
1150 .platform_name = "msm-voip-dsp",
1151 .dynamic = 1,
1152 .dsp_link = &fe_media,
1153 .be_id = MSM_FRONTEND_DAI_VOIP,
1154 },
1155 {
1156 .name = "MSM8960 LPA",
1157 .stream_name = "LPA",
1158 .cpu_dai_name = "MultiMedia3",
1159 .platform_name = "msm-pcm-lpa",
1160 .dynamic = 1,
1161 .dsp_link = &lpa_fe_media,
1162 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA3,
1163 },
1164 /* Hostless PMC purpose */
1165 {
1166 .name = "SLIMBUS_0 Hostless",
1167 .stream_name = "SLIMBUS_0 Hostless",
1168 .cpu_dai_name = "SLIMBUS0_HOSTLESS",
1169 .platform_name = "msm-pcm-hostless",
1170 .dynamic = 1,
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001171 .dsp_link = &bidir_hl_media,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001172 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001173 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001174 /* .be_id = do not care */
1175 },
1176 {
1177 .name = "INT_FM Hostless",
1178 .stream_name = "INT_FM Hostless",
1179 .cpu_dai_name = "INT_FM_HOSTLESS",
1180 .platform_name = "msm-pcm-hostless",
1181 .dynamic = 1,
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001182 .dsp_link = &bidir_hl_media,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001183 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001184 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001185 /* .be_id = do not care */
1186 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301187 {
1188 .name = "MSM AFE-PCM RX",
1189 .stream_name = "AFE-PROXY RX",
1190 .cpu_dai_name = "msm-dai-q6.241",
1191 .codec_name = "msm-stub-codec.1",
1192 .codec_dai_name = "msm-stub-rx",
1193 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001194 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301195 },
1196 {
1197 .name = "MSM AFE-PCM TX",
1198 .stream_name = "AFE-PROXY TX",
1199 .cpu_dai_name = "msm-dai-q6.240",
1200 .codec_name = "msm-stub-codec.1",
1201 .codec_dai_name = "msm-stub-tx",
1202 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001203 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301204 },
Asish Bhattacharya09f9e0a2011-11-11 13:22:47 +05301205 {
1206 .name = "MSM8960 Compr",
1207 .stream_name = "COMPR",
1208 .cpu_dai_name = "MultiMedia4",
1209 .platform_name = "msm-compr-dsp",
1210 .dynamic = 1,
1211 .dsp_link = &lpa_fe_media,
1212 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA4,
1213 },
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001214 {
1215 .name = "AUXPCM Hostless",
1216 .stream_name = "AUXPCM Hostless",
1217 .cpu_dai_name = "AUXPCM_HOSTLESS",
1218 .platform_name = "msm-pcm-hostless",
1219 .dynamic = 1,
1220 .dsp_link = &bidir_hl_media,
1221 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1222 .ignore_suspend = 1,
1223 },
Alex Wongb3d06a02012-01-12 10:00:41 -08001224 /* HDMI Hostless */
1225 {
1226 .name = "HDMI_RX_HOSTLESS",
1227 .stream_name = "HDMI_RX_HOSTLESS",
1228 .cpu_dai_name = "HDMI_HOSTLESS",
1229 .platform_name = "msm-pcm-hostless",
1230 .dynamic = 1,
1231 .dsp_link = &hdmi_rx_hl,
1232 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1233 .no_codec = 1,
1234 .ignore_suspend = 1,
1235 },
Venkat Sudhir1c79c3b2012-04-09 23:42:28 -07001236 {
1237 .name = "VoLTE",
1238 .stream_name = "VoLTE",
1239 .cpu_dai_name = "VoLTE",
1240 .platform_name = "msm-pcm-voice",
1241 .dynamic = 1,
1242 .dsp_link = &fe_media,
1243 .be_id = MSM_FRONTEND_DAI_VOLTE,
1244 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1245 .ignore_suspend = 1,
1246 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001247 /* Backend BT/FM DAI Links */
1248 {
1249 .name = LPASS_BE_INT_BT_SCO_RX,
1250 .stream_name = "Internal BT-SCO Playback",
1251 .cpu_dai_name = "msm-dai-q6.12288",
1252 .platform_name = "msm-pcm-routing",
1253 .codec_name = "msm-stub-codec.1",
1254 .codec_dai_name = "msm-stub-rx",
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001255 .init = &msm8960_btsco_init,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001256 .no_pcm = 1,
1257 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_RX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001258 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001259 },
1260 {
1261 .name = LPASS_BE_INT_BT_SCO_TX,
1262 .stream_name = "Internal BT-SCO Capture",
1263 .cpu_dai_name = "msm-dai-q6.12289",
1264 .platform_name = "msm-pcm-routing",
1265 .codec_name = "msm-stub-codec.1",
1266 .codec_dai_name = "msm-stub-tx",
1267 .no_pcm = 1,
1268 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_TX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001269 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001270 },
1271 {
1272 .name = LPASS_BE_INT_FM_RX,
1273 .stream_name = "Internal FM Playback",
1274 .cpu_dai_name = "msm-dai-q6.12292",
1275 .platform_name = "msm-pcm-routing",
1276 .codec_name = "msm-stub-codec.1",
1277 .codec_dai_name = "msm-stub-rx",
1278 .no_pcm = 1,
1279 .be_id = MSM_BACKEND_DAI_INT_FM_RX,
1280 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1281 },
1282 {
1283 .name = LPASS_BE_INT_FM_TX,
1284 .stream_name = "Internal FM Capture",
1285 .cpu_dai_name = "msm-dai-q6.12293",
1286 .platform_name = "msm-pcm-routing",
1287 .codec_name = "msm-stub-codec.1",
1288 .codec_dai_name = "msm-stub-tx",
1289 .no_pcm = 1,
1290 .be_id = MSM_BACKEND_DAI_INT_FM_TX,
1291 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1292 },
1293 /* HDMI BACK END DAI Link */
1294 {
1295 .name = LPASS_BE_HDMI,
1296 .stream_name = "HDMI Playback",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001297 .cpu_dai_name = "msm-dai-q6-hdmi.8",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001298 .platform_name = "msm-pcm-routing",
1299 .codec_name = "msm-stub-codec.1",
1300 .codec_dai_name = "msm-stub-rx",
1301 .no_pcm = 1,
1302 .no_codec = 1,
1303 .be_id = MSM_BACKEND_DAI_HDMI_RX,
Helen Zeng73f7fc62011-11-18 16:29:59 -08001304 .be_hw_params_fixup = msm8960_hdmi_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001305 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301306 /* Backend AFE DAI Links */
1307 {
1308 .name = LPASS_BE_AFE_PCM_RX,
1309 .stream_name = "AFE Playback",
1310 .cpu_dai_name = "msm-dai-q6.224",
1311 .platform_name = "msm-pcm-routing",
1312 .codec_name = "msm-stub-codec.1",
1313 .codec_dai_name = "msm-stub-rx",
1314 .no_codec = 1,
1315 .no_pcm = 1,
1316 .be_id = MSM_BACKEND_DAI_AFE_PCM_RX,
1317 },
1318 {
1319 .name = LPASS_BE_AFE_PCM_TX,
1320 .stream_name = "AFE Capture",
1321 .cpu_dai_name = "msm-dai-q6.225",
1322 .platform_name = "msm-pcm-routing",
1323 .codec_name = "msm-stub-codec.1",
1324 .codec_dai_name = "msm-stub-tx",
1325 .no_codec = 1,
1326 .no_pcm = 1,
1327 .be_id = MSM_BACKEND_DAI_AFE_PCM_TX,
1328 },
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001329 /* AUX PCM Backend DAI Links */
1330 {
1331 .name = LPASS_BE_AUXPCM_RX,
1332 .stream_name = "AUX PCM Playback",
1333 .cpu_dai_name = "msm-dai-q6.2",
1334 .platform_name = "msm-pcm-routing",
1335 .codec_name = "msm-stub-codec.1",
1336 .codec_dai_name = "msm-stub-rx",
Kuirong Wang054e1832012-05-11 15:43:30 -07001337 .init = &msm8960_auxpcm_init,
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001338 .no_pcm = 1,
1339 .be_id = MSM_BACKEND_DAI_AUXPCM_RX,
1340 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1341 .ops = &msm8960_auxpcm_be_ops,
1342 },
1343 {
1344 .name = LPASS_BE_AUXPCM_TX,
1345 .stream_name = "AUX PCM Capture",
1346 .cpu_dai_name = "msm-dai-q6.3",
1347 .platform_name = "msm-pcm-routing",
1348 .codec_name = "msm-stub-codec.1",
1349 .codec_dai_name = "msm-stub-tx",
1350 .no_pcm = 1,
1351 .be_id = MSM_BACKEND_DAI_AUXPCM_TX,
1352 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1353 },
Helen Zeng0705a5f2011-10-14 15:29:52 -07001354 /* Incall Music BACK END DAI Link */
1355 {
1356 .name = LPASS_BE_VOICE_PLAYBACK_TX,
1357 .stream_name = "Voice Farend Playback",
1358 .cpu_dai_name = "msm-dai-q6.32773",
1359 .platform_name = "msm-pcm-routing",
1360 .codec_name = "msm-stub-codec.1",
1361 .codec_dai_name = "msm-stub-rx",
1362 .no_pcm = 1,
1363 .no_codec = 1,
1364 .be_id = MSM_BACKEND_DAI_VOICE_PLAYBACK_TX,
1365 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1366 },
Helen Zenge3d716a2011-10-14 16:32:16 -07001367 /* Incall Record Uplink BACK END DAI Link */
1368 {
1369 .name = LPASS_BE_INCALL_RECORD_TX,
1370 .stream_name = "Voice Uplink Capture",
1371 .cpu_dai_name = "msm-dai-q6.32772",
1372 .platform_name = "msm-pcm-routing",
1373 .codec_name = "msm-stub-codec.1",
1374 .codec_dai_name = "msm-stub-tx",
1375 .no_pcm = 1,
1376 .no_codec = 1,
1377 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_TX,
1378 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1379 },
1380 /* Incall Record Downlink BACK END DAI Link */
1381 {
1382 .name = LPASS_BE_INCALL_RECORD_RX,
1383 .stream_name = "Voice Downlink Capture",
1384 .cpu_dai_name = "msm-dai-q6.32771",
1385 .platform_name = "msm-pcm-routing",
1386 .codec_name = "msm-stub-codec.1",
1387 .codec_dai_name = "msm-stub-tx",
1388 .no_pcm = 1,
1389 .no_codec = 1,
1390 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_RX,
1391 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1392 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001393};
1394
Kuirong Wangb25838e2012-01-16 23:37:23 -08001395static struct snd_soc_dai_link msm8960_dai_delta_tabla1x[] = {
1396 /* Backend DAI Links */
1397 {
1398 .name = LPASS_BE_SLIMBUS_0_RX,
1399 .stream_name = "Slimbus Playback",
1400 .cpu_dai_name = "msm-dai-q6.16384",
1401 .platform_name = "msm-pcm-routing",
1402 .codec_name = "tabla1x_codec",
1403 .codec_dai_name = "tabla_rx1",
1404 .no_pcm = 1,
1405 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1406 .init = &msm8960_audrx_init,
1407 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1408 .ops = &msm8960_be_ops,
1409 },
1410 {
1411 .name = LPASS_BE_SLIMBUS_0_TX,
1412 .stream_name = "Slimbus Capture",
1413 .cpu_dai_name = "msm-dai-q6.16385",
1414 .platform_name = "msm-pcm-routing",
1415 .codec_name = "tabla1x_codec",
1416 .codec_dai_name = "tabla_tx1",
1417 .no_pcm = 1,
1418 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1419 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1420 .ops = &msm8960_be_ops,
1421 },
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001422 {
1423 .name = "SLIMBUS_2 Hostless",
1424 .stream_name = "SLIMBUS_2 Hostless",
1425 .cpu_dai_name = "msm-dai-q6.16389",
1426 .platform_name = "msm-pcm-hostless",
1427 .codec_name = "tabla1x_codec",
1428 .codec_dai_name = "tabla_tx2",
Swaminathan Sathappan630c5a42012-05-10 17:33:01 -07001429 .ignore_suspend = 1,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001430 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1431 .ops = &msm8960_be_ops,
1432 },
Kuirong Wangb25838e2012-01-16 23:37:23 -08001433};
1434
1435
1436static struct snd_soc_dai_link msm8960_dai_delta_tabla2x[] = {
1437 /* Backend DAI Links */
1438 {
1439 .name = LPASS_BE_SLIMBUS_0_RX,
1440 .stream_name = "Slimbus Playback",
1441 .cpu_dai_name = "msm-dai-q6.16384",
1442 .platform_name = "msm-pcm-routing",
1443 .codec_name = "tabla_codec",
1444 .codec_dai_name = "tabla_rx1",
1445 .no_pcm = 1,
1446 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1447 .init = &msm8960_audrx_init,
1448 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1449 .ops = &msm8960_be_ops,
1450 },
1451 {
1452 .name = LPASS_BE_SLIMBUS_0_TX,
1453 .stream_name = "Slimbus Capture",
1454 .cpu_dai_name = "msm-dai-q6.16385",
1455 .platform_name = "msm-pcm-routing",
1456 .codec_name = "tabla_codec",
1457 .codec_dai_name = "tabla_tx1",
1458 .no_pcm = 1,
1459 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1460 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1461 .ops = &msm8960_be_ops,
1462 },
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001463 {
1464 .name = "SLIMBUS_2 Hostless",
1465 .stream_name = "SLIMBUS_2 Hostless",
1466 .cpu_dai_name = "msm-dai-q6.16389",
1467 .platform_name = "msm-pcm-hostless",
1468 .codec_name = "tabla_codec",
1469 .codec_dai_name = "tabla_tx2",
Swaminathan Sathappan630c5a42012-05-10 17:33:01 -07001470 .ignore_suspend = 1,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001471 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1472 .ops = &msm8960_be_ops,
1473 },
Kuirong Wangb25838e2012-01-16 23:37:23 -08001474};
1475
1476static struct snd_soc_dai_link msm8960_tabla1x_dai[
1477 ARRAY_SIZE(msm8960_dai_common) +
1478 ARRAY_SIZE(msm8960_dai_delta_tabla1x)];
1479
1480
1481static struct snd_soc_dai_link msm8960_dai[
1482 ARRAY_SIZE(msm8960_dai_common) +
1483 ARRAY_SIZE(msm8960_dai_delta_tabla2x)];
1484
1485static struct snd_soc_card snd_soc_tabla1x_card_msm8960 = {
1486 .name = "msm8960-tabla1x-snd-card",
1487 .dai_link = msm8960_tabla1x_dai,
1488 .num_links = ARRAY_SIZE(msm8960_tabla1x_dai),
1489};
1490
1491static struct snd_soc_card snd_soc_card_msm8960 = {
1492 .name = "msm8960-snd-card",
1493 .dai_link = msm8960_dai,
1494 .num_links = ARRAY_SIZE(msm8960_dai),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001495};
1496
1497static struct platform_device *msm8960_snd_device;
Kuirong Wangb25838e2012-01-16 23:37:23 -08001498static struct platform_device *msm8960_snd_tabla1x_device;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001499
1500static int msm8960_configure_headset_mic_gpios(void)
1501{
1502 int ret;
1503 struct pm_gpio param = {
1504 .direction = PM_GPIO_DIR_OUT,
1505 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
1506 .output_value = 1,
1507 .pull = PM_GPIO_PULL_NO,
1508 .vin_sel = PM_GPIO_VIN_S4,
1509 .out_strength = PM_GPIO_STRENGTH_MED,
1510 .function = PM_GPIO_FUNC_NORMAL,
1511 };
1512
1513 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(23), "AV_SWITCH");
1514 if (ret) {
1515 pr_err("%s: Failed to request gpio %d\n", __func__,
1516 PM8921_GPIO_PM_TO_SYS(23));
1517 return ret;
1518 }
1519
1520 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(23), &param);
1521 if (ret)
1522 pr_err("%s: Failed to configure gpio %d\n", __func__,
1523 PM8921_GPIO_PM_TO_SYS(23));
1524 else
1525 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(23), 0);
1526
1527 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(35), "US_EURO_SWITCH");
1528 if (ret) {
1529 pr_err("%s: Failed to request gpio %d\n", __func__,
1530 PM8921_GPIO_PM_TO_SYS(35));
1531 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1532 return ret;
1533 }
1534 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(35), &param);
1535 if (ret)
1536 pr_err("%s: Failed to configure gpio %d\n", __func__,
1537 PM8921_GPIO_PM_TO_SYS(35));
1538 else
Patrick Lai2ae0a002011-09-26 14:51:09 -07001539 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(35), 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001540
1541 return 0;
1542}
1543static void msm8960_free_headset_mic_gpios(void)
1544{
1545 if (msm8960_headset_gpios_configured) {
1546 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1547 gpio_free(PM8921_GPIO_PM_TO_SYS(35));
1548 }
1549}
1550
1551static int __init msm8960_audio_init(void)
1552{
1553 int ret;
1554
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301555 if (!cpu_is_msm8960()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001556 pr_err("%s: Not the right machine type\n", __func__);
1557 return -ENODEV ;
1558 }
Joonwoo Park0976d012011-12-22 11:48:18 -08001559
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001560 mbhc_cfg.calibration = def_tabla_mbhc_cal();
1561 if (!mbhc_cfg.calibration) {
Joonwoo Park0976d012011-12-22 11:48:18 -08001562 pr_err("Calibration data allocation failed\n");
1563 return -ENOMEM;
1564 }
1565
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001566 msm8960_snd_device = platform_device_alloc("soc-audio", 0);
1567 if (!msm8960_snd_device) {
1568 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001569 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001570 return -ENOMEM;
1571 }
1572
Kuirong Wangb25838e2012-01-16 23:37:23 -08001573 memcpy(msm8960_dai, msm8960_dai_common, sizeof(msm8960_dai_common));
1574 memcpy(msm8960_dai + ARRAY_SIZE(msm8960_dai_common),
1575 msm8960_dai_delta_tabla2x, sizeof(msm8960_dai_delta_tabla2x));
1576
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001577 platform_set_drvdata(msm8960_snd_device, &snd_soc_card_msm8960);
1578 ret = platform_device_add(msm8960_snd_device);
1579 if (ret) {
1580 platform_device_put(msm8960_snd_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001581 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001582 return ret;
1583 }
1584
Kuirong Wangb25838e2012-01-16 23:37:23 -08001585 msm8960_snd_tabla1x_device = platform_device_alloc("soc-audio", 1);
1586 if (!msm8960_snd_tabla1x_device) {
1587 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001588 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001589 return -ENOMEM;
1590 }
1591
1592 memcpy(msm8960_tabla1x_dai, msm8960_dai_common,
1593 sizeof(msm8960_dai_common));
1594 memcpy(msm8960_tabla1x_dai + ARRAY_SIZE(msm8960_dai_common),
1595 msm8960_dai_delta_tabla1x, sizeof(msm8960_dai_delta_tabla1x));
1596
1597 platform_set_drvdata(msm8960_snd_tabla1x_device,
1598 &snd_soc_tabla1x_card_msm8960);
1599 ret = platform_device_add(msm8960_snd_tabla1x_device);
1600 if (ret) {
1601 platform_device_put(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001602 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001603 return ret;
1604 }
1605
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001606 if (msm8960_configure_headset_mic_gpios()) {
1607 pr_err("%s Fail to configure headset mic gpios\n", __func__);
1608 msm8960_headset_gpios_configured = 0;
1609 } else
1610 msm8960_headset_gpios_configured = 1;
1611
Joonwoo Park28f49c82012-03-16 12:29:21 -07001612 mutex_init(&cdc_mclk_mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001613 return ret;
1614
1615}
1616module_init(msm8960_audio_init);
1617
1618static void __exit msm8960_audio_exit(void)
1619{
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301620 if (!cpu_is_msm8960()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001621 pr_err("%s: Not the right machine type\n", __func__);
1622 return ;
1623 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001624 msm8960_free_headset_mic_gpios();
1625 platform_device_unregister(msm8960_snd_device);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001626 platform_device_unregister(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001627 kfree(mbhc_cfg.calibration);
Joonwoo Park28f49c82012-03-16 12:29:21 -07001628 mutex_destroy(&cdc_mclk_mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001629}
1630module_exit(msm8960_audio_exit);
1631
1632MODULE_DESCRIPTION("ALSA SoC MSM8960");
1633MODULE_LICENSE("GPL v2");