blob: bb5d817c91992304fad47dd24531bee3065006e3 [file] [log] [blame]
Joonwoo Park0976d012011-12-22 11:48:18 -08001/* Copyright (c) 2011-2012, Code Aurora Forum. All rights reserved.
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
13#include <linux/clk.h>
Kiran Kandidb0a4b02011-08-23 09:32:09 -070014#include <linux/delay.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070015#include <linux/gpio.h>
16#include <linux/mfd/pm8xxx/pm8921.h>
17#include <linux/platform_device.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070018#include <linux/mfd/pm8xxx/pm8921.h>
Joonwoo Park0976d012011-12-22 11:48:18 -080019#include <linux/slab.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070020#include <sound/core.h>
21#include <sound/soc.h>
22#include <sound/soc-dapm.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070023#include <sound/pcm.h>
24#include <sound/jack.h>
Bradley Rubin229c6a52011-07-12 16:18:48 -070025#include <asm/mach-types.h>
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -080026#include <mach/socinfo.h>
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070027#include <linux/mfd/wcd9xxx/core.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070028#include "msm-pcm-routing.h"
Bhalchandra Gajare0b9d4d52011-10-21 16:17:47 -070029#include "../codecs/wcd9310.h"
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070030
31/* 8960 machine driver */
32
33#define PM8921_GPIO_BASE NR_GPIO_IRQS
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070034#define PM8921_IRQ_BASE (NR_MSM_IRQS + NR_GPIO_IRQS)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070035#define PM8921_GPIO_PM_TO_SYS(pm_gpio) (pm_gpio - 1 + PM8921_GPIO_BASE)
36
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070037#define MSM8960_SPK_ON 1
38#define MSM8960_SPK_OFF 0
39
40#define msm8960_SLIM_0_RX_MAX_CHANNELS 2
41#define msm8960_SLIM_0_TX_MAX_CHANNELS 4
42
Kuirong Wang054e1832012-05-11 15:43:30 -070043#define SAMPLE_RATE_8KHZ 8000
44#define SAMPLE_RATE_16KHZ 16000
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070045
Kiran Kandi462acea2011-08-31 23:53:14 -070046#define BOTTOM_SPK_AMP_POS 0x1
47#define BOTTOM_SPK_AMP_NEG 0x2
48#define TOP_SPK_AMP_POS 0x4
49#define TOP_SPK_AMP_NEG 0x8
50
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -070051#define GPIO_AUX_PCM_DOUT 63
52#define GPIO_AUX_PCM_DIN 64
53#define GPIO_AUX_PCM_SYNC 65
54#define GPIO_AUX_PCM_CLK 66
55
Joonwoo Park0976d012011-12-22 11:48:18 -080056#define TABLA_EXT_CLK_RATE 12288000
57
Joonwoo Park6b9b03f2012-01-23 18:48:54 -080058#define TABLA_MBHC_DEF_BUTTONS 8
Joonwoo Park0976d012011-12-22 11:48:18 -080059#define TABLA_MBHC_DEF_RLOADS 5
60
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070061#define JACK_DETECT_GPIO 38
62#define JACK_DETECT_INT PM8921_GPIO_IRQ(PM8921_IRQ_BASE, JACK_DETECT_GPIO)
Joonwoo Park2cc13f02012-05-09 12:44:25 -070063#define JACK_US_EURO_SEL_GPIO 35
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);
Joonwoo Park2cc13f02012-05-09 12:44:25 -070096static bool msm8960_swap_gnd_mic(struct snd_soc_codec *codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070097
98static struct tabla_mbhc_config mbhc_cfg = {
99 .headset_jack = &hs_jack,
100 .button_jack = &button_jack,
101 .read_fw_bin = false,
102 .calibration = NULL,
103 .micbias = TABLA_MICBIAS2,
104 .mclk_cb_fn = msm8960_enable_codec_ext_clk,
105 .mclk_rate = TABLA_EXT_CLK_RATE,
106 .gpio = 0,
107 .gpio_irq = 0,
108 .gpio_level_insert = 1,
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700109 .swap_gnd_mic = NULL,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700110};
Joonwoo Park0976d012011-12-22 11:48:18 -0800111
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700112static u32 us_euro_sel_gpio = PM8921_GPIO_PM_TO_SYS(JACK_US_EURO_SEL_GPIO);
113
Joonwoo Park28f49c82012-03-16 12:29:21 -0700114static struct mutex cdc_mclk_mutex;
115
Kiran Kandi462acea2011-08-31 23:53:14 -0700116static void msm8960_enable_ext_spk_amp_gpio(u32 spk_amp_gpio)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700117{
118 int ret = 0;
119
120 struct pm_gpio param = {
121 .direction = PM_GPIO_DIR_OUT,
122 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
123 .output_value = 1,
124 .pull = PM_GPIO_PULL_NO,
125 .vin_sel = PM_GPIO_VIN_S4,
126 .out_strength = PM_GPIO_STRENGTH_MED,
Kiran Kandi462acea2011-08-31 23:53:14 -0700127 .
128 function = PM_GPIO_FUNC_NORMAL,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700129 };
130
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800131 if (spk_amp_gpio == bottom_spk_pamp_gpio) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700132
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800133 ret = gpio_request(bottom_spk_pamp_gpio, "BOTTOM_SPK_AMP");
Kiran Kandi462acea2011-08-31 23:53:14 -0700134 if (ret) {
135 pr_err("%s: Error requesting BOTTOM SPK AMP GPIO %u\n",
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800136 __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700137 return;
138 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800139 ret = pm8xxx_gpio_config(bottom_spk_pamp_gpio, &param);
Kiran Kandi462acea2011-08-31 23:53:14 -0700140 if (ret)
141 pr_err("%s: Failed to configure Bottom Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800142 " gpio %u\n", __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700143 else {
144 pr_debug("%s: enable Bottom spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800145 gpio_direction_output(bottom_spk_pamp_gpio, 1);
Kiran Kandi462acea2011-08-31 23:53:14 -0700146 }
147
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800148 } else if (spk_amp_gpio == top_spk_pamp_gpio) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700149
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800150 ret = gpio_request(top_spk_pamp_gpio, "TOP_SPK_AMP");
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700151 if (ret) {
152 pr_err("%s: Error requesting GPIO %d\n", __func__,
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800153 top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700154 return;
155 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800156 ret = pm8xxx_gpio_config(top_spk_pamp_gpio, &param);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700157 if (ret)
Kiran Kandi462acea2011-08-31 23:53:14 -0700158 pr_err("%s: Failed to configure Top Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800159 " gpio %u\n", __func__, top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700160 else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700161 pr_debug("%s: enable Top spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800162 gpio_direction_output(top_spk_pamp_gpio, 1);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700163 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700164 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700165 pr_err("%s: ERROR : Invalid External Speaker Ampl GPIO."
166 " gpio = %u\n", __func__, spk_amp_gpio);
167 return;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700168 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700169}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700170
Kiran Kandi462acea2011-08-31 23:53:14 -0700171static void msm8960_ext_spk_power_amp_on(u32 spk)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700172{
Kiran Kandi462acea2011-08-31 23:53:14 -0700173 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
174
175 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
176 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
177
178 pr_debug("%s() External Bottom Speaker Ampl already "
179 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700180 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700181 }
182
183 msm8960_ext_bottom_spk_pamp |= spk;
184
185 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
186 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
187
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800188 msm8960_enable_ext_spk_amp_gpio(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700189 pr_debug("%s: slepping 4 ms after turning on external "
190 " Bottom Speaker Ampl\n", __func__);
191 usleep_range(4000, 4000);
192 }
193
194 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
195
196 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
197 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
198
199 pr_debug("%s() External Top Speaker Ampl already"
200 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700201 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700202 }
203
204 msm8960_ext_top_spk_pamp |= spk;
205
206 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
207 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
208
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800209 msm8960_enable_ext_spk_amp_gpio(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700210 pr_debug("%s: sleeping 4 ms after turning on "
211 " external Top Speaker Ampl\n", __func__);
212 usleep_range(4000, 4000);
213 }
214 } else {
215
216 pr_err("%s: ERROR : Invalid External Speaker Ampl. spk = 0x%08x\n",
217 __func__, spk);
218 return;
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700219 }
Kiran Kandi462acea2011-08-31 23:53:14 -0700220}
221
222static void msm8960_ext_spk_power_amp_off(u32 spk)
223{
224 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
225
226 if (!msm8960_ext_bottom_spk_pamp)
227 return;
228
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800229 gpio_direction_output(bottom_spk_pamp_gpio, 0);
230 gpio_free(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700231 msm8960_ext_bottom_spk_pamp = 0;
232
233 pr_debug("%s: sleeping 4 ms after turning off external Bottom"
234 " Speaker Ampl\n", __func__);
235
236 usleep_range(4000, 4000);
237
238 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
239
240 if (!msm8960_ext_top_spk_pamp)
241 return;
242
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800243 gpio_direction_output(top_spk_pamp_gpio, 0);
244 gpio_free(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700245 msm8960_ext_top_spk_pamp = 0;
246
247 pr_debug("%s: sleeping 4 ms after turning off external Top"
248 " Spkaker Ampl\n", __func__);
249
250 usleep_range(4000, 4000);
251 } else {
252
253 pr_err("%s: ERROR : Invalid Ext Spk Ampl. spk = 0x%08x\n",
254 __func__, spk);
255 return;
256 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700257}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700258
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700259static void msm8960_ext_control(struct snd_soc_codec *codec)
260{
261 struct snd_soc_dapm_context *dapm = &codec->dapm;
262
Jayasena Sangaraboina247d36e2012-04-17 10:16:18 -0700263 mutex_lock(&dapm->codec->mutex);
264
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700265 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700266 if (msm8960_spk_control == MSM8960_SPK_ON) {
267 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
268 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
269 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
270 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
271 } else {
272 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Pos");
273 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Neg");
274 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Pos");
275 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Neg");
276 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700277
278 snd_soc_dapm_sync(dapm);
Jayasena Sangaraboina247d36e2012-04-17 10:16:18 -0700279 mutex_unlock(&dapm->codec->mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700280}
281
282static int msm8960_get_spk(struct snd_kcontrol *kcontrol,
283 struct snd_ctl_elem_value *ucontrol)
284{
285 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
286 ucontrol->value.integer.value[0] = msm8960_spk_control;
287 return 0;
288}
289static int msm8960_set_spk(struct snd_kcontrol *kcontrol,
290 struct snd_ctl_elem_value *ucontrol)
291{
292 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
293
294 pr_debug("%s()\n", __func__);
295 if (msm8960_spk_control == ucontrol->value.integer.value[0])
296 return 0;
297
298 msm8960_spk_control = ucontrol->value.integer.value[0];
299 msm8960_ext_control(codec);
300 return 1;
301}
302static int msm8960_spkramp_event(struct snd_soc_dapm_widget *w,
303 struct snd_kcontrol *k, int event)
304{
305 pr_debug("%s() %x\n", __func__, SND_SOC_DAPM_EVENT_ON(event));
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700306
307 if (SND_SOC_DAPM_EVENT_ON(event)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700308 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
309 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_POS);
310 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
311 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_NEG);
312 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
313 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_POS);
314 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
315 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_NEG);
316 else {
317 pr_err("%s() Invalid Speaker Widget = %s\n",
318 __func__, w->name);
319 return -EINVAL;
320 }
321
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700322 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700323 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
324 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_POS);
325 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
326 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_NEG);
327 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
328 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_POS);
329 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
330 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_NEG);
331 else {
332 pr_err("%s() Invalid Speaker Widget = %s\n",
333 __func__, w->name);
334 return -EINVAL;
335 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700336 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700337 return 0;
338}
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700339
340static int msm8960_enable_codec_ext_clk(struct snd_soc_codec *codec, int enable,
341 bool dapm)
Joonwoo Park0976d012011-12-22 11:48:18 -0800342{
Joonwoo Park28f49c82012-03-16 12:29:21 -0700343 int r = 0;
Joonwoo Park0976d012011-12-22 11:48:18 -0800344 pr_debug("%s: enable = %d\n", __func__, enable);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700345
346 mutex_lock(&cdc_mclk_mutex);
Joonwoo Park0976d012011-12-22 11:48:18 -0800347 if (enable) {
348 clk_users++;
349 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700350 if (clk_users == 1) {
351 if (codec_clk) {
352 clk_set_rate(codec_clk, TABLA_EXT_CLK_RATE);
Asish Bhattacharya4776d962012-04-25 12:08:09 +0530353 clk_prepare_enable(codec_clk);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700354 tabla_mclk_enable(codec, 1, dapm);
355 } else {
356 pr_err("%s: Error setting Tabla MCLK\n",
357 __func__);
358 clk_users--;
359 r = -EINVAL;
360 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800361 }
362 } else {
Joonwoo Park28f49c82012-03-16 12:29:21 -0700363 if (clk_users > 0) {
364 clk_users--;
365 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
366 if (clk_users == 0) {
367 pr_debug("%s: disabling MCLK. clk_users = %d\n",
Joonwoo Park0976d012011-12-22 11:48:18 -0800368 __func__, clk_users);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700369 tabla_mclk_enable(codec, 0, dapm);
Asish Bhattacharya4776d962012-04-25 12:08:09 +0530370 clk_disable_unprepare(codec_clk);
Joonwoo Park28f49c82012-03-16 12:29:21 -0700371 }
372 } else {
373 pr_err("%s: Error releasing Tabla MCLK\n", __func__);
374 r = -EINVAL;
Joonwoo Park0976d012011-12-22 11:48:18 -0800375 }
376 }
Joonwoo Park28f49c82012-03-16 12:29:21 -0700377 mutex_unlock(&cdc_mclk_mutex);
378 return r;
Joonwoo Park0976d012011-12-22 11:48:18 -0800379}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700380
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700381static bool msm8960_swap_gnd_mic(struct snd_soc_codec *codec)
382{
383 int value = gpio_get_value_cansleep(us_euro_sel_gpio);
384 pr_debug("%s: US EURO select switch %d to %d\n", __func__, value,
385 !value);
386 gpio_set_value_cansleep(us_euro_sel_gpio, !value);
387 return true;
388}
389
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700390static int msm8960_mclk_event(struct snd_soc_dapm_widget *w,
391 struct snd_kcontrol *kcontrol, int event)
392{
393 pr_debug("%s: event = %d\n", __func__, event);
394
395 switch (event) {
396 case SND_SOC_DAPM_PRE_PMU:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700397 return msm8960_enable_codec_ext_clk(w->codec, 1, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700398 case SND_SOC_DAPM_POST_PMD:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700399 return msm8960_enable_codec_ext_clk(w->codec, 0, true);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700400 }
401 return 0;
402}
403
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700404static const struct snd_soc_dapm_widget msm8960_dapm_widgets[] = {
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700405
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700406 SND_SOC_DAPM_SUPPLY("MCLK", SND_SOC_NOPM, 0, 0,
407 msm8960_mclk_event, SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
408
Kiran Kandi462acea2011-08-31 23:53:14 -0700409 SND_SOC_DAPM_SPK("Ext Spk Bottom Pos", msm8960_spkramp_event),
410 SND_SOC_DAPM_SPK("Ext Spk Bottom Neg", msm8960_spkramp_event),
411
412 SND_SOC_DAPM_SPK("Ext Spk Top Pos", msm8960_spkramp_event),
413 SND_SOC_DAPM_SPK("Ext Spk Top Neg", msm8960_spkramp_event),
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700414
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700415 SND_SOC_DAPM_MIC("Handset Mic", NULL),
416 SND_SOC_DAPM_MIC("Headset Mic", NULL),
417 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700418 SND_SOC_DAPM_MIC("ANCRight Headset Mic", NULL),
419 SND_SOC_DAPM_MIC("ANCLeft Headset Mic", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700420
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700421 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700422 SND_SOC_DAPM_MIC("Digital Mic2", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700423 SND_SOC_DAPM_MIC("Digital Mic3", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700424 SND_SOC_DAPM_MIC("Digital Mic4", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700425 SND_SOC_DAPM_MIC("Digital Mic5", NULL),
426 SND_SOC_DAPM_MIC("Digital Mic6", NULL),
427
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700428};
429
Bradley Rubin229c6a52011-07-12 16:18:48 -0700430static const struct snd_soc_dapm_route common_audio_map[] = {
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700431
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700432 {"RX_BIAS", NULL, "MCLK"},
433 {"LDO_H", NULL, "MCLK"},
434
435 /* Speaker path */
Kiran Kandi462acea2011-08-31 23:53:14 -0700436 {"Ext Spk Bottom Pos", NULL, "LINEOUT1"},
437 {"Ext Spk Bottom Neg", NULL, "LINEOUT3"},
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700438
Kiran Kandi462acea2011-08-31 23:53:14 -0700439 {"Ext Spk Top Pos", NULL, "LINEOUT2"},
440 {"Ext Spk Top Neg", NULL, "LINEOUT4"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700441
442 /* Microphone path */
Bradley Rubin229c6a52011-07-12 16:18:48 -0700443 {"AMIC1", NULL, "MIC BIAS1 Internal1"},
444 {"MIC BIAS1 Internal1", NULL, "Handset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700445
446 {"AMIC2", NULL, "MIC BIAS2 External"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700447 {"MIC BIAS2 External", NULL, "Headset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700448
Patrick Laie519dd62011-09-28 12:37:11 -0700449 /**
450 * AMIC3 and AMIC4 inputs are connected to ANC microphones
451 * These mics are biased differently on CDP and FLUID
452 * routing entries below are based on bias arrangement
453 * on FLUID.
454 */
455 {"AMIC3", NULL, "MIC BIAS3 Internal1"},
456 {"MIC BIAS3 Internal1", NULL, "ANCRight Headset Mic"},
457
458 {"AMIC4", NULL, "MIC BIAS1 Internal2"},
459 {"MIC BIAS1 Internal2", NULL, "ANCLeft Headset Mic"},
460
Kiran Kandie9bf86a2011-07-21 16:50:41 -0700461 {"HEADPHONE", NULL, "LDO_H"},
462
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700463 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700464 * The digital Mic routes are setup considering
465 * fluid as default device.
466 */
467
468 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700469 * Digital Mic1. Front Bottom left Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700470 * Digital Mic GM5 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700471 * Conncted to DMIC2 Input on Tabla codec.
472 */
473 {"DMIC2", NULL, "MIC BIAS1 External"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700474 {"MIC BIAS1 External", NULL, "Digital Mic1"},
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700475
476 /**
477 * Digital Mic2. Front Bottom right Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700478 * Digital Mic GM6 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700479 * Conncted to DMIC1 Input on Tabla codec.
480 */
481 {"DMIC1", NULL, "MIC BIAS1 External"},
482 {"MIC BIAS1 External", NULL, "Digital Mic2"},
483
484 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700485 * Digital Mic3. Back Bottom Digital Mic on Fluid.
486 * Digital Mic GM1 on CDP mainboard.
487 * Conncted to DMIC4 Input on Tabla codec.
488 */
489 {"DMIC4", NULL, "MIC BIAS3 External"},
490 {"MIC BIAS3 External", NULL, "Digital Mic3"},
491
492 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700493 * Digital Mic4. Back top Digital Mic on Fluid.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700494 * Digital Mic GM2 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700495 * Conncted to DMIC3 Input on Tabla codec.
496 */
497 {"DMIC3", NULL, "MIC BIAS3 External"},
498 {"MIC BIAS3 External", NULL, "Digital Mic4"},
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700499
500 /**
501 * Digital Mic5. Front top Digital Mic on Fluid.
502 * Digital Mic GM3 on CDP mainboard.
503 * Conncted to DMIC5 Input on Tabla codec.
504 */
505 {"DMIC5", NULL, "MIC BIAS4 External"},
506 {"MIC BIAS4 External", NULL, "Digital Mic5"},
507
Patrick Laicb7802b2011-10-04 12:39:18 -0700508 /* Tabla digital Mic6 - back bottom digital Mic on Liquid and
509 * bottom mic on CDP. FLUID/MTP do not have dmic6 installed.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700510 */
511 {"DMIC6", NULL, "MIC BIAS4 External"},
512 {"MIC BIAS4 External", NULL, "Digital Mic6"},
Bradley Rubin229c6a52011-07-12 16:18:48 -0700513};
514
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700515static const char *spk_function[] = {"Off", "On"};
Patrick Lai9f4b4292011-07-16 22:11:09 -0700516static const char *slim0_rx_ch_text[] = {"One", "Two"};
517static const char *slim0_tx_ch_text[] = {"One", "Two", "Three", "Four"};
518
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700519static const struct soc_enum msm8960_enum[] = {
520 SOC_ENUM_SINGLE_EXT(2, spk_function),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700521 SOC_ENUM_SINGLE_EXT(2, slim0_rx_ch_text),
522 SOC_ENUM_SINGLE_EXT(4, slim0_tx_ch_text),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700523};
524
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700525static const char *btsco_rate_text[] = {"8000", "16000"};
526static const struct soc_enum msm8960_btsco_enum[] = {
527 SOC_ENUM_SINGLE_EXT(2, btsco_rate_text),
528};
529
Kuirong Wang054e1832012-05-11 15:43:30 -0700530static const char *auxpcm_rate_text[] = {"rate_8000", "rate_16000"};
531static const struct soc_enum msm8960_auxpcm_enum[] = {
532 SOC_ENUM_SINGLE_EXT(2, auxpcm_rate_text),
533};
534
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700535static int msm8960_slim_0_rx_ch_get(struct snd_kcontrol *kcontrol,
536 struct snd_ctl_elem_value *ucontrol)
537{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700538 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800539 msm8960_slim_0_rx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700540 ucontrol->value.integer.value[0] = msm8960_slim_0_rx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700541 return 0;
542}
543
544static int msm8960_slim_0_rx_ch_put(struct snd_kcontrol *kcontrol,
545 struct snd_ctl_elem_value *ucontrol)
546{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700547 msm8960_slim_0_rx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700548
Patrick Lai9f4b4292011-07-16 22:11:09 -0700549 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800550 msm8960_slim_0_rx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700551 return 1;
552}
553
554static int msm8960_slim_0_tx_ch_get(struct snd_kcontrol *kcontrol,
555 struct snd_ctl_elem_value *ucontrol)
556{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700557 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800558 msm8960_slim_0_tx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700559 ucontrol->value.integer.value[0] = msm8960_slim_0_tx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700560 return 0;
561}
562
563static int msm8960_slim_0_tx_ch_put(struct snd_kcontrol *kcontrol,
564 struct snd_ctl_elem_value *ucontrol)
565{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700566 msm8960_slim_0_tx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700567
568 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800569 msm8960_slim_0_tx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700570 return 1;
571}
572
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700573static int msm8960_btsco_rate_get(struct snd_kcontrol *kcontrol,
574 struct snd_ctl_elem_value *ucontrol)
575{
Joonwoo Park0976d012011-12-22 11:48:18 -0800576 pr_debug("%s: msm8960_btsco_rate = %d", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700577 ucontrol->value.integer.value[0] = msm8960_btsco_rate;
578 return 0;
579}
580
581static int msm8960_btsco_rate_put(struct snd_kcontrol *kcontrol,
582 struct snd_ctl_elem_value *ucontrol)
583{
584 switch (ucontrol->value.integer.value[0]) {
585 case 0:
Kuirong Wang054e1832012-05-11 15:43:30 -0700586 msm8960_btsco_rate = SAMPLE_RATE_8KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700587 break;
588 case 1:
Kuirong Wang054e1832012-05-11 15:43:30 -0700589 msm8960_btsco_rate = SAMPLE_RATE_16KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700590 break;
591 default:
Kuirong Wang054e1832012-05-11 15:43:30 -0700592 msm8960_btsco_rate = SAMPLE_RATE_8KHZ;
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700593 break;
594 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800595 pr_debug("%s: msm8960_btsco_rate = %d\n", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700596 return 0;
597}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700598
Kuirong Wang054e1832012-05-11 15:43:30 -0700599static int msm8960_auxpcm_rate_get(struct snd_kcontrol *kcontrol,
600 struct snd_ctl_elem_value *ucontrol)
601{
602 pr_debug("%s: msm8960_auxpcm_rate = %d", __func__,
603 msm8960_auxpcm_rate);
604 ucontrol->value.integer.value[0] = msm8960_auxpcm_rate;
605 return 0;
606}
607
608static int msm8960_auxpcm_rate_put(struct snd_kcontrol *kcontrol,
609 struct snd_ctl_elem_value *ucontrol)
610{
611 switch (ucontrol->value.integer.value[0]) {
612 case 0:
613 msm8960_auxpcm_rate = SAMPLE_RATE_8KHZ;
614 break;
615 case 1:
616 msm8960_auxpcm_rate = SAMPLE_RATE_16KHZ;
617 break;
618 default:
619 msm8960_auxpcm_rate = SAMPLE_RATE_8KHZ;
620 break;
621 }
622 pr_debug("%s: msm8960_auxpcm_rate = %d"
623 "ucontrol->value.integer.value[0] = %d\n", __func__,
624 msm8960_auxpcm_rate,
625 (int)ucontrol->value.integer.value[0]);
626 return 0;
627}
628
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700629static const struct snd_kcontrol_new tabla_msm8960_controls[] = {
630 SOC_ENUM_EXT("Speaker Function", msm8960_enum[0], msm8960_get_spk,
631 msm8960_set_spk),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700632 SOC_ENUM_EXT("SLIM_0_RX Channels", msm8960_enum[1],
633 msm8960_slim_0_rx_ch_get, msm8960_slim_0_rx_ch_put),
634 SOC_ENUM_EXT("SLIM_0_TX Channels", msm8960_enum[2],
635 msm8960_slim_0_tx_ch_get, msm8960_slim_0_tx_ch_put),
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700636 SOC_ENUM_EXT("Internal BTSCO SampleRate", msm8960_btsco_enum[0],
637 msm8960_btsco_rate_get, msm8960_btsco_rate_put),
Kuirong Wang054e1832012-05-11 15:43:30 -0700638 SOC_ENUM_EXT("AUX PCM SampleRate", msm8960_auxpcm_enum[0],
639 msm8960_auxpcm_rate_get, msm8960_auxpcm_rate_put),
640};
641
Joonwoo Park0976d012011-12-22 11:48:18 -0800642static void *def_tabla_mbhc_cal(void)
643{
644 void *tabla_cal;
645 struct tabla_mbhc_btn_detect_cfg *btn_cfg;
646 u16 *btn_low, *btn_high;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800647 u8 *n_ready, *n_cic, *gain;
Joonwoo Park0976d012011-12-22 11:48:18 -0800648
649 tabla_cal = kzalloc(TABLA_MBHC_CAL_SIZE(TABLA_MBHC_DEF_BUTTONS,
650 TABLA_MBHC_DEF_RLOADS),
651 GFP_KERNEL);
652 if (!tabla_cal) {
653 pr_err("%s: out of memory\n", __func__);
654 return NULL;
655 }
656
657#define S(X, Y) ((TABLA_MBHC_CAL_GENERAL_PTR(tabla_cal)->X) = (Y))
658 S(t_ldoh, 100);
659 S(t_bg_fast_settle, 100);
660 S(t_shutdown_plug_rem, 255);
661 S(mbhc_nsa, 4);
662 S(mbhc_navg, 4);
663#undef S
664#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_DET_PTR(tabla_cal)->X) = (Y))
665 S(mic_current, TABLA_PID_MIC_5_UA);
666 S(hph_current, TABLA_PID_MIC_5_UA);
667 S(t_mic_pid, 100);
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800668 S(t_ins_complete, 250);
Joonwoo Park0976d012011-12-22 11:48:18 -0800669 S(t_ins_retry, 200);
670#undef S
671#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla_cal)->X) = (Y))
672 S(v_no_mic, 30);
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700673 S(v_hs_max, 2400);
Joonwoo Park0976d012011-12-22 11:48:18 -0800674#undef S
675#define S(X, Y) ((TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal)->X) = (Y))
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800676 S(c[0], 62);
677 S(c[1], 124);
678 S(nc, 1);
679 S(n_meas, 3);
Joonwoo Park0976d012011-12-22 11:48:18 -0800680 S(mbhc_nsc, 11);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800681 S(n_btn_meas, 1);
682 S(n_btn_con, 2);
Joonwoo Park0976d012011-12-22 11:48:18 -0800683 S(num_btn, TABLA_MBHC_DEF_BUTTONS);
684 S(v_btn_press_delta_sta, 100);
685 S(v_btn_press_delta_cic, 50);
686#undef S
687 btn_cfg = TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal);
688 btn_low = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_LOW);
689 btn_high = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_HIGH);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800690 btn_low[0] = -50;
Joonwoo Parkfee17432012-04-16 16:33:55 -0700691 btn_high[0] = 20;
692 btn_low[1] = 21;
693 btn_high[1] = 62;
694 btn_low[2] = 63;
695 btn_high[2] = 104;
696 btn_low[3] = 105;
697 btn_high[3] = 143;
698 btn_low[4] = 144;
699 btn_high[4] = 181;
700 btn_low[5] = 182;
701 btn_high[5] = 218;
702 btn_low[6] = 219;
703 btn_high[6] = 254;
704 btn_low[7] = 255;
705 btn_high[7] = 330;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800706 n_ready = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_READY);
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700707 n_ready[0] = 80;
708 n_ready[1] = 68;
Joonwoo Park0976d012011-12-22 11:48:18 -0800709 n_cic = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_CIC);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800710 n_cic[0] = 60;
711 n_cic[1] = 47;
Joonwoo Park0976d012011-12-22 11:48:18 -0800712 gain = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_GAIN);
713 gain[0] = 11;
714 gain[1] = 9;
715
716 return tabla_cal;
717}
718
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800719static int msm8960_hw_params(struct snd_pcm_substream *substream,
720 struct snd_pcm_hw_params *params)
721{
722 struct snd_soc_pcm_runtime *rtd = substream->private_data;
723 struct snd_soc_dai *codec_dai = rtd->codec_dai;
724 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
725 int ret = 0;
726 unsigned int rx_ch[SLIM_MAX_RX_PORTS], tx_ch[SLIM_MAX_TX_PORTS];
727 unsigned int rx_ch_cnt = 0, tx_ch_cnt = 0;
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700728 unsigned int user_set_tx_ch = 0;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800729
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700730
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800731 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700732
733 pr_debug("%s: rx_0_ch=%d\n", __func__, msm8960_slim_0_rx_ch);
734
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800735 ret = snd_soc_dai_get_channel_map(codec_dai,
736 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
737 if (ret < 0) {
738 pr_err("%s: failed to get codec chan map\n", __func__);
739 goto end;
740 }
741
742 ret = snd_soc_dai_set_channel_map(cpu_dai, 0, 0,
743 msm8960_slim_0_rx_ch, rx_ch);
744 if (ret < 0) {
745 pr_err("%s: failed to set cpu chan map\n", __func__);
746 goto end;
747 }
748 ret = snd_soc_dai_set_channel_map(codec_dai, 0, 0,
749 msm8960_slim_0_rx_ch, rx_ch);
750 if (ret < 0) {
751 pr_err("%s: failed to set codec channel map\n",
752 __func__);
753 goto end;
754 }
755 } else {
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700756
757 if (codec_dai->id == 2)
758 user_set_tx_ch = msm8960_slim_0_tx_ch;
759 else if (codec_dai->id == 4)
760 user_set_tx_ch = params_channels(params);
761
762 pr_debug("%s: %s_tx_dai_id_%d_ch=%d\n", __func__,
763 codec_dai->name, codec_dai->id, user_set_tx_ch);
764
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800765 ret = snd_soc_dai_get_channel_map(codec_dai,
766 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
767 if (ret < 0) {
768 pr_err("%s: failed to get codec chan map\n", __func__);
769 goto end;
770 }
771 ret = snd_soc_dai_set_channel_map(cpu_dai,
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700772 user_set_tx_ch, tx_ch, 0 , 0);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800773 if (ret < 0) {
774 pr_err("%s: failed to set cpu chan map\n", __func__);
775 goto end;
776 }
777 ret = snd_soc_dai_set_channel_map(codec_dai,
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700778 user_set_tx_ch, tx_ch, 0, 0);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800779 if (ret < 0) {
780 pr_err("%s: failed to set codec channel map\n",
781 __func__);
782 goto end;
783 }
784
785
786 }
787end:
788 return ret;
789}
790
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700791static int msm8960_audrx_init(struct snd_soc_pcm_runtime *rtd)
792{
793 int err;
794 struct snd_soc_codec *codec = rtd->codec;
795 struct snd_soc_dapm_context *dapm = &codec->dapm;
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800796 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700797 struct pm_gpio jack_gpio_cfg = {
798 .direction = PM_GPIO_DIR_IN,
799 .pull = PM_GPIO_PULL_UP_1P5,
800 .function = PM_GPIO_FUNC_NORMAL,
801 .vin_sel = 2,
802 .inv_int_pol = 0,
803 };
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700804
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800805 pr_debug("%s(), dev_name%s\n", __func__, dev_name(cpu_dai->dev));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700806
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800807 if (machine_is_msm8960_liquid()) {
808 top_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(19));
809 bottom_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(18));
810 }
811
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700812 snd_soc_dapm_new_controls(dapm, msm8960_dapm_widgets,
813 ARRAY_SIZE(msm8960_dapm_widgets));
814
Bradley Rubin229c6a52011-07-12 16:18:48 -0700815 snd_soc_dapm_add_routes(dapm, common_audio_map,
816 ARRAY_SIZE(common_audio_map));
817
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700818 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
819 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
820 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
821 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700822
823 snd_soc_dapm_sync(dapm);
824
825 err = snd_soc_jack_new(codec, "Headset Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800826 (SND_JACK_HEADSET | SND_JACK_OC_HPHL |
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700827 SND_JACK_OC_HPHR | SND_JACK_UNSUPPORTED),
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800828 &hs_jack);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700829 if (err) {
830 pr_err("failed to create new jack\n");
831 return err;
832 }
Bradley Rubincb1e2732011-06-23 16:49:20 -0700833
834 err = snd_soc_jack_new(codec, "Button Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800835 TABLA_JACK_BUTTON_MASK, &button_jack);
Bradley Rubincb1e2732011-06-23 16:49:20 -0700836 if (err) {
837 pr_err("failed to create new jack\n");
838 return err;
839 }
840
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800841 codec_clk = clk_get(cpu_dai->dev, "osr_clk");
842
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700843 if (machine_is_msm8960_cdp())
844 mbhc_cfg.swap_gnd_mic = msm8960_swap_gnd_mic;
845
Joonwoo Park9115ade2012-04-18 13:14:10 -0700846 if (hs_detect_use_gpio) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700847 mbhc_cfg.gpio = PM8921_GPIO_PM_TO_SYS(JACK_DETECT_GPIO);
848 mbhc_cfg.gpio_irq = JACK_DETECT_INT;
849 }
850
851 if (mbhc_cfg.gpio) {
852 err = pm8xxx_gpio_config(mbhc_cfg.gpio, &jack_gpio_cfg);
853 if (err) {
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700854 pr_err("%s: pm8xxx_gpio_config JACK_DETECT failed %d\n",
855 __func__, err);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700856 return err;
857 }
858 }
Joonwoo Park9115ade2012-04-18 13:14:10 -0700859
860 mbhc_cfg.read_fw_bin = hs_detect_use_firmware;
861
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700862 err = tabla_hs_detect(codec, &mbhc_cfg);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700863
Joonwoo Park03324832012-03-19 19:36:16 -0700864 return err;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700865}
866
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700867static int msm8960_slim_0_rx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
868 struct snd_pcm_hw_params *params)
869{
870 struct snd_interval *rate = hw_param_interval(params,
871 SNDRV_PCM_HW_PARAM_RATE);
872
873 struct snd_interval *channels = hw_param_interval(params,
874 SNDRV_PCM_HW_PARAM_CHANNELS);
875
876 pr_debug("%s()\n", __func__);
877 rate->min = rate->max = 48000;
878 channels->min = channels->max = msm8960_slim_0_rx_ch;
879
880 return 0;
881}
882
883static int msm8960_slim_0_tx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
884 struct snd_pcm_hw_params *params)
885{
886 struct snd_interval *rate = hw_param_interval(params,
887 SNDRV_PCM_HW_PARAM_RATE);
888
889 struct snd_interval *channels = hw_param_interval(params,
890 SNDRV_PCM_HW_PARAM_CHANNELS);
891
892 pr_debug("%s()\n", __func__);
893 rate->min = rate->max = 48000;
894 channels->min = channels->max = msm8960_slim_0_tx_ch;
895
896 return 0;
897}
898
899static int msm8960_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
900 struct snd_pcm_hw_params *params)
901{
902 struct snd_interval *rate = hw_param_interval(params,
903 SNDRV_PCM_HW_PARAM_RATE);
904
905 pr_debug("%s()\n", __func__);
906 rate->min = rate->max = 48000;
907
908 return 0;
909}
910
Helen Zeng73f7fc62011-11-18 16:29:59 -0800911static int msm8960_hdmi_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
912 struct snd_pcm_hw_params *params)
913{
914 struct snd_interval *rate = hw_param_interval(params,
915 SNDRV_PCM_HW_PARAM_RATE);
916
917 struct snd_interval *channels = hw_param_interval(params,
918 SNDRV_PCM_HW_PARAM_CHANNELS);
919
Kiran Kandi5e809b02012-01-31 00:24:33 -0800920 pr_debug("%s channels->min %u channels->max %u ()\n", __func__,
921 channels->min, channels->max);
922
Helen Zeng73f7fc62011-11-18 16:29:59 -0800923 rate->min = rate->max = 48000;
Helen Zeng73f7fc62011-11-18 16:29:59 -0800924
925 return 0;
926}
927
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700928static int msm8960_btsco_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
929 struct snd_pcm_hw_params *params)
930{
931 struct snd_interval *rate = hw_param_interval(params,
932 SNDRV_PCM_HW_PARAM_RATE);
933
934 struct snd_interval *channels = hw_param_interval(params,
935 SNDRV_PCM_HW_PARAM_CHANNELS);
936
937 rate->min = rate->max = msm8960_btsco_rate;
938 channels->min = channels->max = msm8960_btsco_ch;
939
940 return 0;
941}
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700942static int msm8960_auxpcm_be_params_fixup(struct snd_soc_pcm_runtime *rtd,
943 struct snd_pcm_hw_params *params)
944{
945 struct snd_interval *rate = hw_param_interval(params,
946 SNDRV_PCM_HW_PARAM_RATE);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700947
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700948 struct snd_interval *channels = hw_param_interval(params,
949 SNDRV_PCM_HW_PARAM_CHANNELS);
950
Kuirong Wang054e1832012-05-11 15:43:30 -0700951 rate->min = rate->max = msm8960_auxpcm_rate;
952 /* PCM only supports mono output */
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700953 channels->min = channels->max = 1;
954
955 return 0;
956}
957static int msm8960_aux_pcm_get_gpios(void)
958{
959 int ret = 0;
960
961 pr_debug("%s\n", __func__);
962
963 ret = gpio_request(GPIO_AUX_PCM_DOUT, "AUX PCM DOUT");
964 if (ret < 0) {
965 pr_err("%s: Failed to request gpio(%d): AUX PCM DOUT",
966 __func__, GPIO_AUX_PCM_DOUT);
967 goto fail_dout;
968 }
969
970 ret = gpio_request(GPIO_AUX_PCM_DIN, "AUX PCM DIN");
971 if (ret < 0) {
972 pr_err("%s: Failed to request gpio(%d): AUX PCM DIN",
973 __func__, GPIO_AUX_PCM_DIN);
974 goto fail_din;
975 }
976
977 ret = gpio_request(GPIO_AUX_PCM_SYNC, "AUX PCM SYNC");
978 if (ret < 0) {
979 pr_err("%s: Failed to request gpio(%d): AUX PCM SYNC",
980 __func__, GPIO_AUX_PCM_SYNC);
981 goto fail_sync;
982 }
983 ret = gpio_request(GPIO_AUX_PCM_CLK, "AUX PCM CLK");
984 if (ret < 0) {
985 pr_err("%s: Failed to request gpio(%d): AUX PCM CLK",
986 __func__, GPIO_AUX_PCM_CLK);
987 goto fail_clk;
988 }
989
990 return 0;
991
992fail_clk:
993 gpio_free(GPIO_AUX_PCM_SYNC);
994fail_sync:
995 gpio_free(GPIO_AUX_PCM_DIN);
996fail_din:
997 gpio_free(GPIO_AUX_PCM_DOUT);
998fail_dout:
999
1000 return ret;
1001}
1002
1003static int msm8960_aux_pcm_free_gpios(void)
1004{
1005 gpio_free(GPIO_AUX_PCM_DIN);
1006 gpio_free(GPIO_AUX_PCM_DOUT);
1007 gpio_free(GPIO_AUX_PCM_SYNC);
1008 gpio_free(GPIO_AUX_PCM_CLK);
1009
1010 return 0;
1011}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001012static int msm8960_startup(struct snd_pcm_substream *substream)
1013{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07001014 pr_debug("%s(): substream = %s stream = %d\n", __func__,
1015 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001016 return 0;
1017}
1018
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001019static int msm8960_auxpcm_startup(struct snd_pcm_substream *substream)
1020{
1021 int ret = 0;
1022
1023 pr_debug("%s(): substream = %s\n", __func__, substream->name);
1024 ret = msm8960_aux_pcm_get_gpios();
1025 if (ret < 0) {
1026 pr_err("%s: Aux PCM GPIO request failed\n", __func__);
1027 return -EINVAL;
1028 }
1029 return 0;
1030}
1031
1032static void msm8960_auxpcm_shutdown(struct snd_pcm_substream *substream)
1033{
1034
1035 pr_debug("%s(): substream = %s\n", __func__, substream->name);
1036 msm8960_aux_pcm_free_gpios();
1037}
1038
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001039static void msm8960_shutdown(struct snd_pcm_substream *substream)
1040{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07001041 pr_debug("%s(): substream = %s stream = %d\n", __func__,
1042 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001043}
1044
1045static struct snd_soc_ops msm8960_be_ops = {
1046 .startup = msm8960_startup,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08001047 .hw_params = msm8960_hw_params,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001048 .shutdown = msm8960_shutdown,
1049};
1050
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001051static struct snd_soc_ops msm8960_auxpcm_be_ops = {
1052 .startup = msm8960_auxpcm_startup,
1053 .shutdown = msm8960_auxpcm_shutdown,
1054};
1055
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001056/* Digital audio interface glue - connects codec <---> CPU */
Kuirong Wangb25838e2012-01-16 23:37:23 -08001057static struct snd_soc_dai_link msm8960_dai_common[] = {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001058 /* FrontEnd DAI Links */
1059 {
1060 .name = "MSM8960 Media1",
1061 .stream_name = "MultiMedia1",
1062 .cpu_dai_name = "MultiMedia1",
1063 .platform_name = "msm-pcm-dsp",
1064 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001065 .codec_dai_name = "snd-soc-dummy-dai",
1066 .codec_name = "snd-soc-dummy",
1067 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1068 .ignore_suspend = 1,
1069 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001070 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA1
1071 },
1072 {
1073 .name = "MSM8960 Media2",
1074 .stream_name = "MultiMedia2",
1075 .cpu_dai_name = "MultiMedia2",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001076 .platform_name = "msm-multi-ch-pcm-dsp",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001077 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001078 .codec_dai_name = "snd-soc-dummy-dai",
1079 .codec_name = "snd-soc-dummy",
1080 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1081 .ignore_suspend = 1,
1082 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001083 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA2,
1084 },
1085 {
1086 .name = "Circuit-Switch Voice",
1087 .stream_name = "CS-Voice",
1088 .cpu_dai_name = "CS-VOICE",
1089 .platform_name = "msm-pcm-voice",
1090 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001091 .codec_dai_name = "snd-soc-dummy-dai",
1092 .codec_name = "snd-soc-dummy",
1093 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
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,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001096 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1097 .be_id = MSM_FRONTEND_DAI_CS_VOICE,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001098 },
1099 {
1100 .name = "MSM VoIP",
1101 .stream_name = "VoIP",
1102 .cpu_dai_name = "VoIP",
1103 .platform_name = "msm-voip-dsp",
1104 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001105 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1106 .codec_dai_name = "snd-soc-dummy-dai",
1107 .codec_name = "snd-soc-dummy",
1108 .ignore_suspend = 1,
1109 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001110 .be_id = MSM_FRONTEND_DAI_VOIP,
1111 },
1112 {
1113 .name = "MSM8960 LPA",
1114 .stream_name = "LPA",
1115 .cpu_dai_name = "MultiMedia3",
1116 .platform_name = "msm-pcm-lpa",
1117 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001118 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1119 .codec_dai_name = "snd-soc-dummy-dai",
1120 .codec_name = "snd-soc-dummy",
1121 .ignore_suspend = 1,
1122 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001123 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA3,
1124 },
1125 /* Hostless PMC purpose */
1126 {
1127 .name = "SLIMBUS_0 Hostless",
1128 .stream_name = "SLIMBUS_0 Hostless",
1129 .cpu_dai_name = "SLIMBUS0_HOSTLESS",
1130 .platform_name = "msm-pcm-hostless",
1131 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001132 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001133 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001134 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001135 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1136 .codec_dai_name = "snd-soc-dummy-dai",
1137 .codec_name = "snd-soc-dummy",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001138 },
1139 {
1140 .name = "INT_FM Hostless",
1141 .stream_name = "INT_FM Hostless",
1142 .cpu_dai_name = "INT_FM_HOSTLESS",
1143 .platform_name = "msm-pcm-hostless",
1144 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001145 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001146 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001147 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001148 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1149 .codec_dai_name = "snd-soc-dummy-dai",
1150 .codec_name = "snd-soc-dummy",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001151 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301152 {
1153 .name = "MSM AFE-PCM RX",
1154 .stream_name = "AFE-PROXY RX",
1155 .cpu_dai_name = "msm-dai-q6.241",
1156 .codec_name = "msm-stub-codec.1",
1157 .codec_dai_name = "msm-stub-rx",
1158 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001159 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001160 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301161 },
1162 {
1163 .name = "MSM AFE-PCM TX",
1164 .stream_name = "AFE-PROXY TX",
1165 .cpu_dai_name = "msm-dai-q6.240",
1166 .codec_name = "msm-stub-codec.1",
1167 .codec_dai_name = "msm-stub-tx",
1168 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001169 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301170 },
Asish Bhattacharya09f9e0a2011-11-11 13:22:47 +05301171 {
1172 .name = "MSM8960 Compr",
1173 .stream_name = "COMPR",
1174 .cpu_dai_name = "MultiMedia4",
1175 .platform_name = "msm-compr-dsp",
1176 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001177 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
1178 .codec_dai_name = "snd-soc-dummy-dai",
1179 .codec_name = "snd-soc-dummy",
1180 .ignore_suspend = 1,
1181 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Asish Bhattacharya09f9e0a2011-11-11 13:22:47 +05301182 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA4,
1183 },
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001184 {
1185 .name = "AUXPCM Hostless",
1186 .stream_name = "AUXPCM Hostless",
1187 .cpu_dai_name = "AUXPCM_HOSTLESS",
1188 .platform_name = "msm-pcm-hostless",
1189 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001190 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001191 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1192 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001193 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1194 .codec_dai_name = "snd-soc-dummy-dai",
1195 .codec_name = "snd-soc-dummy",
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001196 },
Alex Wongb3d06a02012-01-12 10:00:41 -08001197 /* HDMI Hostless */
1198 {
1199 .name = "HDMI_RX_HOSTLESS",
1200 .stream_name = "HDMI_RX_HOSTLESS",
1201 .cpu_dai_name = "HDMI_HOSTLESS",
1202 .platform_name = "msm-pcm-hostless",
1203 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001204 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Alex Wongb3d06a02012-01-12 10:00:41 -08001205 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb3d06a02012-01-12 10:00:41 -08001206 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001207 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1208 .codec_dai_name = "snd-soc-dummy-dai",
1209 .codec_name = "snd-soc-dummy",
Alex Wongb3d06a02012-01-12 10:00:41 -08001210 },
Venkat Sudhir1c79c3b2012-04-09 23:42:28 -07001211 {
1212 .name = "VoLTE",
1213 .stream_name = "VoLTE",
1214 .cpu_dai_name = "VoLTE",
1215 .platform_name = "msm-pcm-voice",
1216 .dynamic = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001217 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST},
Venkat Sudhir1c79c3b2012-04-09 23:42:28 -07001218 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1219 .ignore_suspend = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001220 .ignore_pmdown_time = 1, /* this dainlink has playback support */
1221 .codec_dai_name = "snd-soc-dummy-dai",
1222 .codec_name = "snd-soc-dummy",
1223 .be_id = MSM_FRONTEND_DAI_VOLTE,
Venkat Sudhir1c79c3b2012-04-09 23:42:28 -07001224 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001225 /* Backend BT/FM DAI Links */
1226 {
1227 .name = LPASS_BE_INT_BT_SCO_RX,
1228 .stream_name = "Internal BT-SCO Playback",
1229 .cpu_dai_name = "msm-dai-q6.12288",
1230 .platform_name = "msm-pcm-routing",
1231 .codec_name = "msm-stub-codec.1",
1232 .codec_dai_name = "msm-stub-rx",
1233 .no_pcm = 1,
1234 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_RX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001235 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001236 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001237 },
1238 {
1239 .name = LPASS_BE_INT_BT_SCO_TX,
1240 .stream_name = "Internal BT-SCO Capture",
1241 .cpu_dai_name = "msm-dai-q6.12289",
1242 .platform_name = "msm-pcm-routing",
1243 .codec_name = "msm-stub-codec.1",
1244 .codec_dai_name = "msm-stub-tx",
1245 .no_pcm = 1,
1246 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_TX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001247 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001248 },
1249 {
1250 .name = LPASS_BE_INT_FM_RX,
1251 .stream_name = "Internal FM Playback",
1252 .cpu_dai_name = "msm-dai-q6.12292",
1253 .platform_name = "msm-pcm-routing",
1254 .codec_name = "msm-stub-codec.1",
1255 .codec_dai_name = "msm-stub-rx",
1256 .no_pcm = 1,
1257 .be_id = MSM_BACKEND_DAI_INT_FM_RX,
1258 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001259 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001260 },
1261 {
1262 .name = LPASS_BE_INT_FM_TX,
1263 .stream_name = "Internal FM Capture",
1264 .cpu_dai_name = "msm-dai-q6.12293",
1265 .platform_name = "msm-pcm-routing",
1266 .codec_name = "msm-stub-codec.1",
1267 .codec_dai_name = "msm-stub-tx",
1268 .no_pcm = 1,
1269 .be_id = MSM_BACKEND_DAI_INT_FM_TX,
1270 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1271 },
1272 /* HDMI BACK END DAI Link */
1273 {
1274 .name = LPASS_BE_HDMI,
1275 .stream_name = "HDMI Playback",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001276 .cpu_dai_name = "msm-dai-q6-hdmi.8",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001277 .platform_name = "msm-pcm-routing",
1278 .codec_name = "msm-stub-codec.1",
1279 .codec_dai_name = "msm-stub-rx",
1280 .no_pcm = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001281 .be_id = MSM_BACKEND_DAI_HDMI_RX,
Helen Zeng73f7fc62011-11-18 16:29:59 -08001282 .be_hw_params_fixup = msm8960_hdmi_be_hw_params_fixup,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001283 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001284 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301285 /* Backend AFE DAI Links */
1286 {
1287 .name = LPASS_BE_AFE_PCM_RX,
1288 .stream_name = "AFE Playback",
1289 .cpu_dai_name = "msm-dai-q6.224",
1290 .platform_name = "msm-pcm-routing",
1291 .codec_name = "msm-stub-codec.1",
1292 .codec_dai_name = "msm-stub-rx",
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301293 .no_pcm = 1,
1294 .be_id = MSM_BACKEND_DAI_AFE_PCM_RX,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001295 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301296 },
1297 {
1298 .name = LPASS_BE_AFE_PCM_TX,
1299 .stream_name = "AFE Capture",
1300 .cpu_dai_name = "msm-dai-q6.225",
1301 .platform_name = "msm-pcm-routing",
1302 .codec_name = "msm-stub-codec.1",
1303 .codec_dai_name = "msm-stub-tx",
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301304 .no_pcm = 1,
1305 .be_id = MSM_BACKEND_DAI_AFE_PCM_TX,
1306 },
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001307 /* AUX PCM Backend DAI Links */
1308 {
1309 .name = LPASS_BE_AUXPCM_RX,
1310 .stream_name = "AUX PCM Playback",
1311 .cpu_dai_name = "msm-dai-q6.2",
1312 .platform_name = "msm-pcm-routing",
1313 .codec_name = "msm-stub-codec.1",
1314 .codec_dai_name = "msm-stub-rx",
1315 .no_pcm = 1,
1316 .be_id = MSM_BACKEND_DAI_AUXPCM_RX,
1317 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1318 .ops = &msm8960_auxpcm_be_ops,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001319 .ignore_pmdown_time = 1,
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001320 },
1321 {
1322 .name = LPASS_BE_AUXPCM_TX,
1323 .stream_name = "AUX PCM Capture",
1324 .cpu_dai_name = "msm-dai-q6.3",
1325 .platform_name = "msm-pcm-routing",
1326 .codec_name = "msm-stub-codec.1",
1327 .codec_dai_name = "msm-stub-tx",
1328 .no_pcm = 1,
1329 .be_id = MSM_BACKEND_DAI_AUXPCM_TX,
1330 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1331 },
Helen Zeng0705a5f2011-10-14 15:29:52 -07001332 /* Incall Music BACK END DAI Link */
1333 {
1334 .name = LPASS_BE_VOICE_PLAYBACK_TX,
1335 .stream_name = "Voice Farend Playback",
1336 .cpu_dai_name = "msm-dai-q6.32773",
1337 .platform_name = "msm-pcm-routing",
1338 .codec_name = "msm-stub-codec.1",
1339 .codec_dai_name = "msm-stub-rx",
1340 .no_pcm = 1,
Helen Zeng0705a5f2011-10-14 15:29:52 -07001341 .be_id = MSM_BACKEND_DAI_VOICE_PLAYBACK_TX,
1342 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1343 },
Helen Zenge3d716a2011-10-14 16:32:16 -07001344 /* Incall Record Uplink BACK END DAI Link */
1345 {
1346 .name = LPASS_BE_INCALL_RECORD_TX,
1347 .stream_name = "Voice Uplink Capture",
1348 .cpu_dai_name = "msm-dai-q6.32772",
1349 .platform_name = "msm-pcm-routing",
1350 .codec_name = "msm-stub-codec.1",
1351 .codec_dai_name = "msm-stub-tx",
1352 .no_pcm = 1,
Helen Zenge3d716a2011-10-14 16:32:16 -07001353 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_TX,
1354 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1355 },
1356 /* Incall Record Downlink BACK END DAI Link */
1357 {
1358 .name = LPASS_BE_INCALL_RECORD_RX,
1359 .stream_name = "Voice Downlink Capture",
1360 .cpu_dai_name = "msm-dai-q6.32771",
1361 .platform_name = "msm-pcm-routing",
1362 .codec_name = "msm-stub-codec.1",
1363 .codec_dai_name = "msm-stub-tx",
1364 .no_pcm = 1,
Helen Zenge3d716a2011-10-14 16:32:16 -07001365 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_RX,
1366 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001367 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Helen Zenge3d716a2011-10-14 16:32:16 -07001368 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001369};
1370
Kuirong Wangb25838e2012-01-16 23:37:23 -08001371static struct snd_soc_dai_link msm8960_dai_delta_tabla1x[] = {
1372 /* Backend DAI Links */
1373 {
1374 .name = LPASS_BE_SLIMBUS_0_RX,
1375 .stream_name = "Slimbus Playback",
1376 .cpu_dai_name = "msm-dai-q6.16384",
1377 .platform_name = "msm-pcm-routing",
1378 .codec_name = "tabla1x_codec",
1379 .codec_dai_name = "tabla_rx1",
1380 .no_pcm = 1,
1381 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1382 .init = &msm8960_audrx_init,
1383 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1384 .ops = &msm8960_be_ops,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001385 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Kuirong Wangb25838e2012-01-16 23:37:23 -08001386 },
1387 {
1388 .name = LPASS_BE_SLIMBUS_0_TX,
1389 .stream_name = "Slimbus Capture",
1390 .cpu_dai_name = "msm-dai-q6.16385",
1391 .platform_name = "msm-pcm-routing",
1392 .codec_name = "tabla1x_codec",
1393 .codec_dai_name = "tabla_tx1",
1394 .no_pcm = 1,
1395 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1396 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1397 .ops = &msm8960_be_ops,
1398 },
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001399 {
1400 .name = "SLIMBUS_2 Hostless",
1401 .stream_name = "SLIMBUS_2 Hostless",
1402 .cpu_dai_name = "msm-dai-q6.16389",
1403 .platform_name = "msm-pcm-hostless",
1404 .codec_name = "tabla1x_codec",
1405 .codec_dai_name = "tabla_tx2",
Swaminathan Sathappan630c5a42012-05-10 17:33:01 -07001406 .ignore_suspend = 1,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001407 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1408 .ops = &msm8960_be_ops,
1409 },
Kuirong Wangb25838e2012-01-16 23:37:23 -08001410};
1411
1412
1413static struct snd_soc_dai_link msm8960_dai_delta_tabla2x[] = {
1414 /* Backend DAI Links */
1415 {
1416 .name = LPASS_BE_SLIMBUS_0_RX,
1417 .stream_name = "Slimbus Playback",
1418 .cpu_dai_name = "msm-dai-q6.16384",
1419 .platform_name = "msm-pcm-routing",
1420 .codec_name = "tabla_codec",
1421 .codec_dai_name = "tabla_rx1",
1422 .no_pcm = 1,
1423 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1424 .init = &msm8960_audrx_init,
1425 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1426 .ops = &msm8960_be_ops,
Steve Mucklef132c6c2012-06-06 18:30:57 -07001427 .ignore_pmdown_time = 1, /* this dainlink has playback support */
Kuirong Wangb25838e2012-01-16 23:37:23 -08001428 },
1429 {
1430 .name = LPASS_BE_SLIMBUS_0_TX,
1431 .stream_name = "Slimbus Capture",
1432 .cpu_dai_name = "msm-dai-q6.16385",
1433 .platform_name = "msm-pcm-routing",
1434 .codec_name = "tabla_codec",
1435 .codec_dai_name = "tabla_tx1",
1436 .no_pcm = 1,
1437 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1438 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1439 .ops = &msm8960_be_ops,
1440 },
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001441 {
1442 .name = "SLIMBUS_2 Hostless",
1443 .stream_name = "SLIMBUS_2 Hostless",
1444 .cpu_dai_name = "msm-dai-q6.16389",
1445 .platform_name = "msm-pcm-hostless",
1446 .codec_name = "tabla_codec",
1447 .codec_dai_name = "tabla_tx2",
Swaminathan Sathappan630c5a42012-05-10 17:33:01 -07001448 .ignore_suspend = 1,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001449 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1450 .ops = &msm8960_be_ops,
1451 },
Kuirong Wangb25838e2012-01-16 23:37:23 -08001452};
1453
1454static struct snd_soc_dai_link msm8960_tabla1x_dai[
1455 ARRAY_SIZE(msm8960_dai_common) +
1456 ARRAY_SIZE(msm8960_dai_delta_tabla1x)];
1457
1458
1459static struct snd_soc_dai_link msm8960_dai[
1460 ARRAY_SIZE(msm8960_dai_common) +
1461 ARRAY_SIZE(msm8960_dai_delta_tabla2x)];
1462
1463static struct snd_soc_card snd_soc_tabla1x_card_msm8960 = {
1464 .name = "msm8960-tabla1x-snd-card",
1465 .dai_link = msm8960_tabla1x_dai,
1466 .num_links = ARRAY_SIZE(msm8960_tabla1x_dai),
Steve Mucklef132c6c2012-06-06 18:30:57 -07001467 .controls = tabla_msm8960_controls,
1468 .num_controls = ARRAY_SIZE(tabla_msm8960_controls),
Kuirong Wangb25838e2012-01-16 23:37:23 -08001469};
1470
1471static struct snd_soc_card snd_soc_card_msm8960 = {
1472 .name = "msm8960-snd-card",
1473 .dai_link = msm8960_dai,
1474 .num_links = ARRAY_SIZE(msm8960_dai),
Steve Mucklef132c6c2012-06-06 18:30:57 -07001475 .controls = tabla_msm8960_controls,
1476 .num_controls = ARRAY_SIZE(tabla_msm8960_controls),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001477};
1478
1479static struct platform_device *msm8960_snd_device;
Kuirong Wangb25838e2012-01-16 23:37:23 -08001480static struct platform_device *msm8960_snd_tabla1x_device;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001481
1482static int msm8960_configure_headset_mic_gpios(void)
1483{
1484 int ret;
1485 struct pm_gpio param = {
1486 .direction = PM_GPIO_DIR_OUT,
1487 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
1488 .output_value = 1,
1489 .pull = PM_GPIO_PULL_NO,
1490 .vin_sel = PM_GPIO_VIN_S4,
1491 .out_strength = PM_GPIO_STRENGTH_MED,
1492 .function = PM_GPIO_FUNC_NORMAL,
1493 };
1494
1495 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(23), "AV_SWITCH");
1496 if (ret) {
1497 pr_err("%s: Failed to request gpio %d\n", __func__,
1498 PM8921_GPIO_PM_TO_SYS(23));
1499 return ret;
1500 }
1501
1502 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(23), &param);
1503 if (ret)
1504 pr_err("%s: Failed to configure gpio %d\n", __func__,
1505 PM8921_GPIO_PM_TO_SYS(23));
1506 else
1507 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(23), 0);
1508
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001509 ret = gpio_request(us_euro_sel_gpio, "US_EURO_SWITCH");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001510 if (ret) {
1511 pr_err("%s: Failed to request gpio %d\n", __func__,
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001512 us_euro_sel_gpio);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001513 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1514 return ret;
1515 }
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001516 ret = pm8xxx_gpio_config(us_euro_sel_gpio, &param);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001517 if (ret)
1518 pr_err("%s: Failed to configure gpio %d\n", __func__,
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001519 us_euro_sel_gpio);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001520 else
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001521 gpio_direction_output(us_euro_sel_gpio, 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001522
1523 return 0;
1524}
1525static void msm8960_free_headset_mic_gpios(void)
1526{
1527 if (msm8960_headset_gpios_configured) {
1528 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
Joonwoo Park2cc13f02012-05-09 12:44:25 -07001529 gpio_free(us_euro_sel_gpio);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001530 }
1531}
1532
1533static int __init msm8960_audio_init(void)
1534{
1535 int ret;
1536
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301537 if (!cpu_is_msm8960()) {
Steve Mucklef132c6c2012-06-06 18:30:57 -07001538 pr_debug("%s: Not the right machine type\n", __func__);
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001539 return -ENODEV ;
1540 }
Joonwoo Park0976d012011-12-22 11:48:18 -08001541
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001542 mbhc_cfg.calibration = def_tabla_mbhc_cal();
1543 if (!mbhc_cfg.calibration) {
Joonwoo Park0976d012011-12-22 11:48:18 -08001544 pr_err("Calibration data allocation failed\n");
1545 return -ENOMEM;
1546 }
1547
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001548 msm8960_snd_device = platform_device_alloc("soc-audio", 0);
1549 if (!msm8960_snd_device) {
1550 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001551 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001552 return -ENOMEM;
1553 }
1554
Kuirong Wangb25838e2012-01-16 23:37:23 -08001555 memcpy(msm8960_dai, msm8960_dai_common, sizeof(msm8960_dai_common));
1556 memcpy(msm8960_dai + ARRAY_SIZE(msm8960_dai_common),
1557 msm8960_dai_delta_tabla2x, sizeof(msm8960_dai_delta_tabla2x));
1558
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001559 platform_set_drvdata(msm8960_snd_device, &snd_soc_card_msm8960);
1560 ret = platform_device_add(msm8960_snd_device);
1561 if (ret) {
1562 platform_device_put(msm8960_snd_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001563 kfree(mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001564 return ret;
1565 }
1566
Kuirong Wangb25838e2012-01-16 23:37:23 -08001567 msm8960_snd_tabla1x_device = platform_device_alloc("soc-audio", 1);
1568 if (!msm8960_snd_tabla1x_device) {
1569 pr_err("Platform device allocation failed\n");
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001570 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001571 return -ENOMEM;
1572 }
1573
1574 memcpy(msm8960_tabla1x_dai, msm8960_dai_common,
1575 sizeof(msm8960_dai_common));
1576 memcpy(msm8960_tabla1x_dai + ARRAY_SIZE(msm8960_dai_common),
1577 msm8960_dai_delta_tabla1x, sizeof(msm8960_dai_delta_tabla1x));
1578
1579 platform_set_drvdata(msm8960_snd_tabla1x_device,
1580 &snd_soc_tabla1x_card_msm8960);
1581 ret = platform_device_add(msm8960_snd_tabla1x_device);
1582 if (ret) {
1583 platform_device_put(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001584 kfree(mbhc_cfg.calibration);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001585 return ret;
1586 }
1587
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001588 if (msm8960_configure_headset_mic_gpios()) {
1589 pr_err("%s Fail to configure headset mic gpios\n", __func__);
1590 msm8960_headset_gpios_configured = 0;
1591 } else
1592 msm8960_headset_gpios_configured = 1;
1593
Joonwoo Park28f49c82012-03-16 12:29:21 -07001594 mutex_init(&cdc_mclk_mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001595 return ret;
1596
1597}
1598module_init(msm8960_audio_init);
1599
1600static void __exit msm8960_audio_exit(void)
1601{
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301602 if (!cpu_is_msm8960()) {
Steve Mucklef132c6c2012-06-06 18:30:57 -07001603 pr_debug("%s: Not the right machine type\n", __func__);
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001604 return ;
1605 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001606 msm8960_free_headset_mic_gpios();
1607 platform_device_unregister(msm8960_snd_device);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001608 platform_device_unregister(msm8960_snd_tabla1x_device);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001609 kfree(mbhc_cfg.calibration);
Joonwoo Park28f49c82012-03-16 12:29:21 -07001610 mutex_destroy(&cdc_mclk_mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001611}
1612module_exit(msm8960_audio_exit);
1613
1614MODULE_DESCRIPTION("ALSA SoC MSM8960");
1615MODULE_LICENSE("GPL v2");