blob: a8aa05d3d7aad7346d43fff052944625304b2d6f [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>
18#include <linux/gpio.h>
19#include <linux/mfd/pm8xxx/pm8921.h>
Joonwoo Park0976d012011-12-22 11:48:18 -080020#include <linux/slab.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070021#include <sound/core.h>
22#include <sound/soc.h>
23#include <sound/soc-dapm.h>
24#include <sound/soc-dsp.h>
25#include <sound/pcm.h>
26#include <sound/jack.h>
Bradley Rubin229c6a52011-07-12 16:18:48 -070027#include <asm/mach-types.h>
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -080028#include <mach/socinfo.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
35#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
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -070043#define BTSCO_RATE_8KHZ 8000
44#define BTSCO_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
Harmandeep Singh0dd82412011-11-11 09:46:17 -080061static u32 top_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(18);
62static u32 bottom_spk_pamp_gpio = PM8921_GPIO_PM_TO_SYS(19);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070063static int msm8960_spk_control;
Kiran Kandi462acea2011-08-31 23:53:14 -070064static int msm8960_ext_bottom_spk_pamp;
65static int msm8960_ext_top_spk_pamp;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070066static int msm8960_slim_0_rx_ch = 1;
67static int msm8960_slim_0_tx_ch = 1;
68
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -070069static int msm8960_btsco_rate = BTSCO_RATE_8KHZ;
70static int msm8960_btsco_ch = 1;
71
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070072static struct clk *codec_clk;
73static int clk_users;
74
75static int msm8960_headset_gpios_configured;
76
77static struct snd_soc_jack hs_jack;
Bradley Rubincb1e2732011-06-23 16:49:20 -070078static struct snd_soc_jack button_jack;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070079
Joonwoo Park0976d012011-12-22 11:48:18 -080080static void *tabla_mbhc_cal;
81
Kiran Kandi462acea2011-08-31 23:53:14 -070082static void msm8960_enable_ext_spk_amp_gpio(u32 spk_amp_gpio)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070083{
84 int ret = 0;
85
86 struct pm_gpio param = {
87 .direction = PM_GPIO_DIR_OUT,
88 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
89 .output_value = 1,
90 .pull = PM_GPIO_PULL_NO,
91 .vin_sel = PM_GPIO_VIN_S4,
92 .out_strength = PM_GPIO_STRENGTH_MED,
Kiran Kandi462acea2011-08-31 23:53:14 -070093 .
94 function = PM_GPIO_FUNC_NORMAL,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070095 };
96
Harmandeep Singh0dd82412011-11-11 09:46:17 -080097 if (spk_amp_gpio == bottom_spk_pamp_gpio) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070098
Harmandeep Singh0dd82412011-11-11 09:46:17 -080099 ret = gpio_request(bottom_spk_pamp_gpio, "BOTTOM_SPK_AMP");
Kiran Kandi462acea2011-08-31 23:53:14 -0700100 if (ret) {
101 pr_err("%s: Error requesting BOTTOM SPK AMP GPIO %u\n",
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800102 __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700103 return;
104 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800105 ret = pm8xxx_gpio_config(bottom_spk_pamp_gpio, &param);
Kiran Kandi462acea2011-08-31 23:53:14 -0700106 if (ret)
107 pr_err("%s: Failed to configure Bottom Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800108 " gpio %u\n", __func__, bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700109 else {
110 pr_debug("%s: enable Bottom spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800111 gpio_direction_output(bottom_spk_pamp_gpio, 1);
Kiran Kandi462acea2011-08-31 23:53:14 -0700112 }
113
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800114 } else if (spk_amp_gpio == top_spk_pamp_gpio) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700115
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800116 ret = gpio_request(top_spk_pamp_gpio, "TOP_SPK_AMP");
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700117 if (ret) {
118 pr_err("%s: Error requesting GPIO %d\n", __func__,
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800119 top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700120 return;
121 }
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800122 ret = pm8xxx_gpio_config(top_spk_pamp_gpio, &param);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700123 if (ret)
Kiran Kandi462acea2011-08-31 23:53:14 -0700124 pr_err("%s: Failed to configure Top Spk Ampl"
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800125 " gpio %u\n", __func__, top_spk_pamp_gpio);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700126 else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700127 pr_debug("%s: enable Top spkr amp gpio\n", __func__);
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800128 gpio_direction_output(top_spk_pamp_gpio, 1);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700129 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700130 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700131 pr_err("%s: ERROR : Invalid External Speaker Ampl GPIO."
132 " gpio = %u\n", __func__, spk_amp_gpio);
133 return;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700134 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700135}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700136
Kiran Kandi462acea2011-08-31 23:53:14 -0700137static void msm8960_ext_spk_power_amp_on(u32 spk)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700138{
Kiran Kandi462acea2011-08-31 23:53:14 -0700139 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
140
141 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
142 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
143
144 pr_debug("%s() External Bottom Speaker Ampl already "
145 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700146 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700147 }
148
149 msm8960_ext_bottom_spk_pamp |= spk;
150
151 if ((msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_POS) &&
152 (msm8960_ext_bottom_spk_pamp & BOTTOM_SPK_AMP_NEG)) {
153
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800154 msm8960_enable_ext_spk_amp_gpio(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700155 pr_debug("%s: slepping 4 ms after turning on external "
156 " Bottom Speaker Ampl\n", __func__);
157 usleep_range(4000, 4000);
158 }
159
160 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
161
162 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
163 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
164
165 pr_debug("%s() External Top Speaker Ampl already"
166 "turned on. spk = 0x%08x\n", __func__, spk);
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700167 return;
Kiran Kandi462acea2011-08-31 23:53:14 -0700168 }
169
170 msm8960_ext_top_spk_pamp |= spk;
171
172 if ((msm8960_ext_top_spk_pamp & TOP_SPK_AMP_POS) &&
173 (msm8960_ext_top_spk_pamp & TOP_SPK_AMP_NEG)) {
174
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800175 msm8960_enable_ext_spk_amp_gpio(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700176 pr_debug("%s: sleeping 4 ms after turning on "
177 " external Top Speaker Ampl\n", __func__);
178 usleep_range(4000, 4000);
179 }
180 } else {
181
182 pr_err("%s: ERROR : Invalid External Speaker Ampl. spk = 0x%08x\n",
183 __func__, spk);
184 return;
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700185 }
Kiran Kandi462acea2011-08-31 23:53:14 -0700186}
187
188static void msm8960_ext_spk_power_amp_off(u32 spk)
189{
190 if (spk & (BOTTOM_SPK_AMP_POS | BOTTOM_SPK_AMP_NEG)) {
191
192 if (!msm8960_ext_bottom_spk_pamp)
193 return;
194
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800195 gpio_direction_output(bottom_spk_pamp_gpio, 0);
196 gpio_free(bottom_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700197 msm8960_ext_bottom_spk_pamp = 0;
198
199 pr_debug("%s: sleeping 4 ms after turning off external Bottom"
200 " Speaker Ampl\n", __func__);
201
202 usleep_range(4000, 4000);
203
204 } else if (spk & (TOP_SPK_AMP_POS | TOP_SPK_AMP_NEG)) {
205
206 if (!msm8960_ext_top_spk_pamp)
207 return;
208
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800209 gpio_direction_output(top_spk_pamp_gpio, 0);
210 gpio_free(top_spk_pamp_gpio);
Kiran Kandi462acea2011-08-31 23:53:14 -0700211 msm8960_ext_top_spk_pamp = 0;
212
213 pr_debug("%s: sleeping 4 ms after turning off external Top"
214 " Spkaker Ampl\n", __func__);
215
216 usleep_range(4000, 4000);
217 } else {
218
219 pr_err("%s: ERROR : Invalid Ext Spk Ampl. spk = 0x%08x\n",
220 __func__, spk);
221 return;
222 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700223}
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700224
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700225static void msm8960_ext_control(struct snd_soc_codec *codec)
226{
227 struct snd_soc_dapm_context *dapm = &codec->dapm;
228
229 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700230 if (msm8960_spk_control == MSM8960_SPK_ON) {
231 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
232 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
233 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
234 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
235 } else {
236 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Pos");
237 snd_soc_dapm_disable_pin(dapm, "Ext Spk Bottom Neg");
238 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Pos");
239 snd_soc_dapm_disable_pin(dapm, "Ext Spk Top Neg");
240 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700241
242 snd_soc_dapm_sync(dapm);
243}
244
245static int msm8960_get_spk(struct snd_kcontrol *kcontrol,
246 struct snd_ctl_elem_value *ucontrol)
247{
248 pr_debug("%s: msm8960_spk_control = %d", __func__, msm8960_spk_control);
249 ucontrol->value.integer.value[0] = msm8960_spk_control;
250 return 0;
251}
252static int msm8960_set_spk(struct snd_kcontrol *kcontrol,
253 struct snd_ctl_elem_value *ucontrol)
254{
255 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
256
257 pr_debug("%s()\n", __func__);
258 if (msm8960_spk_control == ucontrol->value.integer.value[0])
259 return 0;
260
261 msm8960_spk_control = ucontrol->value.integer.value[0];
262 msm8960_ext_control(codec);
263 return 1;
264}
265static int msm8960_spkramp_event(struct snd_soc_dapm_widget *w,
266 struct snd_kcontrol *k, int event)
267{
268 pr_debug("%s() %x\n", __func__, SND_SOC_DAPM_EVENT_ON(event));
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700269
270 if (SND_SOC_DAPM_EVENT_ON(event)) {
Kiran Kandi462acea2011-08-31 23:53:14 -0700271 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
272 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_POS);
273 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
274 msm8960_ext_spk_power_amp_on(BOTTOM_SPK_AMP_NEG);
275 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
276 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_POS);
277 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
278 msm8960_ext_spk_power_amp_on(TOP_SPK_AMP_NEG);
279 else {
280 pr_err("%s() Invalid Speaker Widget = %s\n",
281 __func__, w->name);
282 return -EINVAL;
283 }
284
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700285 } else {
Kiran Kandi462acea2011-08-31 23:53:14 -0700286 if (!strncmp(w->name, "Ext Spk Bottom Pos", 18))
287 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_POS);
288 else if (!strncmp(w->name, "Ext Spk Bottom Neg", 18))
289 msm8960_ext_spk_power_amp_off(BOTTOM_SPK_AMP_NEG);
290 else if (!strncmp(w->name, "Ext Spk Top Pos", 15))
291 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_POS);
292 else if (!strncmp(w->name, "Ext Spk Top Neg", 15))
293 msm8960_ext_spk_power_amp_off(TOP_SPK_AMP_NEG);
294 else {
295 pr_err("%s() Invalid Speaker Widget = %s\n",
296 __func__, w->name);
297 return -EINVAL;
298 }
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700299 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700300 return 0;
301}
Joonwoo Park0976d012011-12-22 11:48:18 -0800302int msm8960_enable_codec_ext_clk(
303 struct snd_soc_codec *codec, int enable)
304{
305 pr_debug("%s: enable = %d\n", __func__, enable);
306 if (enable) {
307 clk_users++;
308 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
309 if (clk_users != 1)
310 return 0;
311
Joonwoo Park0976d012011-12-22 11:48:18 -0800312 if (codec_clk) {
313 clk_set_rate(codec_clk, TABLA_EXT_CLK_RATE);
314 clk_enable(codec_clk);
315 tabla_mclk_enable(codec, 1);
316 } else {
317 pr_err("%s: Error setting Tabla MCLK\n", __func__);
318 clk_users--;
319 return -EINVAL;
320 }
321 } else {
322 pr_debug("%s: clk_users = %d\n", __func__, clk_users);
323 if (clk_users == 0)
324 return 0;
325 clk_users--;
326 if (!clk_users) {
327 pr_debug("%s: disabling MCLK. clk_users = %d\n",
328 __func__, clk_users);
329 clk_disable(codec_clk);
Joonwoo Park0976d012011-12-22 11:48:18 -0800330 tabla_mclk_enable(codec, 0);
331 }
332 }
333 return 0;
334}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700335
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700336static int msm8960_mclk_event(struct snd_soc_dapm_widget *w,
337 struct snd_kcontrol *kcontrol, int event)
338{
339 pr_debug("%s: event = %d\n", __func__, event);
340
341 switch (event) {
342 case SND_SOC_DAPM_PRE_PMU:
Joonwoo Park0976d012011-12-22 11:48:18 -0800343 return msm8960_enable_codec_ext_clk(w->codec, 1);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700344 case SND_SOC_DAPM_POST_PMD:
Joonwoo Park0976d012011-12-22 11:48:18 -0800345 return msm8960_enable_codec_ext_clk(w->codec, 0);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700346 }
347 return 0;
348}
349
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700350static const struct snd_soc_dapm_widget msm8960_dapm_widgets[] = {
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700351
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700352 SND_SOC_DAPM_SUPPLY("MCLK", SND_SOC_NOPM, 0, 0,
353 msm8960_mclk_event, SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
354
Kiran Kandi462acea2011-08-31 23:53:14 -0700355 SND_SOC_DAPM_SPK("Ext Spk Bottom Pos", msm8960_spkramp_event),
356 SND_SOC_DAPM_SPK("Ext Spk Bottom Neg", msm8960_spkramp_event),
357
358 SND_SOC_DAPM_SPK("Ext Spk Top Pos", msm8960_spkramp_event),
359 SND_SOC_DAPM_SPK("Ext Spk Top Neg", msm8960_spkramp_event),
Kiran Kandi3a30bda2011-08-15 10:36:42 -0700360
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700361 SND_SOC_DAPM_MIC("Handset Mic", NULL),
362 SND_SOC_DAPM_MIC("Headset Mic", NULL),
363 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700364 SND_SOC_DAPM_MIC("ANCRight Headset Mic", NULL),
365 SND_SOC_DAPM_MIC("ANCLeft Headset Mic", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700366
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700367 SND_SOC_DAPM_MIC("Digital Mic1", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700368 SND_SOC_DAPM_MIC("Digital Mic2", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700369 SND_SOC_DAPM_MIC("Digital Mic3", NULL),
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700370 SND_SOC_DAPM_MIC("Digital Mic4", NULL),
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700371 SND_SOC_DAPM_MIC("Digital Mic5", NULL),
372 SND_SOC_DAPM_MIC("Digital Mic6", NULL),
373
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700374};
375
Bradley Rubin229c6a52011-07-12 16:18:48 -0700376static const struct snd_soc_dapm_route common_audio_map[] = {
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700377
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700378 {"RX_BIAS", NULL, "MCLK"},
379 {"LDO_H", NULL, "MCLK"},
380
381 /* Speaker path */
Kiran Kandi462acea2011-08-31 23:53:14 -0700382 {"Ext Spk Bottom Pos", NULL, "LINEOUT1"},
383 {"Ext Spk Bottom Neg", NULL, "LINEOUT3"},
Kiran Kandidb0a4b02011-08-23 09:32:09 -0700384
Kiran Kandi462acea2011-08-31 23:53:14 -0700385 {"Ext Spk Top Pos", NULL, "LINEOUT2"},
386 {"Ext Spk Top Neg", NULL, "LINEOUT4"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700387
388 /* Microphone path */
Bradley Rubin229c6a52011-07-12 16:18:48 -0700389 {"AMIC1", NULL, "MIC BIAS1 Internal1"},
390 {"MIC BIAS1 Internal1", NULL, "Handset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700391
392 {"AMIC2", NULL, "MIC BIAS2 External"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700393 {"MIC BIAS2 External", NULL, "Headset Mic"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700394
Patrick Laie519dd62011-09-28 12:37:11 -0700395 /**
396 * AMIC3 and AMIC4 inputs are connected to ANC microphones
397 * These mics are biased differently on CDP and FLUID
398 * routing entries below are based on bias arrangement
399 * on FLUID.
400 */
401 {"AMIC3", NULL, "MIC BIAS3 Internal1"},
402 {"MIC BIAS3 Internal1", NULL, "ANCRight Headset Mic"},
403
404 {"AMIC4", NULL, "MIC BIAS1 Internal2"},
405 {"MIC BIAS1 Internal2", NULL, "ANCLeft Headset Mic"},
406
Kiran Kandie9bf86a2011-07-21 16:50:41 -0700407 {"HEADPHONE", NULL, "LDO_H"},
408
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700409 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700410 * The digital Mic routes are setup considering
411 * fluid as default device.
412 */
413
414 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700415 * Digital Mic1. Front Bottom left Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700416 * Digital Mic GM5 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700417 * Conncted to DMIC2 Input on Tabla codec.
418 */
419 {"DMIC2", NULL, "MIC BIAS1 External"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -0700420 {"MIC BIAS1 External", NULL, "Digital Mic1"},
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700421
422 /**
423 * Digital Mic2. Front Bottom right Digital Mic on Fluid and MTP.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700424 * Digital Mic GM6 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700425 * Conncted to DMIC1 Input on Tabla codec.
426 */
427 {"DMIC1", NULL, "MIC BIAS1 External"},
428 {"MIC BIAS1 External", NULL, "Digital Mic2"},
429
430 /**
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700431 * Digital Mic3. Back Bottom Digital Mic on Fluid.
432 * Digital Mic GM1 on CDP mainboard.
433 * Conncted to DMIC4 Input on Tabla codec.
434 */
435 {"DMIC4", NULL, "MIC BIAS3 External"},
436 {"MIC BIAS3 External", NULL, "Digital Mic3"},
437
438 /**
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700439 * Digital Mic4. Back top Digital Mic on Fluid.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700440 * Digital Mic GM2 on CDP mainboard.
Kiran Kandia21d6bc2011-07-17 21:19:59 -0700441 * Conncted to DMIC3 Input on Tabla codec.
442 */
443 {"DMIC3", NULL, "MIC BIAS3 External"},
444 {"MIC BIAS3 External", NULL, "Digital Mic4"},
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700445
446 /**
447 * Digital Mic5. Front top Digital Mic on Fluid.
448 * Digital Mic GM3 on CDP mainboard.
449 * Conncted to DMIC5 Input on Tabla codec.
450 */
451 {"DMIC5", NULL, "MIC BIAS4 External"},
452 {"MIC BIAS4 External", NULL, "Digital Mic5"},
453
Patrick Laicb7802b2011-10-04 12:39:18 -0700454 /* Tabla digital Mic6 - back bottom digital Mic on Liquid and
455 * bottom mic on CDP. FLUID/MTP do not have dmic6 installed.
Bhalchandra Gajarea0101ba2011-08-11 20:05:56 -0700456 */
457 {"DMIC6", NULL, "MIC BIAS4 External"},
458 {"MIC BIAS4 External", NULL, "Digital Mic6"},
Bradley Rubin229c6a52011-07-12 16:18:48 -0700459};
460
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700461static const char *spk_function[] = {"Off", "On"};
Patrick Lai9f4b4292011-07-16 22:11:09 -0700462static const char *slim0_rx_ch_text[] = {"One", "Two"};
463static const char *slim0_tx_ch_text[] = {"One", "Two", "Three", "Four"};
464
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700465static const struct soc_enum msm8960_enum[] = {
466 SOC_ENUM_SINGLE_EXT(2, spk_function),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700467 SOC_ENUM_SINGLE_EXT(2, slim0_rx_ch_text),
468 SOC_ENUM_SINGLE_EXT(4, slim0_tx_ch_text),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700469};
470
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700471static const char *btsco_rate_text[] = {"8000", "16000"};
472static const struct soc_enum msm8960_btsco_enum[] = {
473 SOC_ENUM_SINGLE_EXT(2, btsco_rate_text),
474};
475
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700476static int msm8960_slim_0_rx_ch_get(struct snd_kcontrol *kcontrol,
477 struct snd_ctl_elem_value *ucontrol)
478{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700479 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800480 msm8960_slim_0_rx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700481 ucontrol->value.integer.value[0] = msm8960_slim_0_rx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700482 return 0;
483}
484
485static int msm8960_slim_0_rx_ch_put(struct snd_kcontrol *kcontrol,
486 struct snd_ctl_elem_value *ucontrol)
487{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700488 msm8960_slim_0_rx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700489
Patrick Lai9f4b4292011-07-16 22:11:09 -0700490 pr_debug("%s: msm8960_slim_0_rx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800491 msm8960_slim_0_rx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700492 return 1;
493}
494
495static int msm8960_slim_0_tx_ch_get(struct snd_kcontrol *kcontrol,
496 struct snd_ctl_elem_value *ucontrol)
497{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700498 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800499 msm8960_slim_0_tx_ch);
Patrick Lai9f4b4292011-07-16 22:11:09 -0700500 ucontrol->value.integer.value[0] = msm8960_slim_0_tx_ch - 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700501 return 0;
502}
503
504static int msm8960_slim_0_tx_ch_put(struct snd_kcontrol *kcontrol,
505 struct snd_ctl_elem_value *ucontrol)
506{
Patrick Lai9f4b4292011-07-16 22:11:09 -0700507 msm8960_slim_0_tx_ch = ucontrol->value.integer.value[0] + 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700508
509 pr_debug("%s: msm8960_slim_0_tx_ch = %d\n", __func__,
Joonwoo Park0976d012011-12-22 11:48:18 -0800510 msm8960_slim_0_tx_ch);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700511 return 1;
512}
513
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700514static int msm8960_btsco_rate_get(struct snd_kcontrol *kcontrol,
515 struct snd_ctl_elem_value *ucontrol)
516{
Joonwoo Park0976d012011-12-22 11:48:18 -0800517 pr_debug("%s: msm8960_btsco_rate = %d", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700518 ucontrol->value.integer.value[0] = msm8960_btsco_rate;
519 return 0;
520}
521
522static int msm8960_btsco_rate_put(struct snd_kcontrol *kcontrol,
523 struct snd_ctl_elem_value *ucontrol)
524{
525 switch (ucontrol->value.integer.value[0]) {
526 case 0:
527 msm8960_btsco_rate = BTSCO_RATE_8KHZ;
528 break;
529 case 1:
530 msm8960_btsco_rate = BTSCO_RATE_16KHZ;
531 break;
532 default:
533 msm8960_btsco_rate = BTSCO_RATE_8KHZ;
534 break;
535 }
Joonwoo Park0976d012011-12-22 11:48:18 -0800536 pr_debug("%s: msm8960_btsco_rate = %d\n", __func__, msm8960_btsco_rate);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700537 return 0;
538}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700539
540static const struct snd_kcontrol_new tabla_msm8960_controls[] = {
541 SOC_ENUM_EXT("Speaker Function", msm8960_enum[0], msm8960_get_spk,
542 msm8960_set_spk),
Patrick Lai9f4b4292011-07-16 22:11:09 -0700543 SOC_ENUM_EXT("SLIM_0_RX Channels", msm8960_enum[1],
544 msm8960_slim_0_rx_ch_get, msm8960_slim_0_rx_ch_put),
545 SOC_ENUM_EXT("SLIM_0_TX Channels", msm8960_enum[2],
546 msm8960_slim_0_tx_ch_get, msm8960_slim_0_tx_ch_put),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700547};
548
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700549static const struct snd_kcontrol_new int_btsco_rate_mixer_controls[] = {
550 SOC_ENUM_EXT("Internal BTSCO SampleRate", msm8960_btsco_enum[0],
551 msm8960_btsco_rate_get, msm8960_btsco_rate_put),
552};
553
554static int msm8960_btsco_init(struct snd_soc_pcm_runtime *rtd)
555{
556 int err = 0;
557 struct snd_soc_platform *platform = rtd->platform;
558
559 err = snd_soc_add_platform_controls(platform,
560 int_btsco_rate_mixer_controls,
561 ARRAY_SIZE(int_btsco_rate_mixer_controls));
562 if (err < 0)
563 return err;
564 return 0;
565}
566
Joonwoo Park0976d012011-12-22 11:48:18 -0800567static void *def_tabla_mbhc_cal(void)
568{
569 void *tabla_cal;
570 struct tabla_mbhc_btn_detect_cfg *btn_cfg;
571 u16 *btn_low, *btn_high;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800572 u8 *n_ready, *n_cic, *gain;
Joonwoo Park0976d012011-12-22 11:48:18 -0800573
574 tabla_cal = kzalloc(TABLA_MBHC_CAL_SIZE(TABLA_MBHC_DEF_BUTTONS,
575 TABLA_MBHC_DEF_RLOADS),
576 GFP_KERNEL);
577 if (!tabla_cal) {
578 pr_err("%s: out of memory\n", __func__);
579 return NULL;
580 }
581
582#define S(X, Y) ((TABLA_MBHC_CAL_GENERAL_PTR(tabla_cal)->X) = (Y))
583 S(t_ldoh, 100);
584 S(t_bg_fast_settle, 100);
585 S(t_shutdown_plug_rem, 255);
586 S(mbhc_nsa, 4);
587 S(mbhc_navg, 4);
588#undef S
589#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_DET_PTR(tabla_cal)->X) = (Y))
590 S(mic_current, TABLA_PID_MIC_5_UA);
591 S(hph_current, TABLA_PID_MIC_5_UA);
592 S(t_mic_pid, 100);
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800593 S(t_ins_complete, 250);
Joonwoo Park0976d012011-12-22 11:48:18 -0800594 S(t_ins_retry, 200);
595#undef S
596#define S(X, Y) ((TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla_cal)->X) = (Y))
597 S(v_no_mic, 30);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800598 S(v_hs_max, 1550);
Joonwoo Park0976d012011-12-22 11:48:18 -0800599#undef S
600#define S(X, Y) ((TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal)->X) = (Y))
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800601 S(c[0], 62);
602 S(c[1], 124);
603 S(nc, 1);
604 S(n_meas, 3);
Joonwoo Park0976d012011-12-22 11:48:18 -0800605 S(mbhc_nsc, 11);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800606 S(n_btn_meas, 1);
607 S(n_btn_con, 2);
Joonwoo Park0976d012011-12-22 11:48:18 -0800608 S(num_btn, TABLA_MBHC_DEF_BUTTONS);
609 S(v_btn_press_delta_sta, 100);
610 S(v_btn_press_delta_cic, 50);
611#undef S
612 btn_cfg = TABLA_MBHC_CAL_BTN_DET_PTR(tabla_cal);
613 btn_low = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_LOW);
614 btn_high = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_V_BTN_HIGH);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800615 btn_low[0] = -50;
616 btn_high[0] = 10;
617 btn_low[1] = 11;
618 btn_high[1] = 38;
619 btn_low[2] = 39;
620 btn_high[2] = 64;
621 btn_low[3] = 65;
622 btn_high[3] = 91;
623 btn_low[4] = 92;
624 btn_high[4] = 115;
625 btn_low[5] = 116;
626 btn_high[5] = 141;
627 btn_low[6] = 142;
628 btn_high[6] = 163;
629 btn_low[7] = 164;
630 btn_high[7] = 250;
Joonwoo Parkc0672392012-01-11 11:03:14 -0800631 n_ready = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_READY);
632 n_ready[0] = 48;
633 n_ready[1] = 38;
Joonwoo Park0976d012011-12-22 11:48:18 -0800634 n_cic = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_N_CIC);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800635 n_cic[0] = 60;
636 n_cic[1] = 47;
Joonwoo Park0976d012011-12-22 11:48:18 -0800637 gain = tabla_mbhc_cal_btn_det_mp(btn_cfg, TABLA_BTN_DET_GAIN);
638 gain[0] = 11;
639 gain[1] = 9;
640
641 return tabla_cal;
642}
643
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800644static int msm8960_hw_params(struct snd_pcm_substream *substream,
645 struct snd_pcm_hw_params *params)
646{
647 struct snd_soc_pcm_runtime *rtd = substream->private_data;
648 struct snd_soc_dai *codec_dai = rtd->codec_dai;
649 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
650 int ret = 0;
651 unsigned int rx_ch[SLIM_MAX_RX_PORTS], tx_ch[SLIM_MAX_TX_PORTS];
652 unsigned int rx_ch_cnt = 0, tx_ch_cnt = 0;
653
654 pr_debug("%s: ch=%d\n", __func__,
655 msm8960_slim_0_rx_ch);
656 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
657 ret = snd_soc_dai_get_channel_map(codec_dai,
658 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
659 if (ret < 0) {
660 pr_err("%s: failed to get codec chan map\n", __func__);
661 goto end;
662 }
663
664 ret = snd_soc_dai_set_channel_map(cpu_dai, 0, 0,
665 msm8960_slim_0_rx_ch, rx_ch);
666 if (ret < 0) {
667 pr_err("%s: failed to set cpu chan map\n", __func__);
668 goto end;
669 }
670 ret = snd_soc_dai_set_channel_map(codec_dai, 0, 0,
671 msm8960_slim_0_rx_ch, rx_ch);
672 if (ret < 0) {
673 pr_err("%s: failed to set codec channel map\n",
674 __func__);
675 goto end;
676 }
677 } else {
678 ret = snd_soc_dai_get_channel_map(codec_dai,
679 &tx_ch_cnt, tx_ch, &rx_ch_cnt , rx_ch);
680 if (ret < 0) {
681 pr_err("%s: failed to get codec chan map\n", __func__);
682 goto end;
683 }
684 ret = snd_soc_dai_set_channel_map(cpu_dai,
685 msm8960_slim_0_tx_ch, tx_ch, 0 , 0);
686 if (ret < 0) {
687 pr_err("%s: failed to set cpu chan map\n", __func__);
688 goto end;
689 }
690 ret = snd_soc_dai_set_channel_map(codec_dai,
691 msm8960_slim_0_tx_ch, tx_ch, 0, 0);
692 if (ret < 0) {
693 pr_err("%s: failed to set codec channel map\n",
694 __func__);
695 goto end;
696 }
697
698
699 }
700end:
701 return ret;
702}
703
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700704static int msm8960_audrx_init(struct snd_soc_pcm_runtime *rtd)
705{
706 int err;
707 struct snd_soc_codec *codec = rtd->codec;
708 struct snd_soc_dapm_context *dapm = &codec->dapm;
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800709 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700710
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800711 pr_debug("%s(), dev_name%s\n", __func__, dev_name(cpu_dai->dev));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700712
Harmandeep Singh0dd82412011-11-11 09:46:17 -0800713 if (machine_is_msm8960_liquid()) {
714 top_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(19));
715 bottom_spk_pamp_gpio = (PM8921_GPIO_PM_TO_SYS(18));
716 }
717
Kiran Kandi28ef14e2011-09-16 15:45:47 -0700718 rtd->pmdown_time = 0;
719
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700720 err = snd_soc_add_controls(codec, tabla_msm8960_controls,
721 ARRAY_SIZE(tabla_msm8960_controls));
722 if (err < 0)
723 return err;
724
725 snd_soc_dapm_new_controls(dapm, msm8960_dapm_widgets,
726 ARRAY_SIZE(msm8960_dapm_widgets));
727
Bradley Rubin229c6a52011-07-12 16:18:48 -0700728 snd_soc_dapm_add_routes(dapm, common_audio_map,
729 ARRAY_SIZE(common_audio_map));
730
Peter Lohmannb8203ef2011-10-07 15:05:28 -0700731 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Pos");
732 snd_soc_dapm_enable_pin(dapm, "Ext Spk Bottom Neg");
733 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Pos");
734 snd_soc_dapm_enable_pin(dapm, "Ext Spk Top Neg");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700735
736 snd_soc_dapm_sync(dapm);
737
738 err = snd_soc_jack_new(codec, "Headset Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800739 (SND_JACK_HEADSET | SND_JACK_OC_HPHL |
740 SND_JACK_OC_HPHR),
741 &hs_jack);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700742 if (err) {
743 pr_err("failed to create new jack\n");
744 return err;
745 }
Bradley Rubincb1e2732011-06-23 16:49:20 -0700746
747 err = snd_soc_jack_new(codec, "Button Jack",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800748 TABLA_JACK_BUTTON_MASK, &button_jack);
Bradley Rubincb1e2732011-06-23 16:49:20 -0700749 if (err) {
750 pr_err("failed to create new jack\n");
751 return err;
752 }
753
Kuirong Wanga9c3acc2012-02-09 17:00:45 -0800754 codec_clk = clk_get(cpu_dai->dev, "osr_clk");
755
Joonwoo Park03324832012-03-19 19:36:16 -0700756 err = tabla_hs_detect(codec, &hs_jack, &button_jack, tabla_mbhc_cal,
757 TABLA_MICBIAS2, msm8960_enable_codec_ext_clk, 0,
758 TABLA_EXT_CLK_RATE);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700759
Joonwoo Park03324832012-03-19 19:36:16 -0700760 return err;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700761}
762
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700763static struct snd_soc_dsp_link lpa_fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700764 .playback = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700765 .trigger = {
766 SND_SOC_DSP_TRIGGER_POST,
767 SND_SOC_DSP_TRIGGER_POST
768 },
769};
770
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700771static struct snd_soc_dsp_link fe_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700772 .playback = true,
773 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700774 .trigger = {
775 SND_SOC_DSP_TRIGGER_POST,
776 SND_SOC_DSP_TRIGGER_POST
777 },
778};
779
Patrick Lai9b56e1d2012-01-22 22:14:15 -0800780/* bi-directional media definition for hostless PCM device */
781static struct snd_soc_dsp_link bidir_hl_media = {
Patrick Lai5b3fdfc2011-09-01 11:04:56 -0700782 .playback = true,
783 .capture = true,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700784 .trigger = {
785 SND_SOC_DSP_TRIGGER_POST,
786 SND_SOC_DSP_TRIGGER_POST
787 },
788};
789
Alex Wongb3d06a02012-01-12 10:00:41 -0800790static struct snd_soc_dsp_link hdmi_rx_hl = {
791 .playback = true,
792 .trigger = {
793 SND_SOC_DSP_TRIGGER_POST,
794 SND_SOC_DSP_TRIGGER_POST
795 },
796};
797
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700798static int msm8960_slim_0_rx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
799 struct snd_pcm_hw_params *params)
800{
801 struct snd_interval *rate = hw_param_interval(params,
802 SNDRV_PCM_HW_PARAM_RATE);
803
804 struct snd_interval *channels = hw_param_interval(params,
805 SNDRV_PCM_HW_PARAM_CHANNELS);
806
807 pr_debug("%s()\n", __func__);
808 rate->min = rate->max = 48000;
809 channels->min = channels->max = msm8960_slim_0_rx_ch;
810
811 return 0;
812}
813
814static int msm8960_slim_0_tx_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
815 struct snd_pcm_hw_params *params)
816{
817 struct snd_interval *rate = hw_param_interval(params,
818 SNDRV_PCM_HW_PARAM_RATE);
819
820 struct snd_interval *channels = hw_param_interval(params,
821 SNDRV_PCM_HW_PARAM_CHANNELS);
822
823 pr_debug("%s()\n", __func__);
824 rate->min = rate->max = 48000;
825 channels->min = channels->max = msm8960_slim_0_tx_ch;
826
827 return 0;
828}
829
830static int msm8960_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
831 struct snd_pcm_hw_params *params)
832{
833 struct snd_interval *rate = hw_param_interval(params,
834 SNDRV_PCM_HW_PARAM_RATE);
835
836 pr_debug("%s()\n", __func__);
837 rate->min = rate->max = 48000;
838
839 return 0;
840}
841
Helen Zeng73f7fc62011-11-18 16:29:59 -0800842static int msm8960_hdmi_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
843 struct snd_pcm_hw_params *params)
844{
845 struct snd_interval *rate = hw_param_interval(params,
846 SNDRV_PCM_HW_PARAM_RATE);
847
848 struct snd_interval *channels = hw_param_interval(params,
849 SNDRV_PCM_HW_PARAM_CHANNELS);
850
Kiran Kandi5e809b02012-01-31 00:24:33 -0800851 pr_debug("%s channels->min %u channels->max %u ()\n", __func__,
852 channels->min, channels->max);
853
Helen Zeng73f7fc62011-11-18 16:29:59 -0800854 rate->min = rate->max = 48000;
Helen Zeng73f7fc62011-11-18 16:29:59 -0800855
856 return 0;
857}
858
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700859static int msm8960_btsco_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
860 struct snd_pcm_hw_params *params)
861{
862 struct snd_interval *rate = hw_param_interval(params,
863 SNDRV_PCM_HW_PARAM_RATE);
864
865 struct snd_interval *channels = hw_param_interval(params,
866 SNDRV_PCM_HW_PARAM_CHANNELS);
867
868 rate->min = rate->max = msm8960_btsco_rate;
869 channels->min = channels->max = msm8960_btsco_ch;
870
871 return 0;
872}
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700873static int msm8960_auxpcm_be_params_fixup(struct snd_soc_pcm_runtime *rtd,
874 struct snd_pcm_hw_params *params)
875{
876 struct snd_interval *rate = hw_param_interval(params,
877 SNDRV_PCM_HW_PARAM_RATE);
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -0700878
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700879 struct snd_interval *channels = hw_param_interval(params,
880 SNDRV_PCM_HW_PARAM_CHANNELS);
881
882 /* PCM only supports mono output with 8khz sample rate */
883 rate->min = rate->max = 8000;
884 channels->min = channels->max = 1;
885
886 return 0;
887}
888static int msm8960_aux_pcm_get_gpios(void)
889{
890 int ret = 0;
891
892 pr_debug("%s\n", __func__);
893
894 ret = gpio_request(GPIO_AUX_PCM_DOUT, "AUX PCM DOUT");
895 if (ret < 0) {
896 pr_err("%s: Failed to request gpio(%d): AUX PCM DOUT",
897 __func__, GPIO_AUX_PCM_DOUT);
898 goto fail_dout;
899 }
900
901 ret = gpio_request(GPIO_AUX_PCM_DIN, "AUX PCM DIN");
902 if (ret < 0) {
903 pr_err("%s: Failed to request gpio(%d): AUX PCM DIN",
904 __func__, GPIO_AUX_PCM_DIN);
905 goto fail_din;
906 }
907
908 ret = gpio_request(GPIO_AUX_PCM_SYNC, "AUX PCM SYNC");
909 if (ret < 0) {
910 pr_err("%s: Failed to request gpio(%d): AUX PCM SYNC",
911 __func__, GPIO_AUX_PCM_SYNC);
912 goto fail_sync;
913 }
914 ret = gpio_request(GPIO_AUX_PCM_CLK, "AUX PCM CLK");
915 if (ret < 0) {
916 pr_err("%s: Failed to request gpio(%d): AUX PCM CLK",
917 __func__, GPIO_AUX_PCM_CLK);
918 goto fail_clk;
919 }
920
921 return 0;
922
923fail_clk:
924 gpio_free(GPIO_AUX_PCM_SYNC);
925fail_sync:
926 gpio_free(GPIO_AUX_PCM_DIN);
927fail_din:
928 gpio_free(GPIO_AUX_PCM_DOUT);
929fail_dout:
930
931 return ret;
932}
933
934static int msm8960_aux_pcm_free_gpios(void)
935{
936 gpio_free(GPIO_AUX_PCM_DIN);
937 gpio_free(GPIO_AUX_PCM_DOUT);
938 gpio_free(GPIO_AUX_PCM_SYNC);
939 gpio_free(GPIO_AUX_PCM_CLK);
940
941 return 0;
942}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700943static int msm8960_startup(struct snd_pcm_substream *substream)
944{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700945 pr_debug("%s(): substream = %s stream = %d\n", __func__,
946 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700947 return 0;
948}
949
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700950static int msm8960_auxpcm_startup(struct snd_pcm_substream *substream)
951{
952 int ret = 0;
953
954 pr_debug("%s(): substream = %s\n", __func__, substream->name);
955 ret = msm8960_aux_pcm_get_gpios();
956 if (ret < 0) {
957 pr_err("%s: Aux PCM GPIO request failed\n", __func__);
958 return -EINVAL;
959 }
960 return 0;
961}
962
963static void msm8960_auxpcm_shutdown(struct snd_pcm_substream *substream)
964{
965
966 pr_debug("%s(): substream = %s\n", __func__, substream->name);
967 msm8960_aux_pcm_free_gpios();
968}
969
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700970static void msm8960_shutdown(struct snd_pcm_substream *substream)
971{
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700972 pr_debug("%s(): substream = %s stream = %d\n", __func__,
973 substream->name, substream->stream);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700974}
975
976static struct snd_soc_ops msm8960_be_ops = {
977 .startup = msm8960_startup,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800978 .hw_params = msm8960_hw_params,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700979 .shutdown = msm8960_shutdown,
980};
981
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -0700982static struct snd_soc_ops msm8960_auxpcm_be_ops = {
983 .startup = msm8960_auxpcm_startup,
984 .shutdown = msm8960_auxpcm_shutdown,
985};
986
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700987/* Digital audio interface glue - connects codec <---> CPU */
Kuirong Wangb25838e2012-01-16 23:37:23 -0800988static struct snd_soc_dai_link msm8960_dai_common[] = {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700989 /* FrontEnd DAI Links */
990 {
991 .name = "MSM8960 Media1",
992 .stream_name = "MultiMedia1",
993 .cpu_dai_name = "MultiMedia1",
994 .platform_name = "msm-pcm-dsp",
995 .dynamic = 1,
996 .dsp_link = &fe_media,
997 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA1
998 },
999 {
1000 .name = "MSM8960 Media2",
1001 .stream_name = "MultiMedia2",
1002 .cpu_dai_name = "MultiMedia2",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001003 .platform_name = "msm-multi-ch-pcm-dsp",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001004 .dynamic = 1,
1005 .dsp_link = &fe_media,
1006 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA2,
1007 },
1008 {
1009 .name = "Circuit-Switch Voice",
1010 .stream_name = "CS-Voice",
1011 .cpu_dai_name = "CS-VOICE",
1012 .platform_name = "msm-pcm-voice",
1013 .dynamic = 1,
1014 .dsp_link = &fe_media,
1015 .be_id = MSM_FRONTEND_DAI_CS_VOICE,
1016 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001017 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001018 },
1019 {
1020 .name = "MSM VoIP",
1021 .stream_name = "VoIP",
1022 .cpu_dai_name = "VoIP",
1023 .platform_name = "msm-voip-dsp",
1024 .dynamic = 1,
1025 .dsp_link = &fe_media,
1026 .be_id = MSM_FRONTEND_DAI_VOIP,
1027 },
1028 {
1029 .name = "MSM8960 LPA",
1030 .stream_name = "LPA",
1031 .cpu_dai_name = "MultiMedia3",
1032 .platform_name = "msm-pcm-lpa",
1033 .dynamic = 1,
1034 .dsp_link = &lpa_fe_media,
1035 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA3,
1036 },
1037 /* Hostless PMC purpose */
1038 {
1039 .name = "SLIMBUS_0 Hostless",
1040 .stream_name = "SLIMBUS_0 Hostless",
1041 .cpu_dai_name = "SLIMBUS0_HOSTLESS",
1042 .platform_name = "msm-pcm-hostless",
1043 .dynamic = 1,
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001044 .dsp_link = &bidir_hl_media,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001045 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001046 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001047 /* .be_id = do not care */
1048 },
1049 {
1050 .name = "INT_FM Hostless",
1051 .stream_name = "INT_FM Hostless",
1052 .cpu_dai_name = "INT_FM_HOSTLESS",
1053 .platform_name = "msm-pcm-hostless",
1054 .dynamic = 1,
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001055 .dsp_link = &bidir_hl_media,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001056 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
Alex Wongb9367922011-09-21 23:09:43 -07001057 .ignore_suspend = 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001058 /* .be_id = do not care */
1059 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301060 {
1061 .name = "MSM AFE-PCM RX",
1062 .stream_name = "AFE-PROXY RX",
1063 .cpu_dai_name = "msm-dai-q6.241",
1064 .codec_name = "msm-stub-codec.1",
1065 .codec_dai_name = "msm-stub-rx",
1066 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001067 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301068 },
1069 {
1070 .name = "MSM AFE-PCM TX",
1071 .stream_name = "AFE-PROXY TX",
1072 .cpu_dai_name = "msm-dai-q6.240",
1073 .codec_name = "msm-stub-codec.1",
1074 .codec_dai_name = "msm-stub-tx",
1075 .platform_name = "msm-pcm-afe",
Alex Wongb9367922011-09-21 23:09:43 -07001076 .ignore_suspend = 1,
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301077 },
Asish Bhattacharya09f9e0a2011-11-11 13:22:47 +05301078 {
1079 .name = "MSM8960 Compr",
1080 .stream_name = "COMPR",
1081 .cpu_dai_name = "MultiMedia4",
1082 .platform_name = "msm-compr-dsp",
1083 .dynamic = 1,
1084 .dsp_link = &lpa_fe_media,
1085 .be_id = MSM_FRONTEND_DAI_MULTIMEDIA4,
1086 },
Patrick Lai9b56e1d2012-01-22 22:14:15 -08001087 {
1088 .name = "AUXPCM Hostless",
1089 .stream_name = "AUXPCM Hostless",
1090 .cpu_dai_name = "AUXPCM_HOSTLESS",
1091 .platform_name = "msm-pcm-hostless",
1092 .dynamic = 1,
1093 .dsp_link = &bidir_hl_media,
1094 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1095 .ignore_suspend = 1,
1096 },
Alex Wongb3d06a02012-01-12 10:00:41 -08001097 /* HDMI Hostless */
1098 {
1099 .name = "HDMI_RX_HOSTLESS",
1100 .stream_name = "HDMI_RX_HOSTLESS",
1101 .cpu_dai_name = "HDMI_HOSTLESS",
1102 .platform_name = "msm-pcm-hostless",
1103 .dynamic = 1,
1104 .dsp_link = &hdmi_rx_hl,
1105 .no_host_mode = SND_SOC_DAI_LINK_NO_HOST,
1106 .no_codec = 1,
1107 .ignore_suspend = 1,
1108 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001109 /* Backend BT/FM DAI Links */
1110 {
1111 .name = LPASS_BE_INT_BT_SCO_RX,
1112 .stream_name = "Internal BT-SCO Playback",
1113 .cpu_dai_name = "msm-dai-q6.12288",
1114 .platform_name = "msm-pcm-routing",
1115 .codec_name = "msm-stub-codec.1",
1116 .codec_dai_name = "msm-stub-rx",
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001117 .init = &msm8960_btsco_init,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001118 .no_pcm = 1,
1119 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_RX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001120 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001121 },
1122 {
1123 .name = LPASS_BE_INT_BT_SCO_TX,
1124 .stream_name = "Internal BT-SCO Capture",
1125 .cpu_dai_name = "msm-dai-q6.12289",
1126 .platform_name = "msm-pcm-routing",
1127 .codec_name = "msm-stub-codec.1",
1128 .codec_dai_name = "msm-stub-tx",
1129 .no_pcm = 1,
1130 .be_id = MSM_BACKEND_DAI_INT_BT_SCO_TX,
Jayasena Sangaraboina47c08242011-08-10 11:30:57 -07001131 .be_hw_params_fixup = msm8960_btsco_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001132 },
1133 {
1134 .name = LPASS_BE_INT_FM_RX,
1135 .stream_name = "Internal FM Playback",
1136 .cpu_dai_name = "msm-dai-q6.12292",
1137 .platform_name = "msm-pcm-routing",
1138 .codec_name = "msm-stub-codec.1",
1139 .codec_dai_name = "msm-stub-rx",
1140 .no_pcm = 1,
1141 .be_id = MSM_BACKEND_DAI_INT_FM_RX,
1142 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1143 },
1144 {
1145 .name = LPASS_BE_INT_FM_TX,
1146 .stream_name = "Internal FM Capture",
1147 .cpu_dai_name = "msm-dai-q6.12293",
1148 .platform_name = "msm-pcm-routing",
1149 .codec_name = "msm-stub-codec.1",
1150 .codec_dai_name = "msm-stub-tx",
1151 .no_pcm = 1,
1152 .be_id = MSM_BACKEND_DAI_INT_FM_TX,
1153 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1154 },
1155 /* HDMI BACK END DAI Link */
1156 {
1157 .name = LPASS_BE_HDMI,
1158 .stream_name = "HDMI Playback",
Kiran Kandi5e809b02012-01-31 00:24:33 -08001159 .cpu_dai_name = "msm-dai-q6-hdmi.8",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001160 .platform_name = "msm-pcm-routing",
1161 .codec_name = "msm-stub-codec.1",
1162 .codec_dai_name = "msm-stub-rx",
1163 .no_pcm = 1,
1164 .no_codec = 1,
1165 .be_id = MSM_BACKEND_DAI_HDMI_RX,
Helen Zeng73f7fc62011-11-18 16:29:59 -08001166 .be_hw_params_fixup = msm8960_hdmi_be_hw_params_fixup,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001167 },
Laxminath Kasam32657ec2011-08-01 19:26:57 +05301168 /* Backend AFE DAI Links */
1169 {
1170 .name = LPASS_BE_AFE_PCM_RX,
1171 .stream_name = "AFE Playback",
1172 .cpu_dai_name = "msm-dai-q6.224",
1173 .platform_name = "msm-pcm-routing",
1174 .codec_name = "msm-stub-codec.1",
1175 .codec_dai_name = "msm-stub-rx",
1176 .no_codec = 1,
1177 .no_pcm = 1,
1178 .be_id = MSM_BACKEND_DAI_AFE_PCM_RX,
1179 },
1180 {
1181 .name = LPASS_BE_AFE_PCM_TX,
1182 .stream_name = "AFE Capture",
1183 .cpu_dai_name = "msm-dai-q6.225",
1184 .platform_name = "msm-pcm-routing",
1185 .codec_name = "msm-stub-codec.1",
1186 .codec_dai_name = "msm-stub-tx",
1187 .no_codec = 1,
1188 .no_pcm = 1,
1189 .be_id = MSM_BACKEND_DAI_AFE_PCM_TX,
1190 },
Bhalchandra Gajare0e795c42011-08-15 18:10:30 -07001191 /* AUX PCM Backend DAI Links */
1192 {
1193 .name = LPASS_BE_AUXPCM_RX,
1194 .stream_name = "AUX PCM Playback",
1195 .cpu_dai_name = "msm-dai-q6.2",
1196 .platform_name = "msm-pcm-routing",
1197 .codec_name = "msm-stub-codec.1",
1198 .codec_dai_name = "msm-stub-rx",
1199 .no_pcm = 1,
1200 .be_id = MSM_BACKEND_DAI_AUXPCM_RX,
1201 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1202 .ops = &msm8960_auxpcm_be_ops,
1203 },
1204 {
1205 .name = LPASS_BE_AUXPCM_TX,
1206 .stream_name = "AUX PCM Capture",
1207 .cpu_dai_name = "msm-dai-q6.3",
1208 .platform_name = "msm-pcm-routing",
1209 .codec_name = "msm-stub-codec.1",
1210 .codec_dai_name = "msm-stub-tx",
1211 .no_pcm = 1,
1212 .be_id = MSM_BACKEND_DAI_AUXPCM_TX,
1213 .be_hw_params_fixup = msm8960_auxpcm_be_params_fixup,
1214 },
Helen Zeng0705a5f2011-10-14 15:29:52 -07001215 /* Incall Music BACK END DAI Link */
1216 {
1217 .name = LPASS_BE_VOICE_PLAYBACK_TX,
1218 .stream_name = "Voice Farend Playback",
1219 .cpu_dai_name = "msm-dai-q6.32773",
1220 .platform_name = "msm-pcm-routing",
1221 .codec_name = "msm-stub-codec.1",
1222 .codec_dai_name = "msm-stub-rx",
1223 .no_pcm = 1,
1224 .no_codec = 1,
1225 .be_id = MSM_BACKEND_DAI_VOICE_PLAYBACK_TX,
1226 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1227 },
Helen Zenge3d716a2011-10-14 16:32:16 -07001228 /* Incall Record Uplink BACK END DAI Link */
1229 {
1230 .name = LPASS_BE_INCALL_RECORD_TX,
1231 .stream_name = "Voice Uplink Capture",
1232 .cpu_dai_name = "msm-dai-q6.32772",
1233 .platform_name = "msm-pcm-routing",
1234 .codec_name = "msm-stub-codec.1",
1235 .codec_dai_name = "msm-stub-tx",
1236 .no_pcm = 1,
1237 .no_codec = 1,
1238 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_TX,
1239 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1240 },
1241 /* Incall Record Downlink BACK END DAI Link */
1242 {
1243 .name = LPASS_BE_INCALL_RECORD_RX,
1244 .stream_name = "Voice Downlink Capture",
1245 .cpu_dai_name = "msm-dai-q6.32771",
1246 .platform_name = "msm-pcm-routing",
1247 .codec_name = "msm-stub-codec.1",
1248 .codec_dai_name = "msm-stub-tx",
1249 .no_pcm = 1,
1250 .no_codec = 1,
1251 .be_id = MSM_BACKEND_DAI_INCALL_RECORD_RX,
1252 .be_hw_params_fixup = msm8960_be_hw_params_fixup,
1253 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001254};
1255
Kuirong Wangb25838e2012-01-16 23:37:23 -08001256static struct snd_soc_dai_link msm8960_dai_delta_tabla1x[] = {
1257 /* Backend DAI Links */
1258 {
1259 .name = LPASS_BE_SLIMBUS_0_RX,
1260 .stream_name = "Slimbus Playback",
1261 .cpu_dai_name = "msm-dai-q6.16384",
1262 .platform_name = "msm-pcm-routing",
1263 .codec_name = "tabla1x_codec",
1264 .codec_dai_name = "tabla_rx1",
1265 .no_pcm = 1,
1266 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1267 .init = &msm8960_audrx_init,
1268 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1269 .ops = &msm8960_be_ops,
1270 },
1271 {
1272 .name = LPASS_BE_SLIMBUS_0_TX,
1273 .stream_name = "Slimbus Capture",
1274 .cpu_dai_name = "msm-dai-q6.16385",
1275 .platform_name = "msm-pcm-routing",
1276 .codec_name = "tabla1x_codec",
1277 .codec_dai_name = "tabla_tx1",
1278 .no_pcm = 1,
1279 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1280 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1281 .ops = &msm8960_be_ops,
1282 },
1283};
1284
1285
1286static struct snd_soc_dai_link msm8960_dai_delta_tabla2x[] = {
1287 /* Backend DAI Links */
1288 {
1289 .name = LPASS_BE_SLIMBUS_0_RX,
1290 .stream_name = "Slimbus Playback",
1291 .cpu_dai_name = "msm-dai-q6.16384",
1292 .platform_name = "msm-pcm-routing",
1293 .codec_name = "tabla_codec",
1294 .codec_dai_name = "tabla_rx1",
1295 .no_pcm = 1,
1296 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_RX,
1297 .init = &msm8960_audrx_init,
1298 .be_hw_params_fixup = msm8960_slim_0_rx_be_hw_params_fixup,
1299 .ops = &msm8960_be_ops,
1300 },
1301 {
1302 .name = LPASS_BE_SLIMBUS_0_TX,
1303 .stream_name = "Slimbus Capture",
1304 .cpu_dai_name = "msm-dai-q6.16385",
1305 .platform_name = "msm-pcm-routing",
1306 .codec_name = "tabla_codec",
1307 .codec_dai_name = "tabla_tx1",
1308 .no_pcm = 1,
1309 .be_id = MSM_BACKEND_DAI_SLIMBUS_0_TX,
1310 .be_hw_params_fixup = msm8960_slim_0_tx_be_hw_params_fixup,
1311 .ops = &msm8960_be_ops,
1312 },
1313};
1314
1315static struct snd_soc_dai_link msm8960_tabla1x_dai[
1316 ARRAY_SIZE(msm8960_dai_common) +
1317 ARRAY_SIZE(msm8960_dai_delta_tabla1x)];
1318
1319
1320static struct snd_soc_dai_link msm8960_dai[
1321 ARRAY_SIZE(msm8960_dai_common) +
1322 ARRAY_SIZE(msm8960_dai_delta_tabla2x)];
1323
1324static struct snd_soc_card snd_soc_tabla1x_card_msm8960 = {
1325 .name = "msm8960-tabla1x-snd-card",
1326 .dai_link = msm8960_tabla1x_dai,
1327 .num_links = ARRAY_SIZE(msm8960_tabla1x_dai),
1328};
1329
1330static struct snd_soc_card snd_soc_card_msm8960 = {
1331 .name = "msm8960-snd-card",
1332 .dai_link = msm8960_dai,
1333 .num_links = ARRAY_SIZE(msm8960_dai),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001334};
1335
1336static struct platform_device *msm8960_snd_device;
Kuirong Wangb25838e2012-01-16 23:37:23 -08001337static struct platform_device *msm8960_snd_tabla1x_device;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001338
1339static int msm8960_configure_headset_mic_gpios(void)
1340{
1341 int ret;
1342 struct pm_gpio param = {
1343 .direction = PM_GPIO_DIR_OUT,
1344 .output_buffer = PM_GPIO_OUT_BUF_CMOS,
1345 .output_value = 1,
1346 .pull = PM_GPIO_PULL_NO,
1347 .vin_sel = PM_GPIO_VIN_S4,
1348 .out_strength = PM_GPIO_STRENGTH_MED,
1349 .function = PM_GPIO_FUNC_NORMAL,
1350 };
1351
1352 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(23), "AV_SWITCH");
1353 if (ret) {
1354 pr_err("%s: Failed to request gpio %d\n", __func__,
1355 PM8921_GPIO_PM_TO_SYS(23));
1356 return ret;
1357 }
1358
1359 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(23), &param);
1360 if (ret)
1361 pr_err("%s: Failed to configure gpio %d\n", __func__,
1362 PM8921_GPIO_PM_TO_SYS(23));
1363 else
1364 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(23), 0);
1365
1366 ret = gpio_request(PM8921_GPIO_PM_TO_SYS(35), "US_EURO_SWITCH");
1367 if (ret) {
1368 pr_err("%s: Failed to request gpio %d\n", __func__,
1369 PM8921_GPIO_PM_TO_SYS(35));
1370 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1371 return ret;
1372 }
1373 ret = pm8xxx_gpio_config(PM8921_GPIO_PM_TO_SYS(35), &param);
1374 if (ret)
1375 pr_err("%s: Failed to configure gpio %d\n", __func__,
1376 PM8921_GPIO_PM_TO_SYS(35));
1377 else
Patrick Lai2ae0a002011-09-26 14:51:09 -07001378 gpio_direction_output(PM8921_GPIO_PM_TO_SYS(35), 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001379
1380 return 0;
1381}
1382static void msm8960_free_headset_mic_gpios(void)
1383{
1384 if (msm8960_headset_gpios_configured) {
1385 gpio_free(PM8921_GPIO_PM_TO_SYS(23));
1386 gpio_free(PM8921_GPIO_PM_TO_SYS(35));
1387 }
1388}
1389
1390static int __init msm8960_audio_init(void)
1391{
1392 int ret;
1393
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301394 if (!cpu_is_msm8960()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001395 pr_err("%s: Not the right machine type\n", __func__);
1396 return -ENODEV ;
1397 }
Joonwoo Park0976d012011-12-22 11:48:18 -08001398
1399 tabla_mbhc_cal = def_tabla_mbhc_cal();
1400 if (!tabla_mbhc_cal) {
1401 pr_err("Calibration data allocation failed\n");
1402 return -ENOMEM;
1403 }
1404
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001405 msm8960_snd_device = platform_device_alloc("soc-audio", 0);
1406 if (!msm8960_snd_device) {
1407 pr_err("Platform device allocation failed\n");
Joonwoo Park0976d012011-12-22 11:48:18 -08001408 kfree(tabla_mbhc_cal);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001409 return -ENOMEM;
1410 }
1411
Kuirong Wangb25838e2012-01-16 23:37:23 -08001412 memcpy(msm8960_dai, msm8960_dai_common, sizeof(msm8960_dai_common));
1413 memcpy(msm8960_dai + ARRAY_SIZE(msm8960_dai_common),
1414 msm8960_dai_delta_tabla2x, sizeof(msm8960_dai_delta_tabla2x));
1415
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001416 platform_set_drvdata(msm8960_snd_device, &snd_soc_card_msm8960);
1417 ret = platform_device_add(msm8960_snd_device);
1418 if (ret) {
1419 platform_device_put(msm8960_snd_device);
Joonwoo Park0976d012011-12-22 11:48:18 -08001420 kfree(tabla_mbhc_cal);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001421 return ret;
1422 }
1423
Kuirong Wangb25838e2012-01-16 23:37:23 -08001424 msm8960_snd_tabla1x_device = platform_device_alloc("soc-audio", 1);
1425 if (!msm8960_snd_tabla1x_device) {
1426 pr_err("Platform device allocation failed\n");
1427 kfree(tabla_mbhc_cal);
1428 return -ENOMEM;
1429 }
1430
1431 memcpy(msm8960_tabla1x_dai, msm8960_dai_common,
1432 sizeof(msm8960_dai_common));
1433 memcpy(msm8960_tabla1x_dai + ARRAY_SIZE(msm8960_dai_common),
1434 msm8960_dai_delta_tabla1x, sizeof(msm8960_dai_delta_tabla1x));
1435
1436 platform_set_drvdata(msm8960_snd_tabla1x_device,
1437 &snd_soc_tabla1x_card_msm8960);
1438 ret = platform_device_add(msm8960_snd_tabla1x_device);
1439 if (ret) {
1440 platform_device_put(msm8960_snd_tabla1x_device);
1441 kfree(tabla_mbhc_cal);
1442 return ret;
1443 }
1444
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001445 if (msm8960_configure_headset_mic_gpios()) {
1446 pr_err("%s Fail to configure headset mic gpios\n", __func__);
1447 msm8960_headset_gpios_configured = 0;
1448 } else
1449 msm8960_headset_gpios_configured = 1;
1450
1451 return ret;
1452
1453}
1454module_init(msm8960_audio_init);
1455
1456static void __exit msm8960_audio_exit(void)
1457{
Asish Bhattacharyab86c3472012-02-15 08:31:52 +05301458 if (!cpu_is_msm8960()) {
Bharath Ramachandramurthyb8e797f2011-11-30 12:08:42 -08001459 pr_err("%s: Not the right machine type\n", __func__);
1460 return ;
1461 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001462 msm8960_free_headset_mic_gpios();
1463 platform_device_unregister(msm8960_snd_device);
Kuirong Wangb25838e2012-01-16 23:37:23 -08001464 platform_device_unregister(msm8960_snd_tabla1x_device);
Joonwoo Park0976d012011-12-22 11:48:18 -08001465 kfree(tabla_mbhc_cal);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001466}
1467module_exit(msm8960_audio_exit);
1468
1469MODULE_DESCRIPTION("ALSA SoC MSM8960");
1470MODULE_LICENSE("GPL v2");