blob: 4c446e475cd5ce4d7f76d352fbe2120f3278286e [file] [log] [blame]
Kuirong Wangbc5ac042012-06-26 15:35:22 -07001/* Copyright (c) 2011-2013, The Linux Foundation. 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#include <linux/module.h>
13#include <linux/init.h>
Bradley Rubin229c6a52011-07-12 16:18:48 -070014#include <linux/firmware.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070015#include <linux/slab.h>
16#include <linux/platform_device.h>
Santosh Mardie15e2302011-11-15 10:39:23 +053017#include <linux/device.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070018#include <linux/printk.h>
19#include <linux/ratelimit.h>
Bradley Rubincb3950a2011-08-18 13:07:26 -070020#include <linux/debugfs.h>
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -070021#include <linux/wait.h>
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +053022#include <linux/mfd/wcd9xxx/core.h>
23#include <linux/mfd/wcd9xxx/wcd9xxx_registers.h>
24#include <linux/mfd/wcd9xxx/wcd9310_registers.h>
25#include <linux/mfd/wcd9xxx/pdata.h>
Santosh Mardie15e2302011-11-15 10:39:23 +053026#include <sound/pcm.h>
27#include <sound/pcm_params.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070028#include <sound/soc.h>
29#include <sound/soc-dapm.h>
30#include <sound/tlv.h>
31#include <linux/bitops.h>
32#include <linux/delay.h>
Kuirong Wanga545e722012-02-06 19:12:54 -080033#include <linux/pm_runtime.h>
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070034#include <linux/kernel.h>
35#include <linux/gpio.h>
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -070036#include <linux/irq.h>
37#include <linux/wakelock.h>
38#include <linux/suspend.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070039#include "wcd9310.h"
40
Joonwoo Parkc1c67a92012-08-07 16:05:36 -070041static int cfilt_adjust_ms = 10;
42module_param(cfilt_adjust_ms, int, 0644);
43MODULE_PARM_DESC(cfilt_adjust_ms, "delay after adjusting cfilt voltage in ms");
44
Kiran Kandi1e6371d2012-03-29 11:48:57 -070045#define WCD9310_RATES (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000 |\
46 SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_48000 |\
47 SNDRV_PCM_RATE_96000 | SNDRV_PCM_RATE_192000)
48
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -070049
50#define NUM_DECIMATORS 10
51#define NUM_INTERPOLATORS 7
52#define BITS_PER_REG 8
Bhalchandra Gajare19d9c132011-11-18 14:57:08 -080053#define TABLA_CFILT_FAST_MODE 0x00
54#define TABLA_CFILT_SLOW_MODE 0x40
Patrick Lai64b43262011-12-06 17:29:15 -080055#define MBHC_FW_READ_ATTEMPTS 15
56#define MBHC_FW_READ_TIMEOUT 2000000
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -070057
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -070058#define SLIM_CLOSE_TIMEOUT 1000
59
Joonwoo Park03324832012-03-19 19:36:16 -070060enum {
61 MBHC_USE_HPHL_TRIGGER = 1,
62 MBHC_USE_MB_TRIGGER = 2
63};
64
65#define MBHC_NUM_DCE_PLUG_DETECT 3
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -070066#define NUM_ATTEMPTS_INSERT_DETECT 25
67#define NUM_ATTEMPTS_TO_REPORT 5
Joonwoo Park03324832012-03-19 19:36:16 -070068
Joonwoo Park2cc13f02012-05-09 12:44:25 -070069#define TABLA_JACK_MASK (SND_JACK_HEADSET | SND_JACK_OC_HPHL | \
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -070070 SND_JACK_OC_HPHR | SND_JACK_LINEOUT | \
71 SND_JACK_UNSUPPORTED)
Patrick Lai49efeac2011-11-03 11:01:12 -070072
Santosh Mardie15e2302011-11-15 10:39:23 +053073#define TABLA_I2S_MASTER_MODE_MASK 0x08
74
Patrick Laic7cae882011-11-18 11:52:49 -080075#define TABLA_OCP_ATTEMPT 1
76
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -080077#define AIF1_PB 1
78#define AIF1_CAP 2
Neema Shettyd3a89262012-02-16 10:23:50 -080079#define AIF2_PB 3
Kiran Kandi1e6371d2012-03-29 11:48:57 -070080#define AIF2_CAP 4
Neema Shetty3fb1b802012-04-27 13:53:24 -070081#define AIF3_CAP 5
Kiran Kandia9fffe92012-05-20 23:42:30 -070082#define AIF3_PB 6
Kiran Kandi1e6371d2012-03-29 11:48:57 -070083
Kiran Kandia9fffe92012-05-20 23:42:30 -070084#define NUM_CODEC_DAIS 6
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -080085
86struct tabla_codec_dai_data {
87 u32 rate;
88 u32 *ch_num;
89 u32 ch_act;
90 u32 ch_tot;
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -070091 u32 ch_mask;
92 wait_queue_head_t dai_wait;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -080093};
94
Kuirong Wangbc5ac042012-06-26 15:35:22 -070095#define TABLA_COMP_DIGITAL_GAIN_HP_OFFSET 3
96#define TABLA_COMP_DIGITAL_GAIN_LINEOUT_OFFSET 6
97
Joonwoo Park0976d012011-12-22 11:48:18 -080098#define TABLA_MCLK_RATE_12288KHZ 12288000
99#define TABLA_MCLK_RATE_9600KHZ 9600000
100
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800101#define TABLA_FAKE_INS_THRESHOLD_MS 2500
Joonwoo Park6b9b03f2012-01-23 18:48:54 -0800102#define TABLA_FAKE_REMOVAL_MIN_PERIOD_MS 50
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800103
Joonwoo Park03324832012-03-19 19:36:16 -0700104#define TABLA_MBHC_BUTTON_MIN 0x8000
105
Joonwoo Park03324832012-03-19 19:36:16 -0700106#define TABLA_MBHC_FAKE_INSERT_LOW 10
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700107#define TABLA_MBHC_FAKE_INSERT_HIGH 80
108#define TABLA_MBHC_FAKE_INS_HIGH_NO_GPIO 150
Joonwoo Park03324832012-03-19 19:36:16 -0700109
110#define TABLA_MBHC_STATUS_REL_DETECTION 0x0C
111
Joonwoo Parka3a4f322012-08-11 13:46:30 -0700112#define TABLA_MBHC_GPIO_REL_DEBOUNCE_TIME_MS 50
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700113
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700114#define TABLA_MBHC_FAKE_INS_DELTA_MV 200
115#define TABLA_MBHC_FAKE_INS_DELTA_SCALED_MV 300
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700116
117#define TABLA_HS_DETECT_PLUG_TIME_MS (5 * 1000)
118#define TABLA_HS_DETECT_PLUG_INERVAL_MS 100
119
120#define TABLA_GPIO_IRQ_DEBOUNCE_TIME_US 5000
121
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700122#define TABLA_MBHC_GND_MIC_SWAP_THRESHOLD 2
123
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700124#define TABLA_ACQUIRE_LOCK(x) do { mutex_lock(&x); } while (0)
125#define TABLA_RELEASE_LOCK(x) do { mutex_unlock(&x); } while (0)
126
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700127static const DECLARE_TLV_DB_SCALE(digital_gain, 0, 1, 0);
128static const DECLARE_TLV_DB_SCALE(line_gain, 0, 7, 1);
129static const DECLARE_TLV_DB_SCALE(analog_gain, 0, 25, 1);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800130static struct snd_soc_dai_driver tabla_dai[];
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -0800131static const DECLARE_TLV_DB_SCALE(aux_pga_gain, 0, 2, 0);
Kiran Kandi93923902012-06-20 17:00:25 -0700132static int tabla_codec_enable_slimrx(struct snd_soc_dapm_widget *w,
133 struct snd_kcontrol *kcontrol, int event);
134static int tabla_codec_enable_slimtx(struct snd_soc_dapm_widget *w,
135 struct snd_kcontrol *kcontrol, int event);
136
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700137
138enum tabla_bandgap_type {
139 TABLA_BANDGAP_OFF = 0,
140 TABLA_BANDGAP_AUDIO_MODE,
141 TABLA_BANDGAP_MBHC_MODE,
142};
143
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -0700144struct mbhc_micbias_regs {
145 u16 cfilt_val;
146 u16 cfilt_ctl;
147 u16 mbhc_reg;
148 u16 int_rbias;
149 u16 ctl_reg;
Bhalchandra Gajare19d9c132011-11-18 14:57:08 -0800150 u8 cfilt_sel;
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -0700151};
152
Ben Romberger1f045a72011-11-04 10:14:57 -0700153/* Codec supports 2 IIR filters */
154enum {
155 IIR1 = 0,
156 IIR2,
157 IIR_MAX,
158};
159/* Codec supports 5 bands */
160enum {
161 BAND1 = 0,
162 BAND2,
163 BAND3,
164 BAND4,
165 BAND5,
166 BAND_MAX,
167};
168
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800169enum {
170 COMPANDER_1 = 0,
171 COMPANDER_2,
172 COMPANDER_MAX,
173};
174
175enum {
176 COMPANDER_FS_8KHZ = 0,
177 COMPANDER_FS_16KHZ,
178 COMPANDER_FS_32KHZ,
179 COMPANDER_FS_48KHZ,
Kiran Kandi1e6371d2012-03-29 11:48:57 -0700180 COMPANDER_FS_96KHZ,
181 COMPANDER_FS_192KHZ,
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800182 COMPANDER_FS_MAX,
183};
184
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700185enum {
186 COMP_SHUTDWN_TIMEOUT_PCM_1 = 0,
187 COMP_SHUTDWN_TIMEOUT_PCM_240,
188 COMP_SHUTDWN_TIMEOUT_PCM_480,
189 COMP_SHUTDWN_TIMEOUT_PCM_960,
190 COMP_SHUTDWN_TIMEOUT_PCM_1440,
191 COMP_SHUTDWN_TIMEOUT_PCM_2880,
192 COMP_SHUTDWN_TIMEOUT_PCM_5760,
193};
194
Joonwoo Parka9444452011-12-08 18:48:27 -0800195/* Flags to track of PA and DAC state.
196 * PA and DAC should be tracked separately as AUXPGA loopback requires
197 * only PA to be turned on without DAC being on. */
198enum tabla_priv_ack_flags {
199 TABLA_HPHL_PA_OFF_ACK = 0,
200 TABLA_HPHR_PA_OFF_ACK,
201 TABLA_HPHL_DAC_OFF_ACK,
202 TABLA_HPHR_DAC_OFF_ACK
203};
204
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800205
206struct comp_sample_dependent_params {
207 u32 peak_det_timeout;
208 u32 rms_meter_div_fact;
209 u32 rms_meter_resamp_fact;
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700210 u32 shutdown_timeout;
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800211};
212
Joonwoo Park0976d012011-12-22 11:48:18 -0800213/* Data used by MBHC */
214struct mbhc_internal_cal_data {
215 u16 dce_z;
216 u16 dce_mb;
217 u16 sta_z;
218 u16 sta_mb;
Joonwoo Park433149a2012-01-11 09:53:54 -0800219 u32 t_sta_dce;
Joonwoo Park0976d012011-12-22 11:48:18 -0800220 u32 t_dce;
221 u32 t_sta;
222 u32 micb_mv;
223 u16 v_ins_hu;
224 u16 v_ins_h;
225 u16 v_b1_hu;
226 u16 v_b1_h;
227 u16 v_b1_huc;
228 u16 v_brh;
229 u16 v_brl;
230 u16 v_no_mic;
Joonwoo Park0976d012011-12-22 11:48:18 -0800231 u8 npoll;
232 u8 nbounce_wait;
Joonwoo Parkcf473b42012-03-29 19:48:16 -0700233 s16 adj_v_hs_max;
234 u16 adj_v_ins_hu;
235 u16 adj_v_ins_h;
236 s16 v_inval_ins_low;
237 s16 v_inval_ins_high;
Joonwoo Park0976d012011-12-22 11:48:18 -0800238};
239
Joonwoo Park6c1ebb62012-01-16 19:08:43 -0800240struct tabla_reg_address {
241 u16 micb_4_ctl;
242 u16 micb_4_int_rbias;
243 u16 micb_4_mbhc;
244};
245
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700246enum tabla_mbhc_plug_type {
Joonwoo Park41956722012-04-18 13:13:07 -0700247 PLUG_TYPE_INVALID = -1,
248 PLUG_TYPE_NONE,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700249 PLUG_TYPE_HEADSET,
250 PLUG_TYPE_HEADPHONE,
251 PLUG_TYPE_HIGH_HPH,
Joonwoo Park2cc13f02012-05-09 12:44:25 -0700252 PLUG_TYPE_GND_MIC_SWAP,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700253};
254
255enum tabla_mbhc_state {
256 MBHC_STATE_NONE = -1,
257 MBHC_STATE_POTENTIAL,
258 MBHC_STATE_POTENTIAL_RECOVERY,
259 MBHC_STATE_RELEASE,
260};
261
Kiran Kandid8cf5212012-03-02 15:34:53 -0800262struct hpf_work {
263 struct tabla_priv *tabla;
264 u32 decimator;
265 u8 tx_hpf_cut_of_freq;
266 struct delayed_work dwork;
267};
268
269static struct hpf_work tx_hpf_work[NUM_DECIMATORS];
270
Bradley Rubin229c6a52011-07-12 16:18:48 -0700271struct tabla_priv {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700272 struct snd_soc_codec *codec;
Joonwoo Park6c1ebb62012-01-16 19:08:43 -0800273 struct tabla_reg_address reg_addr;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700274 u32 adc_count;
Patrick Lai3043fba2011-08-01 14:15:57 -0700275 u32 cfilt1_cnt;
276 u32 cfilt2_cnt;
277 u32 cfilt3_cnt;
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700278 u32 rx_bias_count;
Kiran Kandi0ba468f2012-05-08 11:45:05 -0700279 s32 dmic_1_2_clk_cnt;
280 s32 dmic_3_4_clk_cnt;
281 s32 dmic_5_6_clk_cnt;
282
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700283 enum tabla_bandgap_type bandgap_type;
Kiran Kandi6fae8bf2011-08-15 10:36:42 -0700284 bool mclk_enabled;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700285 bool clock_active;
286 bool config_mode_active;
287 bool mbhc_polling_active;
Joonwoo Parkf4267c22012-01-10 13:25:24 -0800288 unsigned long mbhc_fake_ins_start;
Bradley Rubincb1e2732011-06-23 16:49:20 -0700289 int buttons_pressed;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700290 enum tabla_mbhc_state mbhc_state;
291 struct tabla_mbhc_config mbhc_cfg;
Joonwoo Park0976d012011-12-22 11:48:18 -0800292 struct mbhc_internal_cal_data mbhc_data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700293
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +0530294 struct wcd9xxx_pdata *pdata;
Bradley Rubina7096d02011-08-03 18:29:02 -0700295 u32 anc_slot;
Bradley Rubincb3950a2011-08-18 13:07:26 -0700296
297 bool no_mic_headset_override;
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -0700298 /* Delayed work to report long button press */
Joonwoo Park03324832012-03-19 19:36:16 -0700299 struct delayed_work mbhc_btn_dwork;
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -0700300
301 struct mbhc_micbias_regs mbhc_bias_regs;
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -0700302 bool mbhc_micbias_switched;
Patrick Lai49efeac2011-11-03 11:01:12 -0700303
Joonwoo Parka9444452011-12-08 18:48:27 -0800304 /* track PA/DAC state */
305 unsigned long hph_pa_dac_state;
306
Santosh Mardie15e2302011-11-15 10:39:23 +0530307 /*track tabla interface type*/
308 u8 intf_type;
309
Patrick Lai49efeac2011-11-03 11:01:12 -0700310 u32 hph_status; /* track headhpone status */
311 /* define separate work for left and right headphone OCP to avoid
312 * additional checking on which OCP event to report so no locking
313 * to ensure synchronization is required
314 */
315 struct work_struct hphlocp_work; /* reporting left hph ocp off */
316 struct work_struct hphrocp_work; /* reporting right hph ocp off */
Joonwoo Park8b1f0982011-12-08 17:12:45 -0800317
Patrick Laic7cae882011-11-18 11:52:49 -0800318 u8 hphlocp_cnt; /* headphone left ocp retry */
319 u8 hphrocp_cnt; /* headphone right ocp retry */
Joonwoo Park0976d012011-12-22 11:48:18 -0800320
Patrick Lai64b43262011-12-06 17:29:15 -0800321 /* Work to perform MBHC Firmware Read */
322 struct delayed_work mbhc_firmware_dwork;
323 const struct firmware *mbhc_fw;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -0800324
325 /* num of slim ports required */
326 struct tabla_codec_dai_data dai[NUM_CODEC_DAIS];
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800327
328 /*compander*/
329 int comp_enabled[COMPANDER_MAX];
330 u32 comp_fs[COMPANDER_MAX];
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -0800331
332 /* Maintain the status of AUX PGA */
333 int aux_pga_cnt;
334 u8 aux_l_gain;
335 u8 aux_r_gain;
Joonwoo Park03324832012-03-19 19:36:16 -0700336
Joonwoo Park03324832012-03-19 19:36:16 -0700337 struct delayed_work mbhc_insert_dwork;
338 unsigned long mbhc_last_resume; /* in jiffies */
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -0700339
340 u8 current_plug;
341 struct work_struct hs_correct_plug_work;
342 bool hs_detect_work_stop;
343 bool hs_polling_irq_prepared;
344 bool lpi_enabled; /* low power insertion detection */
345 bool in_gpio_handler;
346 /* Currently, only used for mbhc purpose, to protect
347 * concurrent execution of mbhc threaded irq handlers and
348 * kill race between DAPM and MBHC.But can serve as a
349 * general lock to protect codec resource
350 */
351 struct mutex codec_resource_lock;
352
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -0700353 /* Work to perform polling on microphone voltage
354 * in order to correct plug type once plug type
355 * is detected as headphone
356 */
357 struct work_struct hs_correct_plug_work_nogpio;
358
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -0700359 bool gpio_irq_resend;
360 struct wake_lock irq_resend_wlock;
361
Bradley Rubincb3950a2011-08-18 13:07:26 -0700362#ifdef CONFIG_DEBUG_FS
Joonwoo Park179b9ec2012-03-26 10:56:20 -0700363 struct dentry *debugfs_poke;
364 struct dentry *debugfs_mbhc;
Bradley Rubincb3950a2011-08-18 13:07:26 -0700365#endif
Joonwoo Park179b9ec2012-03-26 10:56:20 -0700366};
367
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800368static const u32 comp_shift[] = {
369 0,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700370 1,
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800371};
372
373static const int comp_rx_path[] = {
374 COMPANDER_1,
375 COMPANDER_1,
376 COMPANDER_2,
377 COMPANDER_2,
378 COMPANDER_2,
379 COMPANDER_2,
380 COMPANDER_MAX,
381};
382
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700383static const struct comp_sample_dependent_params
384 comp_samp_params[COMPANDER_FS_MAX] = {
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800385 {
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700386 .peak_det_timeout = 0x6,
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800387 .rms_meter_div_fact = 0x9 << 4,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700388 .rms_meter_resamp_fact = 0x06,
389 .shutdown_timeout = COMP_SHUTDWN_TIMEOUT_PCM_240 << 3,
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800390 },
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800391 {
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700392 .peak_det_timeout = 0x7,
393 .rms_meter_div_fact = 0xA << 4,
394 .rms_meter_resamp_fact = 0x0C,
395 .shutdown_timeout = COMP_SHUTDWN_TIMEOUT_PCM_480 << 3,
396 },
397 {
398 .peak_det_timeout = 0x8,
399 .rms_meter_div_fact = 0xB << 4,
400 .rms_meter_resamp_fact = 0x30,
401 .shutdown_timeout = COMP_SHUTDWN_TIMEOUT_PCM_960 << 3,
402 },
403 {
404 .peak_det_timeout = 0x9,
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800405 .rms_meter_div_fact = 0xB << 4,
406 .rms_meter_resamp_fact = 0x28,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700407 .shutdown_timeout = COMP_SHUTDWN_TIMEOUT_PCM_1440 << 3,
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800408 },
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800409 {
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700410 .peak_det_timeout = 0xA,
411 .rms_meter_div_fact = 0xC << 4,
412 .rms_meter_resamp_fact = 0x50,
413 .shutdown_timeout = COMP_SHUTDWN_TIMEOUT_PCM_2880 << 3,
414 },
415 {
416 .peak_det_timeout = 0xB,
417 .rms_meter_div_fact = 0xC << 4,
418 .rms_meter_resamp_fact = 0x50,
419 .shutdown_timeout = COMP_SHUTDWN_TIMEOUT_PCM_5760 << 3,
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800420 },
421};
422
Kuirong Wange9c8a222012-03-28 16:24:09 -0700423static unsigned short rx_digital_gain_reg[] = {
424 TABLA_A_CDC_RX1_VOL_CTL_B2_CTL,
425 TABLA_A_CDC_RX2_VOL_CTL_B2_CTL,
426 TABLA_A_CDC_RX3_VOL_CTL_B2_CTL,
427 TABLA_A_CDC_RX4_VOL_CTL_B2_CTL,
428 TABLA_A_CDC_RX5_VOL_CTL_B2_CTL,
429 TABLA_A_CDC_RX6_VOL_CTL_B2_CTL,
430 TABLA_A_CDC_RX7_VOL_CTL_B2_CTL,
431};
432
433
434static unsigned short tx_digital_gain_reg[] = {
435 TABLA_A_CDC_TX1_VOL_CTL_GAIN,
436 TABLA_A_CDC_TX2_VOL_CTL_GAIN,
437 TABLA_A_CDC_TX3_VOL_CTL_GAIN,
438 TABLA_A_CDC_TX4_VOL_CTL_GAIN,
439 TABLA_A_CDC_TX5_VOL_CTL_GAIN,
440 TABLA_A_CDC_TX6_VOL_CTL_GAIN,
441 TABLA_A_CDC_TX7_VOL_CTL_GAIN,
442 TABLA_A_CDC_TX8_VOL_CTL_GAIN,
443 TABLA_A_CDC_TX9_VOL_CTL_GAIN,
444 TABLA_A_CDC_TX10_VOL_CTL_GAIN,
445};
446
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700447static int tabla_codec_enable_charge_pump(struct snd_soc_dapm_widget *w,
448 struct snd_kcontrol *kcontrol, int event)
449{
450 struct snd_soc_codec *codec = w->codec;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700451
452 pr_debug("%s %d\n", __func__, event);
453 switch (event) {
454 case SND_SOC_DAPM_POST_PMU:
ty.leecd34e122012-08-23 21:33:19 +0900455 msleep(15);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700456 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_OTHR_CTL, 0x01,
457 0x01);
458 snd_soc_update_bits(codec, TABLA_A_CDC_CLSG_CTL, 0x08, 0x08);
459 usleep_range(200, 200);
460 snd_soc_update_bits(codec, TABLA_A_CP_STATIC, 0x10, 0x00);
461 break;
462 case SND_SOC_DAPM_PRE_PMD:
463 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_OTHR_RESET_CTL, 0x10,
464 0x10);
465 usleep_range(20, 20);
466 snd_soc_update_bits(codec, TABLA_A_CP_STATIC, 0x08, 0x08);
467 snd_soc_update_bits(codec, TABLA_A_CP_STATIC, 0x10, 0x10);
468 snd_soc_update_bits(codec, TABLA_A_CDC_CLSG_CTL, 0x08, 0x00);
469 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_OTHR_CTL, 0x01,
470 0x00);
471 snd_soc_update_bits(codec, TABLA_A_CP_STATIC, 0x08, 0x00);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700472 break;
473 }
474 return 0;
475}
476
Bradley Rubina7096d02011-08-03 18:29:02 -0700477static int tabla_get_anc_slot(struct snd_kcontrol *kcontrol,
478 struct snd_ctl_elem_value *ucontrol)
479{
480 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
481 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
482 ucontrol->value.integer.value[0] = tabla->anc_slot;
483 return 0;
484}
485
486static int tabla_put_anc_slot(struct snd_kcontrol *kcontrol,
487 struct snd_ctl_elem_value *ucontrol)
488{
489 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
490 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
491 tabla->anc_slot = ucontrol->value.integer.value[0];
492 return 0;
493}
494
Kiran Kandid2d86b52011-09-09 17:44:28 -0700495static int tabla_pa_gain_get(struct snd_kcontrol *kcontrol,
496 struct snd_ctl_elem_value *ucontrol)
497{
498 u8 ear_pa_gain;
499 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
500
501 ear_pa_gain = snd_soc_read(codec, TABLA_A_RX_EAR_GAIN);
502
503 ear_pa_gain = ear_pa_gain >> 5;
504
505 if (ear_pa_gain == 0x00) {
506 ucontrol->value.integer.value[0] = 0;
507 } else if (ear_pa_gain == 0x04) {
508 ucontrol->value.integer.value[0] = 1;
509 } else {
510 pr_err("%s: ERROR: Unsupported Ear Gain = 0x%x\n",
511 __func__, ear_pa_gain);
512 return -EINVAL;
513 }
514
515 pr_debug("%s: ear_pa_gain = 0x%x\n", __func__, ear_pa_gain);
516
517 return 0;
518}
519
520static int tabla_pa_gain_put(struct snd_kcontrol *kcontrol,
521 struct snd_ctl_elem_value *ucontrol)
522{
523 u8 ear_pa_gain;
524 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
525
526 pr_debug("%s: ucontrol->value.integer.value[0] = %ld\n", __func__,
527 ucontrol->value.integer.value[0]);
528
529 switch (ucontrol->value.integer.value[0]) {
530 case 0:
531 ear_pa_gain = 0x00;
532 break;
533 case 1:
534 ear_pa_gain = 0x80;
535 break;
536 default:
537 return -EINVAL;
538 }
539
540 snd_soc_update_bits(codec, TABLA_A_RX_EAR_GAIN, 0xE0, ear_pa_gain);
541 return 0;
542}
543
Ben Romberger1f045a72011-11-04 10:14:57 -0700544static int tabla_get_iir_enable_audio_mixer(
545 struct snd_kcontrol *kcontrol,
546 struct snd_ctl_elem_value *ucontrol)
547{
548 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
549 int iir_idx = ((struct soc_multi_mixer_control *)
550 kcontrol->private_value)->reg;
551 int band_idx = ((struct soc_multi_mixer_control *)
552 kcontrol->private_value)->shift;
553
554 ucontrol->value.integer.value[0] =
555 snd_soc_read(codec, (TABLA_A_CDC_IIR1_CTL + 16 * iir_idx)) &
556 (1 << band_idx);
557
558 pr_debug("%s: IIR #%d band #%d enable %d\n", __func__,
559 iir_idx, band_idx,
560 (uint32_t)ucontrol->value.integer.value[0]);
561 return 0;
562}
563
564static int tabla_put_iir_enable_audio_mixer(
565 struct snd_kcontrol *kcontrol,
566 struct snd_ctl_elem_value *ucontrol)
567{
568 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
569 int iir_idx = ((struct soc_multi_mixer_control *)
570 kcontrol->private_value)->reg;
571 int band_idx = ((struct soc_multi_mixer_control *)
572 kcontrol->private_value)->shift;
573 int value = ucontrol->value.integer.value[0];
574
575 /* Mask first 5 bits, 6-8 are reserved */
576 snd_soc_update_bits(codec, (TABLA_A_CDC_IIR1_CTL + 16 * iir_idx),
577 (1 << band_idx), (value << band_idx));
578
579 pr_debug("%s: IIR #%d band #%d enable %d\n", __func__,
580 iir_idx, band_idx, value);
581 return 0;
582}
583static uint32_t get_iir_band_coeff(struct snd_soc_codec *codec,
584 int iir_idx, int band_idx,
585 int coeff_idx)
586{
587 /* Address does not automatically update if reading */
Ben Romberger0915aae2012-02-06 23:32:43 -0800588 snd_soc_write(codec,
Ben Romberger1f045a72011-11-04 10:14:57 -0700589 (TABLA_A_CDC_IIR1_COEF_B1_CTL + 16 * iir_idx),
Ben Romberger0915aae2012-02-06 23:32:43 -0800590 (band_idx * BAND_MAX + coeff_idx) & 0x1F);
Ben Romberger1f045a72011-11-04 10:14:57 -0700591
592 /* Mask bits top 2 bits since they are reserved */
593 return ((snd_soc_read(codec,
594 (TABLA_A_CDC_IIR1_COEF_B2_CTL + 16 * iir_idx)) << 24) |
595 (snd_soc_read(codec,
596 (TABLA_A_CDC_IIR1_COEF_B3_CTL + 16 * iir_idx)) << 16) |
597 (snd_soc_read(codec,
598 (TABLA_A_CDC_IIR1_COEF_B4_CTL + 16 * iir_idx)) << 8) |
599 (snd_soc_read(codec,
600 (TABLA_A_CDC_IIR1_COEF_B5_CTL + 16 * iir_idx)))) &
601 0x3FFFFFFF;
602}
603
604static int tabla_get_iir_band_audio_mixer(
605 struct snd_kcontrol *kcontrol,
606 struct snd_ctl_elem_value *ucontrol)
607{
608 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
609 int iir_idx = ((struct soc_multi_mixer_control *)
610 kcontrol->private_value)->reg;
611 int band_idx = ((struct soc_multi_mixer_control *)
612 kcontrol->private_value)->shift;
613
614 ucontrol->value.integer.value[0] =
615 get_iir_band_coeff(codec, iir_idx, band_idx, 0);
616 ucontrol->value.integer.value[1] =
617 get_iir_band_coeff(codec, iir_idx, band_idx, 1);
618 ucontrol->value.integer.value[2] =
619 get_iir_band_coeff(codec, iir_idx, band_idx, 2);
620 ucontrol->value.integer.value[3] =
621 get_iir_band_coeff(codec, iir_idx, band_idx, 3);
622 ucontrol->value.integer.value[4] =
623 get_iir_band_coeff(codec, iir_idx, band_idx, 4);
624
625 pr_debug("%s: IIR #%d band #%d b0 = 0x%x\n"
626 "%s: IIR #%d band #%d b1 = 0x%x\n"
627 "%s: IIR #%d band #%d b2 = 0x%x\n"
628 "%s: IIR #%d band #%d a1 = 0x%x\n"
629 "%s: IIR #%d band #%d a2 = 0x%x\n",
630 __func__, iir_idx, band_idx,
631 (uint32_t)ucontrol->value.integer.value[0],
632 __func__, iir_idx, band_idx,
633 (uint32_t)ucontrol->value.integer.value[1],
634 __func__, iir_idx, band_idx,
635 (uint32_t)ucontrol->value.integer.value[2],
636 __func__, iir_idx, band_idx,
637 (uint32_t)ucontrol->value.integer.value[3],
638 __func__, iir_idx, band_idx,
639 (uint32_t)ucontrol->value.integer.value[4]);
640 return 0;
641}
642
643static void set_iir_band_coeff(struct snd_soc_codec *codec,
644 int iir_idx, int band_idx,
645 int coeff_idx, uint32_t value)
646{
647 /* Mask top 3 bits, 6-8 are reserved */
648 /* Update address manually each time */
Ben Romberger0915aae2012-02-06 23:32:43 -0800649 snd_soc_write(codec,
Ben Romberger1f045a72011-11-04 10:14:57 -0700650 (TABLA_A_CDC_IIR1_COEF_B1_CTL + 16 * iir_idx),
Ben Romberger0915aae2012-02-06 23:32:43 -0800651 (band_idx * BAND_MAX + coeff_idx) & 0x1F);
Ben Romberger1f045a72011-11-04 10:14:57 -0700652
653 /* Mask top 2 bits, 7-8 are reserved */
Ben Romberger0915aae2012-02-06 23:32:43 -0800654 snd_soc_write(codec,
Ben Romberger1f045a72011-11-04 10:14:57 -0700655 (TABLA_A_CDC_IIR1_COEF_B2_CTL + 16 * iir_idx),
Ben Romberger0915aae2012-02-06 23:32:43 -0800656 (value >> 24) & 0x3F);
Ben Romberger1f045a72011-11-04 10:14:57 -0700657
658 /* Isolate 8bits at a time */
Ben Romberger0915aae2012-02-06 23:32:43 -0800659 snd_soc_write(codec,
Ben Romberger1f045a72011-11-04 10:14:57 -0700660 (TABLA_A_CDC_IIR1_COEF_B3_CTL + 16 * iir_idx),
Ben Romberger0915aae2012-02-06 23:32:43 -0800661 (value >> 16) & 0xFF);
Ben Romberger1f045a72011-11-04 10:14:57 -0700662
Ben Romberger0915aae2012-02-06 23:32:43 -0800663 snd_soc_write(codec,
Ben Romberger1f045a72011-11-04 10:14:57 -0700664 (TABLA_A_CDC_IIR1_COEF_B4_CTL + 16 * iir_idx),
Ben Romberger0915aae2012-02-06 23:32:43 -0800665 (value >> 8) & 0xFF);
Ben Romberger1f045a72011-11-04 10:14:57 -0700666
Ben Romberger0915aae2012-02-06 23:32:43 -0800667 snd_soc_write(codec,
Ben Romberger1f045a72011-11-04 10:14:57 -0700668 (TABLA_A_CDC_IIR1_COEF_B5_CTL + 16 * iir_idx),
Ben Romberger0915aae2012-02-06 23:32:43 -0800669 value & 0xFF);
Ben Romberger1f045a72011-11-04 10:14:57 -0700670}
671
672static int tabla_put_iir_band_audio_mixer(
673 struct snd_kcontrol *kcontrol,
674 struct snd_ctl_elem_value *ucontrol)
675{
676 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
677 int iir_idx = ((struct soc_multi_mixer_control *)
678 kcontrol->private_value)->reg;
679 int band_idx = ((struct soc_multi_mixer_control *)
680 kcontrol->private_value)->shift;
681
682 set_iir_band_coeff(codec, iir_idx, band_idx, 0,
683 ucontrol->value.integer.value[0]);
684 set_iir_band_coeff(codec, iir_idx, band_idx, 1,
685 ucontrol->value.integer.value[1]);
686 set_iir_band_coeff(codec, iir_idx, band_idx, 2,
687 ucontrol->value.integer.value[2]);
688 set_iir_band_coeff(codec, iir_idx, band_idx, 3,
689 ucontrol->value.integer.value[3]);
690 set_iir_band_coeff(codec, iir_idx, band_idx, 4,
691 ucontrol->value.integer.value[4]);
692
693 pr_debug("%s: IIR #%d band #%d b0 = 0x%x\n"
694 "%s: IIR #%d band #%d b1 = 0x%x\n"
695 "%s: IIR #%d band #%d b2 = 0x%x\n"
696 "%s: IIR #%d band #%d a1 = 0x%x\n"
697 "%s: IIR #%d band #%d a2 = 0x%x\n",
698 __func__, iir_idx, band_idx,
699 get_iir_band_coeff(codec, iir_idx, band_idx, 0),
700 __func__, iir_idx, band_idx,
701 get_iir_band_coeff(codec, iir_idx, band_idx, 1),
702 __func__, iir_idx, band_idx,
703 get_iir_band_coeff(codec, iir_idx, band_idx, 2),
704 __func__, iir_idx, band_idx,
705 get_iir_band_coeff(codec, iir_idx, band_idx, 3),
706 __func__, iir_idx, band_idx,
707 get_iir_band_coeff(codec, iir_idx, band_idx, 4));
708 return 0;
709}
710
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800711static int tabla_compander_gain_offset(
712 struct snd_soc_codec *codec, u32 enable,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700713 unsigned int reg, int mask, int event, u32 comp)
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800714{
715 int pa_mode = snd_soc_read(codec, reg) & mask;
716 int gain_offset = 0;
717 /* if PMU && enable is 1-> offset is 3
718 * if PMU && enable is 0-> offset is 0
719 * if PMD && pa_mode is PA -> offset is 0: PMU compander is off
720 * if PMD && pa_mode is comp -> offset is -3: PMU compander is on.
721 */
722
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700723 if (SND_SOC_DAPM_EVENT_ON(event) && (enable != 0)) {
724 if (comp == COMPANDER_1)
725 gain_offset = TABLA_COMP_DIGITAL_GAIN_HP_OFFSET;
726 if (comp == COMPANDER_2)
727 gain_offset = TABLA_COMP_DIGITAL_GAIN_LINEOUT_OFFSET;
728 }
729 if (SND_SOC_DAPM_EVENT_OFF(event) && (pa_mode == 0)) {
730 if (comp == COMPANDER_1)
731 gain_offset = -TABLA_COMP_DIGITAL_GAIN_HP_OFFSET;
732 if (comp == COMPANDER_2)
733 gain_offset = -TABLA_COMP_DIGITAL_GAIN_LINEOUT_OFFSET;
734
735 }
736 pr_debug("%s: compander #%d gain_offset %d\n",
737 __func__, comp + 1, gain_offset);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800738 return gain_offset;
739}
740
741
742static int tabla_config_gain_compander(
743 struct snd_soc_codec *codec,
744 u32 compander, u32 enable, int event)
745{
746 int value = 0;
747 int mask = 1 << 4;
748 int gain = 0;
749 int gain_offset;
750 if (compander >= COMPANDER_MAX) {
751 pr_err("%s: Error, invalid compander channel\n", __func__);
752 return -EINVAL;
753 }
754
755 if ((enable == 0) || SND_SOC_DAPM_EVENT_OFF(event))
756 value = 1 << 4;
757
758 if (compander == COMPANDER_1) {
759 gain_offset = tabla_compander_gain_offset(codec, enable,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700760 TABLA_A_RX_HPH_L_GAIN, mask, event, compander);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800761 snd_soc_update_bits(codec, TABLA_A_RX_HPH_L_GAIN, mask, value);
762 gain = snd_soc_read(codec, TABLA_A_CDC_RX1_VOL_CTL_B2_CTL);
763 snd_soc_update_bits(codec, TABLA_A_CDC_RX1_VOL_CTL_B2_CTL,
764 0xFF, gain - gain_offset);
765 gain_offset = tabla_compander_gain_offset(codec, enable,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700766 TABLA_A_RX_HPH_R_GAIN, mask, event, compander);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800767 snd_soc_update_bits(codec, TABLA_A_RX_HPH_R_GAIN, mask, value);
768 gain = snd_soc_read(codec, TABLA_A_CDC_RX2_VOL_CTL_B2_CTL);
769 snd_soc_update_bits(codec, TABLA_A_CDC_RX2_VOL_CTL_B2_CTL,
770 0xFF, gain - gain_offset);
771 } else if (compander == COMPANDER_2) {
772 gain_offset = tabla_compander_gain_offset(codec, enable,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700773 TABLA_A_RX_LINE_1_GAIN, mask, event, compander);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800774 snd_soc_update_bits(codec, TABLA_A_RX_LINE_1_GAIN, mask, value);
775 gain = snd_soc_read(codec, TABLA_A_CDC_RX3_VOL_CTL_B2_CTL);
776 snd_soc_update_bits(codec, TABLA_A_CDC_RX3_VOL_CTL_B2_CTL,
777 0xFF, gain - gain_offset);
778 gain_offset = tabla_compander_gain_offset(codec, enable,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700779 TABLA_A_RX_LINE_3_GAIN, mask, event, compander);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800780 snd_soc_update_bits(codec, TABLA_A_RX_LINE_3_GAIN, mask, value);
781 gain = snd_soc_read(codec, TABLA_A_CDC_RX4_VOL_CTL_B2_CTL);
782 snd_soc_update_bits(codec, TABLA_A_CDC_RX4_VOL_CTL_B2_CTL,
783 0xFF, gain - gain_offset);
784 gain_offset = tabla_compander_gain_offset(codec, enable,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700785 TABLA_A_RX_LINE_2_GAIN, mask, event, compander);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800786 snd_soc_update_bits(codec, TABLA_A_RX_LINE_2_GAIN, mask, value);
787 gain = snd_soc_read(codec, TABLA_A_CDC_RX5_VOL_CTL_B2_CTL);
788 snd_soc_update_bits(codec, TABLA_A_CDC_RX5_VOL_CTL_B2_CTL,
789 0xFF, gain - gain_offset);
790 gain_offset = tabla_compander_gain_offset(codec, enable,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700791 TABLA_A_RX_LINE_4_GAIN, mask, event, compander);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800792 snd_soc_update_bits(codec, TABLA_A_RX_LINE_4_GAIN, mask, value);
793 gain = snd_soc_read(codec, TABLA_A_CDC_RX6_VOL_CTL_B2_CTL);
794 snd_soc_update_bits(codec, TABLA_A_CDC_RX6_VOL_CTL_B2_CTL,
795 0xFF, gain - gain_offset);
796 }
797 return 0;
798}
799static int tabla_get_compander(struct snd_kcontrol *kcontrol,
800 struct snd_ctl_elem_value *ucontrol)
801{
802
803 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
804 int comp = ((struct soc_multi_mixer_control *)
805 kcontrol->private_value)->max;
806 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
807
808 ucontrol->value.integer.value[0] = tabla->comp_enabled[comp];
809
810 return 0;
811}
812
813static int tabla_set_compander(struct snd_kcontrol *kcontrol,
814 struct snd_ctl_elem_value *ucontrol)
815{
816 struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
817 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
818 int comp = ((struct soc_multi_mixer_control *)
819 kcontrol->private_value)->max;
820 int value = ucontrol->value.integer.value[0];
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700821 pr_debug("%s: compander #%d enable %d\n",
822 __func__, comp + 1, value);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800823 if (value == tabla->comp_enabled[comp]) {
824 pr_debug("%s: compander #%d enable %d no change\n",
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700825 __func__, comp + 1, value);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800826 return 0;
827 }
828 tabla->comp_enabled[comp] = value;
829 return 0;
830}
831
832
833static int tabla_config_compander(struct snd_soc_dapm_widget *w,
834 struct snd_kcontrol *kcontrol,
835 int event)
836{
837 struct snd_soc_codec *codec = w->codec;
838 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
839 u32 rate = tabla->comp_fs[w->shift];
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700840 u32 status;
841 unsigned long timeout;
842 pr_debug("%s: compander #%d enable %d event %d\n",
843 __func__, w->shift + 1,
844 tabla->comp_enabled[w->shift], event);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800845 switch (event) {
846 case SND_SOC_DAPM_PRE_PMU:
847 if (tabla->comp_enabled[w->shift] != 0) {
848 /* Enable both L/R compander clocks */
849 snd_soc_update_bits(codec,
850 TABLA_A_CDC_CLK_RX_B2_CTL,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700851 1 << comp_shift[w->shift],
852 1 << comp_shift[w->shift]);
853 /* Clear the HALT for the compander*/
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800854 snd_soc_update_bits(codec,
855 TABLA_A_CDC_COMP1_B1_CTL +
856 w->shift * 8, 1 << 2, 0);
857 /* Toggle compander reset bits*/
858 snd_soc_update_bits(codec,
859 TABLA_A_CDC_CLK_OTHR_RESET_CTL,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700860 1 << comp_shift[w->shift],
861 1 << comp_shift[w->shift]);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800862 snd_soc_update_bits(codec,
863 TABLA_A_CDC_CLK_OTHR_RESET_CTL,
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700864 1 << comp_shift[w->shift], 0);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800865 tabla_config_gain_compander(codec, w->shift, 1, event);
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700866 /* Compander enable -> 0x370/0x378*/
867 snd_soc_update_bits(codec, TABLA_A_CDC_COMP1_B1_CTL +
868 w->shift * 8, 0x03, 0x03);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800869 /* Update the RMS meter resampling*/
870 snd_soc_update_bits(codec,
871 TABLA_A_CDC_COMP1_B3_CTL +
872 w->shift * 8, 0xFF, 0x01);
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700873 snd_soc_update_bits(codec,
874 TABLA_A_CDC_COMP1_B2_CTL +
875 w->shift * 8, 0xF0, 0x50);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800876 /* Wait for 1ms*/
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700877 usleep_range(5000, 5000);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800878 }
879 break;
880 case SND_SOC_DAPM_POST_PMU:
881 /* Set sample rate dependent paramater*/
882 if (tabla->comp_enabled[w->shift] != 0) {
883 snd_soc_update_bits(codec, TABLA_A_CDC_COMP1_FS_CFG +
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700884 w->shift * 8, 0x07, rate);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800885 snd_soc_update_bits(codec, TABLA_A_CDC_COMP1_B2_CTL +
886 w->shift * 8, 0x0F,
887 comp_samp_params[rate].peak_det_timeout);
888 snd_soc_update_bits(codec, TABLA_A_CDC_COMP1_B2_CTL +
889 w->shift * 8, 0xF0,
890 comp_samp_params[rate].rms_meter_div_fact);
891 snd_soc_update_bits(codec, TABLA_A_CDC_COMP1_B3_CTL +
892 w->shift * 8, 0xFF,
893 comp_samp_params[rate].rms_meter_resamp_fact);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800894 snd_soc_update_bits(codec, TABLA_A_CDC_COMP1_B1_CTL +
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700895 w->shift * 8, 0x38,
896 comp_samp_params[rate].shutdown_timeout);
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800897 }
898 break;
899 case SND_SOC_DAPM_PRE_PMD:
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700900 if (tabla->comp_enabled[w->shift] != 0) {
901 status = snd_soc_read(codec,
902 TABLA_A_CDC_COMP1_SHUT_DOWN_STATUS +
903 w->shift * 8);
904 pr_debug("%s: compander #%d shutdown status %d in event %d\n",
905 __func__, w->shift + 1, status, event);
906 /* Halt the compander*/
907 snd_soc_update_bits(codec, TABLA_A_CDC_COMP1_B1_CTL +
908 w->shift * 8, 1 << 2, 1 << 2);
909 }
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800910 break;
911 case SND_SOC_DAPM_POST_PMD:
Kuirong Wangbc5ac042012-06-26 15:35:22 -0700912 if (tabla->comp_enabled[w->shift] != 0) {
913 /* Wait up to a second for shutdown complete */
914 timeout = jiffies + HZ;
915 do {
916 status = snd_soc_read(codec,
917 TABLA_A_CDC_COMP1_SHUT_DOWN_STATUS +
918 w->shift * 8);
919 if (status == 0x3)
920 break;
921 usleep_range(5000, 5000);
922 } while (!(time_after(jiffies, timeout)));
923 /* Restore the gain */
924 tabla_config_gain_compander(codec, w->shift,
925 tabla->comp_enabled[w->shift],
926 event);
927 /* Disable the compander*/
928 snd_soc_update_bits(codec, TABLA_A_CDC_COMP1_B1_CTL +
929 w->shift * 8, 0x03, 0x00);
930 /* Turn off the clock for compander in pair*/
931 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_RX_B2_CTL,
932 0x03 << comp_shift[w->shift], 0);
933 /* Clear the HALT for the compander*/
934 snd_soc_update_bits(codec,
935 TABLA_A_CDC_COMP1_B1_CTL +
936 w->shift * 8, 1 << 2, 0);
937 }
Kuirong Wang0f8ade32012-02-27 16:29:45 -0800938 break;
939 }
940 return 0;
941}
942
Kiran Kandid2d86b52011-09-09 17:44:28 -0700943static const char *tabla_ear_pa_gain_text[] = {"POS_6_DB", "POS_2_DB"};
944static const struct soc_enum tabla_ear_pa_gain_enum[] = {
945 SOC_ENUM_SINGLE_EXT(2, tabla_ear_pa_gain_text),
946};
947
Santosh Mardi024010f2011-10-18 06:27:21 +0530948/*cut of frequency for high pass filter*/
949static const char *cf_text[] = {
950 "MIN_3DB_4Hz", "MIN_3DB_75Hz", "MIN_3DB_150Hz"
951};
952
953static const struct soc_enum cf_dec1_enum =
954 SOC_ENUM_SINGLE(TABLA_A_CDC_TX1_MUX_CTL, 4, 3, cf_text);
955
956static const struct soc_enum cf_dec2_enum =
957 SOC_ENUM_SINGLE(TABLA_A_CDC_TX2_MUX_CTL, 4, 3, cf_text);
958
959static const struct soc_enum cf_dec3_enum =
960 SOC_ENUM_SINGLE(TABLA_A_CDC_TX3_MUX_CTL, 4, 3, cf_text);
961
962static const struct soc_enum cf_dec4_enum =
963 SOC_ENUM_SINGLE(TABLA_A_CDC_TX4_MUX_CTL, 4, 3, cf_text);
964
965static const struct soc_enum cf_dec5_enum =
966 SOC_ENUM_SINGLE(TABLA_A_CDC_TX5_MUX_CTL, 4, 3, cf_text);
967
968static const struct soc_enum cf_dec6_enum =
969 SOC_ENUM_SINGLE(TABLA_A_CDC_TX6_MUX_CTL, 4, 3, cf_text);
970
971static const struct soc_enum cf_dec7_enum =
972 SOC_ENUM_SINGLE(TABLA_A_CDC_TX7_MUX_CTL, 4, 3, cf_text);
973
974static const struct soc_enum cf_dec8_enum =
975 SOC_ENUM_SINGLE(TABLA_A_CDC_TX8_MUX_CTL, 4, 3, cf_text);
976
977static const struct soc_enum cf_dec9_enum =
978 SOC_ENUM_SINGLE(TABLA_A_CDC_TX9_MUX_CTL, 4, 3, cf_text);
979
980static const struct soc_enum cf_dec10_enum =
981 SOC_ENUM_SINGLE(TABLA_A_CDC_TX10_MUX_CTL, 4, 3, cf_text);
982
983static const struct soc_enum cf_rxmix1_enum =
984 SOC_ENUM_SINGLE(TABLA_A_CDC_RX1_B4_CTL, 1, 3, cf_text);
985
986static const struct soc_enum cf_rxmix2_enum =
987 SOC_ENUM_SINGLE(TABLA_A_CDC_RX2_B4_CTL, 1, 3, cf_text);
988
989static const struct soc_enum cf_rxmix3_enum =
990 SOC_ENUM_SINGLE(TABLA_A_CDC_RX3_B4_CTL, 1, 3, cf_text);
991
992static const struct soc_enum cf_rxmix4_enum =
993 SOC_ENUM_SINGLE(TABLA_A_CDC_RX4_B4_CTL, 1, 3, cf_text);
994
995static const struct soc_enum cf_rxmix5_enum =
996 SOC_ENUM_SINGLE(TABLA_A_CDC_RX5_B4_CTL, 1, 3, cf_text)
997;
998static const struct soc_enum cf_rxmix6_enum =
999 SOC_ENUM_SINGLE(TABLA_A_CDC_RX6_B4_CTL, 1, 3, cf_text);
1000
1001static const struct soc_enum cf_rxmix7_enum =
1002 SOC_ENUM_SINGLE(TABLA_A_CDC_RX7_B4_CTL, 1, 3, cf_text);
1003
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001004static const struct snd_kcontrol_new tabla_snd_controls[] = {
Kiran Kandid2d86b52011-09-09 17:44:28 -07001005
1006 SOC_ENUM_EXT("EAR PA Gain", tabla_ear_pa_gain_enum[0],
1007 tabla_pa_gain_get, tabla_pa_gain_put),
1008
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001009 SOC_SINGLE_TLV("LINEOUT1 Volume", TABLA_A_RX_LINE_1_GAIN, 0, 12, 1,
1010 line_gain),
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001011 SOC_SINGLE_TLV("LINEOUT2 Volume", TABLA_A_RX_LINE_2_GAIN, 0, 12, 1,
1012 line_gain),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001013 SOC_SINGLE_TLV("LINEOUT3 Volume", TABLA_A_RX_LINE_3_GAIN, 0, 12, 1,
1014 line_gain),
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001015 SOC_SINGLE_TLV("LINEOUT4 Volume", TABLA_A_RX_LINE_4_GAIN, 0, 12, 1,
1016 line_gain),
Bhalchandra Gajare0a8ad172011-08-12 13:32:22 -07001017 SOC_SINGLE_TLV("LINEOUT5 Volume", TABLA_A_RX_LINE_5_GAIN, 0, 12, 1,
1018 line_gain),
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001019
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001020 SOC_SINGLE_TLV("HPHL Volume", TABLA_A_RX_HPH_L_GAIN, 0, 12, 1,
1021 line_gain),
1022 SOC_SINGLE_TLV("HPHR Volume", TABLA_A_RX_HPH_R_GAIN, 0, 12, 1,
1023 line_gain),
1024
Bradley Rubin410383f2011-07-22 13:44:23 -07001025 SOC_SINGLE_S8_TLV("RX1 Digital Volume", TABLA_A_CDC_RX1_VOL_CTL_B2_CTL,
1026 -84, 40, digital_gain),
1027 SOC_SINGLE_S8_TLV("RX2 Digital Volume", TABLA_A_CDC_RX2_VOL_CTL_B2_CTL,
1028 -84, 40, digital_gain),
1029 SOC_SINGLE_S8_TLV("RX3 Digital Volume", TABLA_A_CDC_RX3_VOL_CTL_B2_CTL,
1030 -84, 40, digital_gain),
1031 SOC_SINGLE_S8_TLV("RX4 Digital Volume", TABLA_A_CDC_RX4_VOL_CTL_B2_CTL,
1032 -84, 40, digital_gain),
1033 SOC_SINGLE_S8_TLV("RX5 Digital Volume", TABLA_A_CDC_RX5_VOL_CTL_B2_CTL,
1034 -84, 40, digital_gain),
1035 SOC_SINGLE_S8_TLV("RX6 Digital Volume", TABLA_A_CDC_RX6_VOL_CTL_B2_CTL,
1036 -84, 40, digital_gain),
Neema Shettyd3a89262012-02-16 10:23:50 -08001037 SOC_SINGLE_S8_TLV("RX7 Digital Volume", TABLA_A_CDC_RX7_VOL_CTL_B2_CTL,
1038 -84, 40, digital_gain),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001039
Bradley Rubin410383f2011-07-22 13:44:23 -07001040 SOC_SINGLE_S8_TLV("DEC1 Volume", TABLA_A_CDC_TX1_VOL_CTL_GAIN, -84, 40,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001041 digital_gain),
Bradley Rubin410383f2011-07-22 13:44:23 -07001042 SOC_SINGLE_S8_TLV("DEC2 Volume", TABLA_A_CDC_TX2_VOL_CTL_GAIN, -84, 40,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001043 digital_gain),
Bradley Rubin410383f2011-07-22 13:44:23 -07001044 SOC_SINGLE_S8_TLV("DEC3 Volume", TABLA_A_CDC_TX3_VOL_CTL_GAIN, -84, 40,
1045 digital_gain),
1046 SOC_SINGLE_S8_TLV("DEC4 Volume", TABLA_A_CDC_TX4_VOL_CTL_GAIN, -84, 40,
1047 digital_gain),
1048 SOC_SINGLE_S8_TLV("DEC5 Volume", TABLA_A_CDC_TX5_VOL_CTL_GAIN, -84, 40,
1049 digital_gain),
1050 SOC_SINGLE_S8_TLV("DEC6 Volume", TABLA_A_CDC_TX6_VOL_CTL_GAIN, -84, 40,
1051 digital_gain),
1052 SOC_SINGLE_S8_TLV("DEC7 Volume", TABLA_A_CDC_TX7_VOL_CTL_GAIN, -84, 40,
1053 digital_gain),
1054 SOC_SINGLE_S8_TLV("DEC8 Volume", TABLA_A_CDC_TX8_VOL_CTL_GAIN, -84, 40,
1055 digital_gain),
1056 SOC_SINGLE_S8_TLV("DEC9 Volume", TABLA_A_CDC_TX9_VOL_CTL_GAIN, -84, 40,
1057 digital_gain),
1058 SOC_SINGLE_S8_TLV("DEC10 Volume", TABLA_A_CDC_TX10_VOL_CTL_GAIN, -84,
1059 40, digital_gain),
Patrick Lai29006372011-09-28 17:57:42 -07001060 SOC_SINGLE_S8_TLV("IIR1 INP1 Volume", TABLA_A_CDC_IIR1_GAIN_B1_CTL, -84,
1061 40, digital_gain),
1062 SOC_SINGLE_S8_TLV("IIR1 INP2 Volume", TABLA_A_CDC_IIR1_GAIN_B2_CTL, -84,
1063 40, digital_gain),
1064 SOC_SINGLE_S8_TLV("IIR1 INP3 Volume", TABLA_A_CDC_IIR1_GAIN_B3_CTL, -84,
1065 40, digital_gain),
1066 SOC_SINGLE_S8_TLV("IIR1 INP4 Volume", TABLA_A_CDC_IIR1_GAIN_B4_CTL, -84,
1067 40, digital_gain),
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001068 SOC_SINGLE_TLV("ADC1 Volume", TABLA_A_TX_1_2_EN, 5, 3, 0, analog_gain),
1069 SOC_SINGLE_TLV("ADC2 Volume", TABLA_A_TX_1_2_EN, 1, 3, 0, analog_gain),
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001070 SOC_SINGLE_TLV("ADC3 Volume", TABLA_A_TX_3_4_EN, 5, 3, 0, analog_gain),
1071 SOC_SINGLE_TLV("ADC4 Volume", TABLA_A_TX_3_4_EN, 1, 3, 0, analog_gain),
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001072 SOC_SINGLE_TLV("ADC5 Volume", TABLA_A_TX_5_6_EN, 5, 3, 0, analog_gain),
1073 SOC_SINGLE_TLV("ADC6 Volume", TABLA_A_TX_5_6_EN, 1, 3, 0, analog_gain),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001074
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001075 SOC_SINGLE_TLV("AUX_PGA_LEFT Volume", TABLA_A_AUX_L_GAIN, 0, 39, 0,
1076 aux_pga_gain),
1077 SOC_SINGLE_TLV("AUX_PGA_RIGHT Volume", TABLA_A_AUX_R_GAIN, 0, 39, 0,
1078 aux_pga_gain),
1079
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001080 SOC_SINGLE("MICBIAS1 CAPLESS Switch", TABLA_A_MICB_1_CTL, 4, 1, 1),
Santosh Mardi680b41e2011-11-22 16:51:16 -08001081 SOC_SINGLE("MICBIAS2 CAPLESS Switch", TABLA_A_MICB_2_CTL, 4, 1, 1),
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001082 SOC_SINGLE("MICBIAS3 CAPLESS Switch", TABLA_A_MICB_3_CTL, 4, 1, 1),
Bradley Rubina7096d02011-08-03 18:29:02 -07001083
1084 SOC_SINGLE_EXT("ANC Slot", SND_SOC_NOPM, 0, 0, 100, tabla_get_anc_slot,
1085 tabla_put_anc_slot),
Santosh Mardi024010f2011-10-18 06:27:21 +05301086 SOC_ENUM("TX1 HPF cut off", cf_dec1_enum),
1087 SOC_ENUM("TX2 HPF cut off", cf_dec2_enum),
1088 SOC_ENUM("TX3 HPF cut off", cf_dec3_enum),
1089 SOC_ENUM("TX4 HPF cut off", cf_dec4_enum),
1090 SOC_ENUM("TX5 HPF cut off", cf_dec5_enum),
1091 SOC_ENUM("TX6 HPF cut off", cf_dec6_enum),
1092 SOC_ENUM("TX7 HPF cut off", cf_dec7_enum),
1093 SOC_ENUM("TX8 HPF cut off", cf_dec8_enum),
1094 SOC_ENUM("TX9 HPF cut off", cf_dec9_enum),
1095 SOC_ENUM("TX10 HPF cut off", cf_dec10_enum),
1096
1097 SOC_SINGLE("TX1 HPF Switch", TABLA_A_CDC_TX1_MUX_CTL, 3, 1, 0),
1098 SOC_SINGLE("TX2 HPF Switch", TABLA_A_CDC_TX2_MUX_CTL, 3, 1, 0),
1099 SOC_SINGLE("TX3 HPF Switch", TABLA_A_CDC_TX3_MUX_CTL, 3, 1, 0),
1100 SOC_SINGLE("TX4 HPF Switch", TABLA_A_CDC_TX4_MUX_CTL, 3, 1, 0),
1101 SOC_SINGLE("TX5 HPF Switch", TABLA_A_CDC_TX5_MUX_CTL, 3, 1, 0),
1102 SOC_SINGLE("TX6 HPF Switch", TABLA_A_CDC_TX6_MUX_CTL, 3, 1, 0),
1103 SOC_SINGLE("TX7 HPF Switch", TABLA_A_CDC_TX7_MUX_CTL, 3, 1, 0),
1104 SOC_SINGLE("TX8 HPF Switch", TABLA_A_CDC_TX8_MUX_CTL, 3, 1, 0),
1105 SOC_SINGLE("TX9 HPF Switch", TABLA_A_CDC_TX9_MUX_CTL, 3, 1, 0),
1106 SOC_SINGLE("TX10 HPF Switch", TABLA_A_CDC_TX10_MUX_CTL, 3, 1, 0),
1107
1108 SOC_SINGLE("RX1 HPF Switch", TABLA_A_CDC_RX1_B5_CTL, 2, 1, 0),
1109 SOC_SINGLE("RX2 HPF Switch", TABLA_A_CDC_RX2_B5_CTL, 2, 1, 0),
1110 SOC_SINGLE("RX3 HPF Switch", TABLA_A_CDC_RX3_B5_CTL, 2, 1, 0),
1111 SOC_SINGLE("RX4 HPF Switch", TABLA_A_CDC_RX4_B5_CTL, 2, 1, 0),
1112 SOC_SINGLE("RX5 HPF Switch", TABLA_A_CDC_RX5_B5_CTL, 2, 1, 0),
1113 SOC_SINGLE("RX6 HPF Switch", TABLA_A_CDC_RX6_B5_CTL, 2, 1, 0),
1114 SOC_SINGLE("RX7 HPF Switch", TABLA_A_CDC_RX7_B5_CTL, 2, 1, 0),
1115
1116 SOC_ENUM("RX1 HPF cut off", cf_rxmix1_enum),
1117 SOC_ENUM("RX2 HPF cut off", cf_rxmix2_enum),
1118 SOC_ENUM("RX3 HPF cut off", cf_rxmix3_enum),
1119 SOC_ENUM("RX4 HPF cut off", cf_rxmix4_enum),
1120 SOC_ENUM("RX5 HPF cut off", cf_rxmix5_enum),
1121 SOC_ENUM("RX6 HPF cut off", cf_rxmix6_enum),
1122 SOC_ENUM("RX7 HPF cut off", cf_rxmix7_enum),
Ben Romberger1f045a72011-11-04 10:14:57 -07001123
1124 SOC_SINGLE_EXT("IIR1 Enable Band1", IIR1, BAND1, 1, 0,
1125 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1126 SOC_SINGLE_EXT("IIR1 Enable Band2", IIR1, BAND2, 1, 0,
1127 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1128 SOC_SINGLE_EXT("IIR1 Enable Band3", IIR1, BAND3, 1, 0,
1129 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1130 SOC_SINGLE_EXT("IIR1 Enable Band4", IIR1, BAND4, 1, 0,
1131 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1132 SOC_SINGLE_EXT("IIR1 Enable Band5", IIR1, BAND5, 1, 0,
1133 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1134 SOC_SINGLE_EXT("IIR2 Enable Band1", IIR2, BAND1, 1, 0,
1135 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1136 SOC_SINGLE_EXT("IIR2 Enable Band2", IIR2, BAND2, 1, 0,
1137 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1138 SOC_SINGLE_EXT("IIR2 Enable Band3", IIR2, BAND3, 1, 0,
1139 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1140 SOC_SINGLE_EXT("IIR2 Enable Band4", IIR2, BAND4, 1, 0,
1141 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1142 SOC_SINGLE_EXT("IIR2 Enable Band5", IIR2, BAND5, 1, 0,
1143 tabla_get_iir_enable_audio_mixer, tabla_put_iir_enable_audio_mixer),
1144
1145 SOC_SINGLE_MULTI_EXT("IIR1 Band1", IIR1, BAND1, 255, 0, 5,
1146 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
1147 SOC_SINGLE_MULTI_EXT("IIR1 Band2", IIR1, BAND2, 255, 0, 5,
1148 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
1149 SOC_SINGLE_MULTI_EXT("IIR1 Band3", IIR1, BAND3, 255, 0, 5,
1150 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
1151 SOC_SINGLE_MULTI_EXT("IIR1 Band4", IIR1, BAND4, 255, 0, 5,
1152 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
1153 SOC_SINGLE_MULTI_EXT("IIR1 Band5", IIR1, BAND5, 255, 0, 5,
1154 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
1155 SOC_SINGLE_MULTI_EXT("IIR2 Band1", IIR2, BAND1, 255, 0, 5,
1156 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
1157 SOC_SINGLE_MULTI_EXT("IIR2 Band2", IIR2, BAND2, 255, 0, 5,
1158 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
1159 SOC_SINGLE_MULTI_EXT("IIR2 Band3", IIR2, BAND3, 255, 0, 5,
1160 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
1161 SOC_SINGLE_MULTI_EXT("IIR2 Band4", IIR2, BAND4, 255, 0, 5,
1162 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
1163 SOC_SINGLE_MULTI_EXT("IIR2 Band5", IIR2, BAND5, 255, 0, 5,
1164 tabla_get_iir_band_audio_mixer, tabla_put_iir_band_audio_mixer),
Kuirong Wang0f8ade32012-02-27 16:29:45 -08001165 SOC_SINGLE_EXT("COMP1 Switch", SND_SOC_NOPM, 1, COMPANDER_1, 0,
1166 tabla_get_compander, tabla_set_compander),
1167 SOC_SINGLE_EXT("COMP2 Switch", SND_SOC_NOPM, 0, COMPANDER_2, 0,
1168 tabla_get_compander, tabla_set_compander),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001169};
1170
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08001171static const struct snd_kcontrol_new tabla_1_x_snd_controls[] = {
1172 SOC_SINGLE("MICBIAS4 CAPLESS Switch", TABLA_1_A_MICB_4_CTL, 4, 1, 1),
1173};
1174
1175static const struct snd_kcontrol_new tabla_2_higher_snd_controls[] = {
1176 SOC_SINGLE("MICBIAS4 CAPLESS Switch", TABLA_2_A_MICB_4_CTL, 4, 1, 1),
1177};
1178
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001179static const char *rx_mix1_text[] = {
1180 "ZERO", "SRC1", "SRC2", "IIR1", "IIR2", "RX1", "RX2", "RX3", "RX4",
1181 "RX5", "RX6", "RX7"
1182};
1183
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08001184static const char *rx_mix2_text[] = {
1185 "ZERO", "SRC1", "SRC2", "IIR1", "IIR2"
1186};
1187
Kiran Kandi8b3a8302011-09-27 16:13:28 -07001188static const char *rx_dsm_text[] = {
1189 "CIC_OUT", "DSM_INV"
1190};
1191
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001192static const char *sb_tx1_mux_text[] = {
1193 "ZERO", "RMIX1", "RMIX2", "RMIX3", "RMIX4", "RMIX5", "RMIX6", "RMIX7",
1194 "DEC1"
1195};
1196
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001197static const char *sb_tx2_mux_text[] = {
1198 "ZERO", "RMIX1", "RMIX2", "RMIX3", "RMIX4", "RMIX5", "RMIX6", "RMIX7",
1199 "DEC2"
1200};
1201
1202static const char *sb_tx3_mux_text[] = {
1203 "ZERO", "RMIX1", "RMIX2", "RMIX3", "RMIX4", "RMIX5", "RMIX6", "RMIX7",
1204 "DEC3"
1205};
1206
1207static const char *sb_tx4_mux_text[] = {
1208 "ZERO", "RMIX1", "RMIX2", "RMIX3", "RMIX4", "RMIX5", "RMIX6", "RMIX7",
1209 "DEC4"
1210};
1211
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001212static const char *sb_tx5_mux_text[] = {
1213 "ZERO", "RMIX1", "RMIX2", "RMIX3", "RMIX4", "RMIX5", "RMIX6", "RMIX7",
1214 "DEC5"
1215};
1216
1217static const char *sb_tx6_mux_text[] = {
1218 "ZERO", "RMIX1", "RMIX2", "RMIX3", "RMIX4", "RMIX5", "RMIX6", "RMIX7",
1219 "DEC6"
1220};
1221
1222static const char const *sb_tx7_to_tx10_mux_text[] = {
1223 "ZERO", "RMIX1", "RMIX2", "RMIX3", "RMIX4", "RMIX5", "RMIX6", "RMIX7",
1224 "DEC1", "DEC2", "DEC3", "DEC4", "DEC5", "DEC6", "DEC7", "DEC8",
1225 "DEC9", "DEC10"
1226};
1227
1228static const char *dec1_mux_text[] = {
1229 "ZERO", "DMIC1", "ADC6",
1230};
1231
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001232static const char *dec2_mux_text[] = {
1233 "ZERO", "DMIC2", "ADC5",
1234};
1235
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001236static const char *dec3_mux_text[] = {
1237 "ZERO", "DMIC3", "ADC4",
1238};
1239
1240static const char *dec4_mux_text[] = {
1241 "ZERO", "DMIC4", "ADC3",
1242};
1243
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001244static const char *dec5_mux_text[] = {
1245 "ZERO", "DMIC5", "ADC2",
1246};
1247
1248static const char *dec6_mux_text[] = {
1249 "ZERO", "DMIC6", "ADC1",
1250};
1251
1252static const char const *dec7_mux_text[] = {
1253 "ZERO", "DMIC1", "DMIC6", "ADC1", "ADC6", "ANC1_FB", "ANC2_FB",
1254};
1255
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001256static const char *dec8_mux_text[] = {
1257 "ZERO", "DMIC2", "DMIC5", "ADC2", "ADC5",
1258};
1259
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001260static const char *dec9_mux_text[] = {
1261 "ZERO", "DMIC4", "DMIC5", "ADC2", "ADC3", "ADCMB", "ANC1_FB", "ANC2_FB",
1262};
1263
1264static const char *dec10_mux_text[] = {
1265 "ZERO", "DMIC3", "DMIC6", "ADC1", "ADC4", "ADCMB", "ANC1_FB", "ANC2_FB",
1266};
1267
Bradley Rubin229c6a52011-07-12 16:18:48 -07001268static const char const *anc_mux_text[] = {
1269 "ZERO", "ADC1", "ADC2", "ADC3", "ADC4", "ADC5", "ADC6", "ADC_MB",
1270 "RSVD_1", "DMIC1", "DMIC2", "DMIC3", "DMIC4", "DMIC5", "DMIC6"
1271};
1272
1273static const char const *anc1_fb_mux_text[] = {
1274 "ZERO", "EAR_HPH_L", "EAR_LINE_1",
1275};
1276
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001277static const char *iir1_inp1_text[] = {
1278 "ZERO", "DEC1", "DEC2", "DEC3", "DEC4", "DEC5", "DEC6", "DEC7", "DEC8",
1279 "DEC9", "DEC10", "RX1", "RX2", "RX3", "RX4", "RX5", "RX6", "RX7"
1280};
1281
1282static const struct soc_enum rx_mix1_inp1_chain_enum =
1283 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX1_B1_CTL, 0, 12, rx_mix1_text);
1284
Bradley Rubin229c6a52011-07-12 16:18:48 -07001285static const struct soc_enum rx_mix1_inp2_chain_enum =
1286 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX1_B1_CTL, 4, 12, rx_mix1_text);
1287
Kiran Kandia9fffe92012-05-20 23:42:30 -07001288static const struct soc_enum rx_mix1_inp3_chain_enum =
1289 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX1_B2_CTL, 0, 12, rx_mix1_text);
1290
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001291static const struct soc_enum rx2_mix1_inp1_chain_enum =
1292 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX2_B1_CTL, 0, 12, rx_mix1_text);
1293
Bradley Rubin229c6a52011-07-12 16:18:48 -07001294static const struct soc_enum rx2_mix1_inp2_chain_enum =
1295 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX2_B1_CTL, 4, 12, rx_mix1_text);
1296
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001297static const struct soc_enum rx3_mix1_inp1_chain_enum =
1298 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX3_B1_CTL, 0, 12, rx_mix1_text);
1299
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001300static const struct soc_enum rx3_mix1_inp2_chain_enum =
1301 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX3_B1_CTL, 4, 12, rx_mix1_text);
1302
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001303static const struct soc_enum rx4_mix1_inp1_chain_enum =
1304 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX4_B1_CTL, 0, 12, rx_mix1_text);
1305
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001306static const struct soc_enum rx4_mix1_inp2_chain_enum =
1307 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX4_B1_CTL, 4, 12, rx_mix1_text);
1308
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001309static const struct soc_enum rx5_mix1_inp1_chain_enum =
1310 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX5_B1_CTL, 0, 12, rx_mix1_text);
1311
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001312static const struct soc_enum rx5_mix1_inp2_chain_enum =
1313 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX5_B1_CTL, 4, 12, rx_mix1_text);
1314
1315static const struct soc_enum rx6_mix1_inp1_chain_enum =
1316 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX6_B1_CTL, 0, 12, rx_mix1_text);
1317
1318static const struct soc_enum rx6_mix1_inp2_chain_enum =
1319 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX6_B1_CTL, 4, 12, rx_mix1_text);
1320
Bhalchandra Gajare0a8ad172011-08-12 13:32:22 -07001321static const struct soc_enum rx7_mix1_inp1_chain_enum =
1322 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX7_B1_CTL, 0, 12, rx_mix1_text);
1323
1324static const struct soc_enum rx7_mix1_inp2_chain_enum =
1325 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX7_B1_CTL, 4, 12, rx_mix1_text);
1326
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08001327static const struct soc_enum rx1_mix2_inp1_chain_enum =
1328 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX1_B3_CTL, 0, 5, rx_mix2_text);
1329
1330static const struct soc_enum rx1_mix2_inp2_chain_enum =
1331 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX1_B3_CTL, 3, 5, rx_mix2_text);
1332
1333static const struct soc_enum rx2_mix2_inp1_chain_enum =
1334 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX2_B3_CTL, 0, 5, rx_mix2_text);
1335
1336static const struct soc_enum rx2_mix2_inp2_chain_enum =
1337 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX2_B3_CTL, 3, 5, rx_mix2_text);
1338
1339static const struct soc_enum rx3_mix2_inp1_chain_enum =
1340 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX3_B3_CTL, 0, 5, rx_mix2_text);
1341
1342static const struct soc_enum rx3_mix2_inp2_chain_enum =
1343 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_RX3_B3_CTL, 3, 5, rx_mix2_text);
1344
Kiran Kandi8b3a8302011-09-27 16:13:28 -07001345static const struct soc_enum rx4_dsm_enum =
1346 SOC_ENUM_SINGLE(TABLA_A_CDC_RX4_B6_CTL, 4, 2, rx_dsm_text);
1347
1348static const struct soc_enum rx6_dsm_enum =
1349 SOC_ENUM_SINGLE(TABLA_A_CDC_RX6_B6_CTL, 4, 2, rx_dsm_text);
1350
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001351static const struct soc_enum sb_tx1_mux_enum =
1352 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B1_CTL, 0, 9, sb_tx1_mux_text);
1353
1354static const struct soc_enum sb_tx2_mux_enum =
1355 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B2_CTL, 0, 9, sb_tx2_mux_text);
1356
1357static const struct soc_enum sb_tx3_mux_enum =
1358 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B3_CTL, 0, 9, sb_tx3_mux_text);
1359
1360static const struct soc_enum sb_tx4_mux_enum =
1361 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B4_CTL, 0, 9, sb_tx4_mux_text);
1362
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001363static const struct soc_enum sb_tx5_mux_enum =
1364 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B5_CTL, 0, 9, sb_tx5_mux_text);
1365
1366static const struct soc_enum sb_tx6_mux_enum =
1367 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B6_CTL, 0, 9, sb_tx6_mux_text);
1368
1369static const struct soc_enum sb_tx7_mux_enum =
1370 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B7_CTL, 0, 18,
1371 sb_tx7_to_tx10_mux_text);
1372
1373static const struct soc_enum sb_tx8_mux_enum =
1374 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B8_CTL, 0, 18,
1375 sb_tx7_to_tx10_mux_text);
1376
Kiran Kandi3426e512011-09-13 22:50:10 -07001377static const struct soc_enum sb_tx9_mux_enum =
1378 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B9_CTL, 0, 18,
1379 sb_tx7_to_tx10_mux_text);
1380
1381static const struct soc_enum sb_tx10_mux_enum =
1382 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_SB_B10_CTL, 0, 18,
1383 sb_tx7_to_tx10_mux_text);
1384
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001385static const struct soc_enum dec1_mux_enum =
1386 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B1_CTL, 0, 3, dec1_mux_text);
1387
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001388static const struct soc_enum dec2_mux_enum =
1389 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B1_CTL, 2, 3, dec2_mux_text);
1390
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001391static const struct soc_enum dec3_mux_enum =
1392 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B1_CTL, 4, 3, dec3_mux_text);
1393
1394static const struct soc_enum dec4_mux_enum =
1395 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B1_CTL, 6, 3, dec4_mux_text);
1396
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001397static const struct soc_enum dec5_mux_enum =
1398 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B2_CTL, 0, 3, dec5_mux_text);
1399
1400static const struct soc_enum dec6_mux_enum =
1401 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B2_CTL, 2, 3, dec6_mux_text);
1402
1403static const struct soc_enum dec7_mux_enum =
1404 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B2_CTL, 4, 7, dec7_mux_text);
1405
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001406static const struct soc_enum dec8_mux_enum =
1407 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B3_CTL, 0, 7, dec8_mux_text);
1408
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001409static const struct soc_enum dec9_mux_enum =
1410 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B3_CTL, 3, 8, dec9_mux_text);
1411
1412static const struct soc_enum dec10_mux_enum =
1413 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_TX_B4_CTL, 0, 8, dec10_mux_text);
1414
Bradley Rubin229c6a52011-07-12 16:18:48 -07001415static const struct soc_enum anc1_mux_enum =
1416 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_ANC_B1_CTL, 0, 16, anc_mux_text);
1417
1418static const struct soc_enum anc2_mux_enum =
1419 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_ANC_B1_CTL, 4, 16, anc_mux_text);
1420
1421static const struct soc_enum anc1_fb_mux_enum =
1422 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_ANC_B2_CTL, 0, 3, anc1_fb_mux_text);
1423
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001424static const struct soc_enum iir1_inp1_mux_enum =
1425 SOC_ENUM_SINGLE(TABLA_A_CDC_CONN_EQ1_B1_CTL, 0, 18, iir1_inp1_text);
1426
1427static const struct snd_kcontrol_new rx_mix1_inp1_mux =
1428 SOC_DAPM_ENUM("RX1 MIX1 INP1 Mux", rx_mix1_inp1_chain_enum);
1429
Bradley Rubin229c6a52011-07-12 16:18:48 -07001430static const struct snd_kcontrol_new rx_mix1_inp2_mux =
1431 SOC_DAPM_ENUM("RX1 MIX1 INP2 Mux", rx_mix1_inp2_chain_enum);
1432
Kiran Kandia9fffe92012-05-20 23:42:30 -07001433static const struct snd_kcontrol_new rx_mix1_inp3_mux =
1434 SOC_DAPM_ENUM("RX1 MIX1 INP3 Mux", rx_mix1_inp3_chain_enum);
1435
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001436static const struct snd_kcontrol_new rx2_mix1_inp1_mux =
1437 SOC_DAPM_ENUM("RX2 MIX1 INP1 Mux", rx2_mix1_inp1_chain_enum);
1438
Bradley Rubin229c6a52011-07-12 16:18:48 -07001439static const struct snd_kcontrol_new rx2_mix1_inp2_mux =
1440 SOC_DAPM_ENUM("RX2 MIX1 INP2 Mux", rx2_mix1_inp2_chain_enum);
1441
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001442static const struct snd_kcontrol_new rx3_mix1_inp1_mux =
1443 SOC_DAPM_ENUM("RX3 MIX1 INP1 Mux", rx3_mix1_inp1_chain_enum);
1444
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001445static const struct snd_kcontrol_new rx3_mix1_inp2_mux =
1446 SOC_DAPM_ENUM("RX3 MIX1 INP2 Mux", rx3_mix1_inp2_chain_enum);
1447
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001448static const struct snd_kcontrol_new rx4_mix1_inp1_mux =
1449 SOC_DAPM_ENUM("RX4 MIX1 INP1 Mux", rx4_mix1_inp1_chain_enum);
1450
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001451static const struct snd_kcontrol_new rx4_mix1_inp2_mux =
1452 SOC_DAPM_ENUM("RX4 MIX1 INP2 Mux", rx4_mix1_inp2_chain_enum);
1453
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001454static const struct snd_kcontrol_new rx5_mix1_inp1_mux =
1455 SOC_DAPM_ENUM("RX5 MIX1 INP1 Mux", rx5_mix1_inp1_chain_enum);
1456
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001457static const struct snd_kcontrol_new rx5_mix1_inp2_mux =
1458 SOC_DAPM_ENUM("RX5 MIX1 INP2 Mux", rx5_mix1_inp2_chain_enum);
1459
1460static const struct snd_kcontrol_new rx6_mix1_inp1_mux =
1461 SOC_DAPM_ENUM("RX6 MIX1 INP1 Mux", rx6_mix1_inp1_chain_enum);
1462
1463static const struct snd_kcontrol_new rx6_mix1_inp2_mux =
1464 SOC_DAPM_ENUM("RX6 MIX1 INP2 Mux", rx6_mix1_inp2_chain_enum);
1465
Bhalchandra Gajare0a8ad172011-08-12 13:32:22 -07001466static const struct snd_kcontrol_new rx7_mix1_inp1_mux =
1467 SOC_DAPM_ENUM("RX7 MIX1 INP1 Mux", rx7_mix1_inp1_chain_enum);
1468
1469static const struct snd_kcontrol_new rx7_mix1_inp2_mux =
1470 SOC_DAPM_ENUM("RX7 MIX1 INP2 Mux", rx7_mix1_inp2_chain_enum);
1471
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08001472static const struct snd_kcontrol_new rx1_mix2_inp1_mux =
1473 SOC_DAPM_ENUM("RX1 MIX2 INP1 Mux", rx1_mix2_inp1_chain_enum);
1474
1475static const struct snd_kcontrol_new rx1_mix2_inp2_mux =
1476 SOC_DAPM_ENUM("RX1 MIX2 INP2 Mux", rx1_mix2_inp2_chain_enum);
1477
1478static const struct snd_kcontrol_new rx2_mix2_inp1_mux =
1479 SOC_DAPM_ENUM("RX2 MIX2 INP1 Mux", rx2_mix2_inp1_chain_enum);
1480
1481static const struct snd_kcontrol_new rx2_mix2_inp2_mux =
1482 SOC_DAPM_ENUM("RX2 MIX2 INP2 Mux", rx2_mix2_inp2_chain_enum);
1483
1484static const struct snd_kcontrol_new rx3_mix2_inp1_mux =
1485 SOC_DAPM_ENUM("RX3 MIX2 INP1 Mux", rx3_mix2_inp1_chain_enum);
1486
1487static const struct snd_kcontrol_new rx3_mix2_inp2_mux =
1488 SOC_DAPM_ENUM("RX3 MIX2 INP2 Mux", rx3_mix2_inp2_chain_enum);
1489
Kiran Kandi8b3a8302011-09-27 16:13:28 -07001490static const struct snd_kcontrol_new rx4_dsm_mux =
1491 SOC_DAPM_ENUM("RX4 DSM MUX Mux", rx4_dsm_enum);
1492
1493static const struct snd_kcontrol_new rx6_dsm_mux =
1494 SOC_DAPM_ENUM("RX6 DSM MUX Mux", rx6_dsm_enum);
1495
Kiran Kandi1e6371d2012-03-29 11:48:57 -07001496static const struct snd_kcontrol_new sb_tx1_mux =
1497 SOC_DAPM_ENUM("SLIM TX1 MUX Mux", sb_tx1_mux_enum);
1498
1499static const struct snd_kcontrol_new sb_tx2_mux =
1500 SOC_DAPM_ENUM("SLIM TX2 MUX Mux", sb_tx2_mux_enum);
1501
1502static const struct snd_kcontrol_new sb_tx3_mux =
1503 SOC_DAPM_ENUM("SLIM TX3 MUX Mux", sb_tx3_mux_enum);
1504
1505static const struct snd_kcontrol_new sb_tx4_mux =
1506 SOC_DAPM_ENUM("SLIM TX4 MUX Mux", sb_tx4_mux_enum);
1507
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001508static const struct snd_kcontrol_new sb_tx5_mux =
1509 SOC_DAPM_ENUM("SLIM TX5 MUX Mux", sb_tx5_mux_enum);
1510
1511static const struct snd_kcontrol_new sb_tx6_mux =
1512 SOC_DAPM_ENUM("SLIM TX6 MUX Mux", sb_tx6_mux_enum);
1513
1514static const struct snd_kcontrol_new sb_tx7_mux =
1515 SOC_DAPM_ENUM("SLIM TX7 MUX Mux", sb_tx7_mux_enum);
1516
1517static const struct snd_kcontrol_new sb_tx8_mux =
1518 SOC_DAPM_ENUM("SLIM TX8 MUX Mux", sb_tx8_mux_enum);
1519
Kiran Kandi3426e512011-09-13 22:50:10 -07001520static const struct snd_kcontrol_new sb_tx9_mux =
1521 SOC_DAPM_ENUM("SLIM TX9 MUX Mux", sb_tx9_mux_enum);
1522
1523static const struct snd_kcontrol_new sb_tx10_mux =
1524 SOC_DAPM_ENUM("SLIM TX10 MUX Mux", sb_tx10_mux_enum);
1525
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001526
Kiran Kandi59a96b12012-01-16 02:20:03 -08001527static int wcd9310_put_dec_enum(struct snd_kcontrol *kcontrol,
1528 struct snd_ctl_elem_value *ucontrol)
1529{
1530 struct snd_soc_dapm_widget_list *wlist = snd_kcontrol_chip(kcontrol);
1531 struct snd_soc_dapm_widget *w = wlist->widgets[0];
1532 struct snd_soc_codec *codec = w->codec;
1533 struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
1534 unsigned int dec_mux, decimator;
1535 char *dec_name = NULL;
1536 char *widget_name = NULL;
1537 char *temp;
1538 u16 tx_mux_ctl_reg;
1539 u8 adc_dmic_sel = 0x0;
1540 int ret = 0;
1541
1542 if (ucontrol->value.enumerated.item[0] > e->max - 1)
1543 return -EINVAL;
1544
1545 dec_mux = ucontrol->value.enumerated.item[0];
1546
1547 widget_name = kstrndup(w->name, 15, GFP_KERNEL);
1548 if (!widget_name)
1549 return -ENOMEM;
1550 temp = widget_name;
1551
1552 dec_name = strsep(&widget_name, " ");
1553 widget_name = temp;
1554 if (!dec_name) {
1555 pr_err("%s: Invalid decimator = %s\n", __func__, w->name);
1556 ret = -EINVAL;
1557 goto out;
1558 }
1559
1560 ret = kstrtouint(strpbrk(dec_name, "123456789"), 10, &decimator);
1561 if (ret < 0) {
1562 pr_err("%s: Invalid decimator = %s\n", __func__, dec_name);
1563 ret = -EINVAL;
1564 goto out;
1565 }
1566
1567 dev_dbg(w->dapm->dev, "%s(): widget = %s dec_name = %s decimator = %u"
1568 " dec_mux = %u\n", __func__, w->name, dec_name, decimator,
1569 dec_mux);
1570
1571
1572 switch (decimator) {
1573 case 1:
1574 case 2:
1575 case 3:
1576 case 4:
1577 case 5:
1578 case 6:
1579 if (dec_mux == 1)
1580 adc_dmic_sel = 0x1;
1581 else
1582 adc_dmic_sel = 0x0;
1583 break;
1584 case 7:
1585 case 8:
1586 case 9:
1587 case 10:
1588 if ((dec_mux == 1) || (dec_mux == 2))
1589 adc_dmic_sel = 0x1;
1590 else
1591 adc_dmic_sel = 0x0;
1592 break;
1593 default:
1594 pr_err("%s: Invalid Decimator = %u\n", __func__, decimator);
1595 ret = -EINVAL;
1596 goto out;
1597 }
1598
1599 tx_mux_ctl_reg = TABLA_A_CDC_TX1_MUX_CTL + 8 * (decimator - 1);
1600
1601 snd_soc_update_bits(codec, tx_mux_ctl_reg, 0x1, adc_dmic_sel);
1602
1603 ret = snd_soc_dapm_put_enum_double(kcontrol, ucontrol);
1604
1605out:
1606 kfree(widget_name);
1607 return ret;
1608}
1609
1610#define WCD9310_DEC_ENUM(xname, xenum) \
1611{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
1612 .info = snd_soc_info_enum_double, \
1613 .get = snd_soc_dapm_get_enum_double, \
1614 .put = wcd9310_put_dec_enum, \
1615 .private_value = (unsigned long)&xenum }
1616
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001617static const struct snd_kcontrol_new dec1_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001618 WCD9310_DEC_ENUM("DEC1 MUX Mux", dec1_mux_enum);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001619
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001620static const struct snd_kcontrol_new dec2_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001621 WCD9310_DEC_ENUM("DEC2 MUX Mux", dec2_mux_enum);
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001622
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001623static const struct snd_kcontrol_new dec3_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001624 WCD9310_DEC_ENUM("DEC3 MUX Mux", dec3_mux_enum);
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001625
1626static const struct snd_kcontrol_new dec4_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001627 WCD9310_DEC_ENUM("DEC4 MUX Mux", dec4_mux_enum);
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001628
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001629static const struct snd_kcontrol_new dec5_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001630 WCD9310_DEC_ENUM("DEC5 MUX Mux", dec5_mux_enum);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001631
1632static const struct snd_kcontrol_new dec6_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001633 WCD9310_DEC_ENUM("DEC6 MUX Mux", dec6_mux_enum);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001634
1635static const struct snd_kcontrol_new dec7_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001636 WCD9310_DEC_ENUM("DEC7 MUX Mux", dec7_mux_enum);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001637
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001638static const struct snd_kcontrol_new dec8_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001639 WCD9310_DEC_ENUM("DEC8 MUX Mux", dec8_mux_enum);
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07001640
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001641static const struct snd_kcontrol_new dec9_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001642 WCD9310_DEC_ENUM("DEC9 MUX Mux", dec9_mux_enum);
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001643
1644static const struct snd_kcontrol_new dec10_mux =
Kiran Kandi59a96b12012-01-16 02:20:03 -08001645 WCD9310_DEC_ENUM("DEC10 MUX Mux", dec10_mux_enum);
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07001646
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001647static const struct snd_kcontrol_new iir1_inp1_mux =
1648 SOC_DAPM_ENUM("IIR1 INP1 Mux", iir1_inp1_mux_enum);
1649
Kiran Kandi59a96b12012-01-16 02:20:03 -08001650static const struct snd_kcontrol_new anc1_mux =
1651 SOC_DAPM_ENUM("ANC1 MUX Mux", anc1_mux_enum);
1652
Bradley Rubin229c6a52011-07-12 16:18:48 -07001653static const struct snd_kcontrol_new anc2_mux =
1654 SOC_DAPM_ENUM("ANC2 MUX Mux", anc2_mux_enum);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001655
Bradley Rubin229c6a52011-07-12 16:18:48 -07001656static const struct snd_kcontrol_new anc1_fb_mux =
1657 SOC_DAPM_ENUM("ANC1 FB MUX Mux", anc1_fb_mux_enum);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001658
Bradley Rubin229c6a52011-07-12 16:18:48 -07001659static const struct snd_kcontrol_new dac1_switch[] = {
1660 SOC_DAPM_SINGLE("Switch", TABLA_A_RX_EAR_EN, 5, 1, 0)
1661};
1662static const struct snd_kcontrol_new hphl_switch[] = {
1663 SOC_DAPM_SINGLE("Switch", TABLA_A_RX_HPH_L_DAC_CTL, 6, 1, 0)
1664};
Kiran Kandi8b3a8302011-09-27 16:13:28 -07001665
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001666static const struct snd_kcontrol_new hphl_pa_mix[] = {
1667 SOC_DAPM_SINGLE("AUX_PGA_L Switch", TABLA_A_AUX_L_PA_CONN,
1668 7, 1, 0),
1669 SOC_DAPM_SINGLE("AUX_PGA_R Switch", TABLA_A_AUX_R_PA_CONN,
1670 7, 1, 0),
1671 SOC_DAPM_SINGLE("AUX_PGA_L_INV Switch",
1672 TABLA_A_AUX_L_PA_CONN_INV, 7, 1, 0),
1673 SOC_DAPM_SINGLE("AUX_PGA_R_INV Switch",
1674 TABLA_A_AUX_R_PA_CONN_INV, 7, 1, 0),
1675};
1676
1677static const struct snd_kcontrol_new hphr_pa_mix[] = {
1678 SOC_DAPM_SINGLE("AUX_PGA_L Switch", TABLA_A_AUX_L_PA_CONN,
1679 6, 1, 0),
1680 SOC_DAPM_SINGLE("AUX_PGA_R Switch", TABLA_A_AUX_R_PA_CONN,
1681 6, 1, 0),
1682 SOC_DAPM_SINGLE("AUX_PGA_L_INV Switch",
1683 TABLA_A_AUX_L_PA_CONN_INV, 6, 1, 0),
1684 SOC_DAPM_SINGLE("AUX_PGA_R_INV Switch",
1685 TABLA_A_AUX_R_PA_CONN_INV, 6, 1, 0),
1686};
1687
1688static const struct snd_kcontrol_new lineout1_pa_mix[] = {
1689 SOC_DAPM_SINGLE("AUX_PGA_L Switch", TABLA_A_AUX_L_PA_CONN,
1690 5, 1, 0),
1691 SOC_DAPM_SINGLE("AUX_PGA_R Switch", TABLA_A_AUX_R_PA_CONN,
1692 5, 1, 0),
1693 SOC_DAPM_SINGLE("AUX_PGA_L_INV Switch",
1694 TABLA_A_AUX_L_PA_CONN_INV, 5, 1, 0),
1695 SOC_DAPM_SINGLE("AUX_PGA_R_INV Switch",
1696 TABLA_A_AUX_R_PA_CONN_INV, 5, 1, 0),
1697};
1698
1699static const struct snd_kcontrol_new lineout2_pa_mix[] = {
1700 SOC_DAPM_SINGLE("AUX_PGA_L Switch", TABLA_A_AUX_L_PA_CONN,
1701 4, 1, 0),
1702 SOC_DAPM_SINGLE("AUX_PGA_R Switch", TABLA_A_AUX_R_PA_CONN,
1703 4, 1, 0),
1704 SOC_DAPM_SINGLE("AUX_PGA_L_INV Switch",
1705 TABLA_A_AUX_L_PA_CONN_INV, 4, 1, 0),
1706 SOC_DAPM_SINGLE("AUX_PGA_R_INV Switch",
1707 TABLA_A_AUX_R_PA_CONN_INV, 4, 1, 0),
1708};
1709
1710static const struct snd_kcontrol_new lineout3_pa_mix[] = {
1711 SOC_DAPM_SINGLE("AUX_PGA_L Switch", TABLA_A_AUX_L_PA_CONN,
1712 3, 1, 0),
1713 SOC_DAPM_SINGLE("AUX_PGA_R Switch", TABLA_A_AUX_R_PA_CONN,
1714 3, 1, 0),
1715 SOC_DAPM_SINGLE("AUX_PGA_L_INV Switch",
1716 TABLA_A_AUX_L_PA_CONN_INV, 3, 1, 0),
1717 SOC_DAPM_SINGLE("AUX_PGA_R_INV Switch",
1718 TABLA_A_AUX_R_PA_CONN_INV, 3, 1, 0),
1719};
1720
1721static const struct snd_kcontrol_new lineout4_pa_mix[] = {
1722 SOC_DAPM_SINGLE("AUX_PGA_L Switch", TABLA_A_AUX_L_PA_CONN,
1723 2, 1, 0),
1724 SOC_DAPM_SINGLE("AUX_PGA_R Switch", TABLA_A_AUX_R_PA_CONN,
1725 2, 1, 0),
1726 SOC_DAPM_SINGLE("AUX_PGA_L_INV Switch",
1727 TABLA_A_AUX_L_PA_CONN_INV, 2, 1, 0),
1728 SOC_DAPM_SINGLE("AUX_PGA_R_INV Switch",
1729 TABLA_A_AUX_R_PA_CONN_INV, 2, 1, 0),
1730};
1731
1732static const struct snd_kcontrol_new lineout5_pa_mix[] = {
1733 SOC_DAPM_SINGLE("AUX_PGA_L Switch", TABLA_A_AUX_L_PA_CONN,
1734 1, 1, 0),
1735 SOC_DAPM_SINGLE("AUX_PGA_R Switch", TABLA_A_AUX_R_PA_CONN,
1736 1, 1, 0),
1737 SOC_DAPM_SINGLE("AUX_PGA_L_INV Switch",
1738 TABLA_A_AUX_L_PA_CONN_INV, 1, 1, 0),
1739 SOC_DAPM_SINGLE("AUX_PGA_R_INV Switch",
1740 TABLA_A_AUX_R_PA_CONN_INV, 1, 1, 0),
1741};
1742
1743static const struct snd_kcontrol_new ear_pa_mix[] = {
1744 SOC_DAPM_SINGLE("AUX_PGA_L Switch", TABLA_A_AUX_L_PA_CONN,
1745 0, 1, 0),
1746 SOC_DAPM_SINGLE("AUX_PGA_R Switch", TABLA_A_AUX_R_PA_CONN,
1747 0, 1, 0),
1748 SOC_DAPM_SINGLE("AUX_PGA_L_INV Switch",
1749 TABLA_A_AUX_L_PA_CONN_INV, 0, 1, 0),
1750 SOC_DAPM_SINGLE("AUX_PGA_R_INV Switch",
1751 TABLA_A_AUX_R_PA_CONN_INV, 0, 1, 0),
1752};
1753
Kiran Kandi8b3a8302011-09-27 16:13:28 -07001754static const struct snd_kcontrol_new lineout3_ground_switch =
1755 SOC_DAPM_SINGLE("Switch", TABLA_A_RX_LINE_3_DAC_CTL, 6, 1, 0);
1756
1757static const struct snd_kcontrol_new lineout4_ground_switch =
1758 SOC_DAPM_SINGLE("Switch", TABLA_A_RX_LINE_4_DAC_CTL, 6, 1, 0);
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07001759
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001760static void tabla_codec_enable_adc_block(struct snd_soc_codec *codec,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001761 int enable)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001762{
1763 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
1764
1765 pr_debug("%s %d\n", __func__, enable);
1766
1767 if (enable) {
1768 tabla->adc_count++;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001769 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_OTHR_CTL, 0x2, 0x2);
1770 } else {
1771 tabla->adc_count--;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001772 if (!tabla->adc_count)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001773 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_OTHR_CTL,
Joonwoo Park03324832012-03-19 19:36:16 -07001774 0x2, 0x0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001775 }
1776}
1777
1778static int tabla_codec_enable_adc(struct snd_soc_dapm_widget *w,
1779 struct snd_kcontrol *kcontrol, int event)
1780{
1781 struct snd_soc_codec *codec = w->codec;
1782 u16 adc_reg;
Kiran Kandi9a2c62a82011-12-07 13:13:26 -08001783 u8 init_bit_shift;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001784
1785 pr_debug("%s %d\n", __func__, event);
1786
1787 if (w->reg == TABLA_A_TX_1_2_EN)
1788 adc_reg = TABLA_A_TX_1_2_TEST_CTL;
1789 else if (w->reg == TABLA_A_TX_3_4_EN)
1790 adc_reg = TABLA_A_TX_3_4_TEST_CTL;
1791 else if (w->reg == TABLA_A_TX_5_6_EN)
1792 adc_reg = TABLA_A_TX_5_6_TEST_CTL;
1793 else {
1794 pr_err("%s: Error, invalid adc register\n", __func__);
1795 return -EINVAL;
1796 }
1797
Kiran Kandi9a2c62a82011-12-07 13:13:26 -08001798 if (w->shift == 3)
1799 init_bit_shift = 6;
1800 else if (w->shift == 7)
1801 init_bit_shift = 7;
1802 else {
1803 pr_err("%s: Error, invalid init bit postion adc register\n",
1804 __func__);
1805 return -EINVAL;
1806 }
1807
1808
1809
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001810 switch (event) {
1811 case SND_SOC_DAPM_PRE_PMU:
1812 tabla_codec_enable_adc_block(codec, 1);
Kiran Kandi9a2c62a82011-12-07 13:13:26 -08001813 snd_soc_update_bits(codec, adc_reg, 1 << init_bit_shift,
1814 1 << init_bit_shift);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001815 break;
1816 case SND_SOC_DAPM_POST_PMU:
Kiran Kandi9a2c62a82011-12-07 13:13:26 -08001817
1818 snd_soc_update_bits(codec, adc_reg, 1 << init_bit_shift, 0x00);
1819
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001820 break;
1821 case SND_SOC_DAPM_POST_PMD:
1822 tabla_codec_enable_adc_block(codec, 0);
1823 break;
1824 }
1825 return 0;
1826}
1827
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001828static void tabla_codec_enable_audio_mode_bandgap(struct snd_soc_codec *codec)
1829{
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001830 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x80,
1831 0x80);
1832 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x04,
1833 0x04);
1834 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x01,
1835 0x01);
1836 usleep_range(1000, 1000);
1837 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x80,
1838 0x00);
1839}
1840
1841static void tabla_codec_enable_bandgap(struct snd_soc_codec *codec,
1842 enum tabla_bandgap_type choice)
1843{
1844 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
1845
1846 /* TODO lock resources accessed by audio streams and threaded
1847 * interrupt handlers
1848 */
1849
1850 pr_debug("%s, choice is %d, current is %d\n", __func__, choice,
1851 tabla->bandgap_type);
1852
1853 if (tabla->bandgap_type == choice)
1854 return;
1855
1856 if ((tabla->bandgap_type == TABLA_BANDGAP_OFF) &&
1857 (choice == TABLA_BANDGAP_AUDIO_MODE)) {
1858 tabla_codec_enable_audio_mode_bandgap(codec);
1859 } else if (choice == TABLA_BANDGAP_MBHC_MODE) {
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001860 /* bandgap mode becomes fast,
1861 * mclk should be off or clk buff source souldn't be VBG
1862 * Let's turn off mclk always */
1863 WARN_ON(snd_soc_read(codec, TABLA_A_CLK_BUFF_EN2) & (1 << 2));
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001864 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x2,
1865 0x2);
1866 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x80,
1867 0x80);
1868 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x4,
1869 0x4);
1870 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x01,
1871 0x01);
1872 usleep_range(1000, 1000);
1873 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x80,
1874 0x00);
1875 } else if ((tabla->bandgap_type == TABLA_BANDGAP_MBHC_MODE) &&
1876 (choice == TABLA_BANDGAP_AUDIO_MODE)) {
Damir Didjusto52900462012-08-16 21:22:29 -07001877 snd_soc_write(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x50);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001878 usleep_range(100, 100);
1879 tabla_codec_enable_audio_mode_bandgap(codec);
1880 } else if (choice == TABLA_BANDGAP_OFF) {
Damir Didjusto52900462012-08-16 21:22:29 -07001881 snd_soc_write(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x50);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001882 } else {
1883 pr_err("%s: Error, Invalid bandgap settings\n", __func__);
1884 }
1885 tabla->bandgap_type = choice;
1886}
1887
1888static void tabla_codec_disable_clock_block(struct snd_soc_codec *codec)
1889{
1890 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
1891 pr_debug("%s\n", __func__);
1892 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN2, 0x04, 0x00);
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001893 usleep_range(50, 50);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001894 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN2, 0x02, 0x02);
1895 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN1, 0x05, 0x00);
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001896 usleep_range(50, 50);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001897 tabla->clock_active = false;
1898}
1899
1900static int tabla_codec_mclk_index(const struct tabla_priv *tabla)
1901{
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001902 if (tabla->mbhc_cfg.mclk_rate == TABLA_MCLK_RATE_12288KHZ)
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001903 return 0;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07001904 else if (tabla->mbhc_cfg.mclk_rate == TABLA_MCLK_RATE_9600KHZ)
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001905 return 1;
1906 else {
1907 BUG_ON(1);
1908 return -EINVAL;
1909 }
1910}
1911
1912static void tabla_enable_rx_bias(struct snd_soc_codec *codec, u32 enable)
1913{
1914 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
1915
1916 if (enable) {
1917 tabla->rx_bias_count++;
1918 if (tabla->rx_bias_count == 1)
1919 snd_soc_update_bits(codec, TABLA_A_RX_COM_BIAS,
1920 0x80, 0x80);
1921 } else {
1922 tabla->rx_bias_count--;
1923 if (!tabla->rx_bias_count)
1924 snd_soc_update_bits(codec, TABLA_A_RX_COM_BIAS,
1925 0x80, 0x00);
1926 }
1927}
1928
1929static int tabla_codec_enable_config_mode(struct snd_soc_codec *codec,
1930 int enable)
1931{
1932 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
1933
1934 pr_debug("%s: enable = %d\n", __func__, enable);
1935 if (enable) {
1936 snd_soc_update_bits(codec, TABLA_A_CONFIG_MODE_FREQ, 0x10, 0);
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001937 /* bandgap mode to fast */
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001938 snd_soc_write(codec, TABLA_A_BIAS_CONFIG_MODE_BG_CTL, 0x17);
1939 usleep_range(5, 5);
1940 snd_soc_update_bits(codec, TABLA_A_CONFIG_MODE_FREQ, 0x80,
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001941 0x80);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001942 snd_soc_update_bits(codec, TABLA_A_CONFIG_MODE_TEST, 0x80,
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001943 0x80);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001944 usleep_range(10, 10);
1945 snd_soc_update_bits(codec, TABLA_A_CONFIG_MODE_TEST, 0x80, 0);
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001946 usleep_range(10000, 10000);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001947 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN1, 0x08, 0x08);
1948 } else {
1949 snd_soc_update_bits(codec, TABLA_A_BIAS_CONFIG_MODE_BG_CTL, 0x1,
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001950 0);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001951 snd_soc_update_bits(codec, TABLA_A_CONFIG_MODE_FREQ, 0x80, 0);
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001952 /* clk source to ext clk and clk buff ref to VBG */
1953 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN1, 0x0C, 0x04);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001954 }
1955 tabla->config_mode_active = enable ? true : false;
1956
1957 return 0;
1958}
1959
1960static int tabla_codec_enable_clock_block(struct snd_soc_codec *codec,
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001961 int config_mode)
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001962{
1963 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
1964
1965 pr_debug("%s: config_mode = %d\n", __func__, config_mode);
1966
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001967 /* transit to RCO requires mclk off */
1968 WARN_ON(snd_soc_read(codec, TABLA_A_CLK_BUFF_EN2) & (1 << 2));
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001969 if (config_mode) {
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001970 /* enable RCO and switch to it */
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001971 tabla_codec_enable_config_mode(codec, 1);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001972 snd_soc_write(codec, TABLA_A_CLK_BUFF_EN2, 0x02);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001973 usleep_range(1000, 1000);
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001974 } else {
1975 /* switch to MCLK */
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001976 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN1, 0x08, 0x00);
1977
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001978 if (tabla->mbhc_polling_active) {
1979 snd_soc_write(codec, TABLA_A_CLK_BUFF_EN2, 0x02);
1980 tabla_codec_enable_config_mode(codec, 0);
1981 }
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001982 }
1983
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001984 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN1, 0x01, 0x01);
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001985 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN2, 0x02, 0x00);
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07001986 /* on MCLK */
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08001987 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN2, 0x04, 0x04);
1988 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_MCLK_CTL, 0x01, 0x01);
1989 usleep_range(50, 50);
1990 tabla->clock_active = true;
1991 return 0;
1992}
1993
1994static int tabla_codec_enable_aux_pga(struct snd_soc_dapm_widget *w,
1995 struct snd_kcontrol *kcontrol, int event)
1996{
1997 struct snd_soc_codec *codec = w->codec;
1998 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
1999
2000 pr_debug("%s: %d\n", __func__, event);
2001
2002 switch (event) {
2003 case SND_SOC_DAPM_PRE_PMU:
2004 tabla_codec_enable_bandgap(codec,
2005 TABLA_BANDGAP_AUDIO_MODE);
2006 tabla_enable_rx_bias(codec, 1);
2007
2008 snd_soc_update_bits(codec, TABLA_A_AUX_COM_CTL,
2009 0x08, 0x08);
2010 /* Enable Zero Cross detect for AUX PGA channel
2011 * and set the initial AUX PGA gain to NEG_0P0_DB
2012 * to avoid glitches.
2013 */
2014 if (w->reg == TABLA_A_AUX_L_EN) {
2015 snd_soc_update_bits(codec, TABLA_A_AUX_L_EN,
2016 0x20, 0x20);
2017 tabla->aux_l_gain = snd_soc_read(codec,
2018 TABLA_A_AUX_L_GAIN);
2019 snd_soc_write(codec, TABLA_A_AUX_L_GAIN, 0x1F);
2020 } else {
2021 snd_soc_update_bits(codec, TABLA_A_AUX_R_EN,
2022 0x20, 0x20);
2023 tabla->aux_r_gain = snd_soc_read(codec,
2024 TABLA_A_AUX_R_GAIN);
2025 snd_soc_write(codec, TABLA_A_AUX_R_GAIN, 0x1F);
2026 }
2027 if (tabla->aux_pga_cnt++ == 1
2028 && !tabla->mclk_enabled) {
2029 tabla_codec_enable_clock_block(codec, 1);
2030 pr_debug("AUX PGA enabled RC osc\n");
2031 }
2032 break;
2033
2034 case SND_SOC_DAPM_POST_PMU:
2035 if (w->reg == TABLA_A_AUX_L_EN)
2036 snd_soc_write(codec, TABLA_A_AUX_L_GAIN,
2037 tabla->aux_l_gain);
2038 else
2039 snd_soc_write(codec, TABLA_A_AUX_R_GAIN,
2040 tabla->aux_r_gain);
2041 break;
2042
2043 case SND_SOC_DAPM_PRE_PMD:
2044 /* Mute AUX PGA channel in use before disabling AUX PGA */
2045 if (w->reg == TABLA_A_AUX_L_EN) {
2046 tabla->aux_l_gain = snd_soc_read(codec,
2047 TABLA_A_AUX_L_GAIN);
2048 snd_soc_write(codec, TABLA_A_AUX_L_GAIN, 0x1F);
2049 } else {
2050 tabla->aux_r_gain = snd_soc_read(codec,
2051 TABLA_A_AUX_R_GAIN);
2052 snd_soc_write(codec, TABLA_A_AUX_R_GAIN, 0x1F);
2053 }
2054 break;
2055
2056 case SND_SOC_DAPM_POST_PMD:
2057 tabla_enable_rx_bias(codec, 0);
2058
2059 snd_soc_update_bits(codec, TABLA_A_AUX_COM_CTL,
2060 0x08, 0x00);
2061 if (w->reg == TABLA_A_AUX_L_EN) {
2062 snd_soc_write(codec, TABLA_A_AUX_L_GAIN,
2063 tabla->aux_l_gain);
2064 snd_soc_update_bits(codec, TABLA_A_AUX_L_EN,
2065 0x20, 0x00);
2066 } else {
2067 snd_soc_write(codec, TABLA_A_AUX_R_GAIN,
2068 tabla->aux_r_gain);
2069 snd_soc_update_bits(codec, TABLA_A_AUX_R_EN,
2070 0x20, 0x00);
2071 }
2072
2073 if (tabla->aux_pga_cnt-- == 0) {
2074 if (tabla->mbhc_polling_active)
2075 tabla_codec_enable_bandgap(codec,
2076 TABLA_BANDGAP_MBHC_MODE);
2077 else
2078 tabla_codec_enable_bandgap(codec,
2079 TABLA_BANDGAP_OFF);
2080
2081 if (!tabla->mclk_enabled &&
2082 !tabla->mbhc_polling_active) {
2083 tabla_codec_enable_clock_block(codec, 0);
2084 }
2085 }
2086 break;
2087 }
2088 return 0;
2089}
2090
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002091static int tabla_codec_enable_lineout(struct snd_soc_dapm_widget *w,
2092 struct snd_kcontrol *kcontrol, int event)
2093{
2094 struct snd_soc_codec *codec = w->codec;
2095 u16 lineout_gain_reg;
2096
Kiran Kandidb0a4b02011-08-23 09:32:09 -07002097 pr_debug("%s %d %s\n", __func__, event, w->name);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002098
2099 switch (w->shift) {
2100 case 0:
2101 lineout_gain_reg = TABLA_A_RX_LINE_1_GAIN;
2102 break;
2103 case 1:
2104 lineout_gain_reg = TABLA_A_RX_LINE_2_GAIN;
2105 break;
2106 case 2:
2107 lineout_gain_reg = TABLA_A_RX_LINE_3_GAIN;
2108 break;
2109 case 3:
2110 lineout_gain_reg = TABLA_A_RX_LINE_4_GAIN;
2111 break;
2112 case 4:
2113 lineout_gain_reg = TABLA_A_RX_LINE_5_GAIN;
2114 break;
2115 default:
2116 pr_err("%s: Error, incorrect lineout register value\n",
2117 __func__);
2118 return -EINVAL;
2119 }
2120
2121 switch (event) {
2122 case SND_SOC_DAPM_PRE_PMU:
2123 snd_soc_update_bits(codec, lineout_gain_reg, 0x40, 0x40);
2124 break;
2125 case SND_SOC_DAPM_POST_PMU:
SathishKumar Mani74ef8912012-09-11 14:25:46 -07002126 pr_debug("%s: sleeping 16 us after %s PA turn on\n",
Kiran Kandidb0a4b02011-08-23 09:32:09 -07002127 __func__, w->name);
SathishKumar Mani74ef8912012-09-11 14:25:46 -07002128 usleep_range(16, 16);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002129 break;
2130 case SND_SOC_DAPM_POST_PMD:
2131 snd_soc_update_bits(codec, lineout_gain_reg, 0x40, 0x00);
2132 break;
2133 }
2134 return 0;
2135}
2136
Kiran Kandicf45f6a2011-07-17 21:10:19 -07002137
2138static int tabla_codec_enable_dmic(struct snd_soc_dapm_widget *w,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002139 struct snd_kcontrol *kcontrol, int event)
2140{
2141 struct snd_soc_codec *codec = w->codec;
Kiran Kandi0ba468f2012-05-08 11:45:05 -07002142 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
2143 u8 dmic_clk_en;
2144 s32 *dmic_clk_cnt;
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07002145 unsigned int dmic;
2146 int ret;
Kiran Kandicf45f6a2011-07-17 21:10:19 -07002147
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07002148 ret = kstrtouint(strpbrk(w->name, "123456"), 10, &dmic);
2149 if (ret < 0) {
2150 pr_err("%s: Invalid DMIC line on the codec\n", __func__);
Kiran Kandicf45f6a2011-07-17 21:10:19 -07002151 return -EINVAL;
2152 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002153
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07002154 switch (dmic) {
2155 case 1:
2156 case 2:
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07002157 dmic_clk_en = 0x01;
Kiran Kandi0ba468f2012-05-08 11:45:05 -07002158 dmic_clk_cnt = &(tabla->dmic_1_2_clk_cnt);
2159
2160 pr_debug("%s() event %d DMIC%d dmic_1_2_clk_cnt %d\n",
2161 __func__, event, dmic, *dmic_clk_cnt);
2162
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07002163 break;
2164
2165 case 3:
2166 case 4:
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07002167 dmic_clk_en = 0x04;
Kiran Kandi0ba468f2012-05-08 11:45:05 -07002168 dmic_clk_cnt = &(tabla->dmic_3_4_clk_cnt);
2169
2170 pr_debug("%s() event %d DMIC%d dmic_3_4_clk_cnt %d\n",
2171 __func__, event, dmic, *dmic_clk_cnt);
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07002172 break;
2173
2174 case 5:
2175 case 6:
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07002176 dmic_clk_en = 0x10;
Kiran Kandi0ba468f2012-05-08 11:45:05 -07002177 dmic_clk_cnt = &(tabla->dmic_5_6_clk_cnt);
2178
2179 pr_debug("%s() event %d DMIC%d dmic_5_6_clk_cnt %d\n",
2180 __func__, event, dmic, *dmic_clk_cnt);
2181
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07002182 break;
2183
2184 default:
2185 pr_err("%s: Invalid DMIC Selection\n", __func__);
2186 return -EINVAL;
2187 }
2188
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002189 switch (event) {
2190 case SND_SOC_DAPM_PRE_PMU:
Kiran Kandicf45f6a2011-07-17 21:10:19 -07002191
Kiran Kandi0ba468f2012-05-08 11:45:05 -07002192 (*dmic_clk_cnt)++;
2193 if (*dmic_clk_cnt == 1)
2194 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_DMIC_CTL,
2195 dmic_clk_en, dmic_clk_en);
Kiran Kandicf45f6a2011-07-17 21:10:19 -07002196
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002197 break;
2198 case SND_SOC_DAPM_POST_PMD:
Kiran Kandi0ba468f2012-05-08 11:45:05 -07002199
2200 (*dmic_clk_cnt)--;
2201 if (*dmic_clk_cnt == 0)
2202 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_DMIC_CTL,
2203 dmic_clk_en, 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002204 break;
2205 }
2206 return 0;
2207}
2208
Bradley Rubin229c6a52011-07-12 16:18:48 -07002209static int tabla_codec_enable_anc(struct snd_soc_dapm_widget *w,
2210 struct snd_kcontrol *kcontrol, int event)
2211{
2212 struct snd_soc_codec *codec = w->codec;
2213 const char *filename;
2214 const struct firmware *fw;
2215 int i;
2216 int ret;
Bradley Rubina7096d02011-08-03 18:29:02 -07002217 int num_anc_slots;
2218 struct anc_header *anc_head;
Bradley Rubin229c6a52011-07-12 16:18:48 -07002219 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Bradley Rubina7096d02011-08-03 18:29:02 -07002220 u32 anc_writes_size = 0;
2221 int anc_size_remaining;
2222 u32 *anc_ptr;
Bradley Rubin229c6a52011-07-12 16:18:48 -07002223 u16 reg;
2224 u8 mask, val, old_val;
2225
2226 pr_debug("%s %d\n", __func__, event);
2227 switch (event) {
2228 case SND_SOC_DAPM_PRE_PMU:
2229
Bradley Rubin4283a4c2011-07-29 16:18:54 -07002230 filename = "wcd9310/wcd9310_anc.bin";
Bradley Rubin229c6a52011-07-12 16:18:48 -07002231
2232 ret = request_firmware(&fw, filename, codec->dev);
2233 if (ret != 0) {
2234 dev_err(codec->dev, "Failed to acquire ANC data: %d\n",
2235 ret);
2236 return -ENODEV;
2237 }
2238
Bradley Rubina7096d02011-08-03 18:29:02 -07002239 if (fw->size < sizeof(struct anc_header)) {
Bradley Rubin229c6a52011-07-12 16:18:48 -07002240 dev_err(codec->dev, "Not enough data\n");
2241 release_firmware(fw);
2242 return -ENOMEM;
2243 }
2244
2245 /* First number is the number of register writes */
Bradley Rubina7096d02011-08-03 18:29:02 -07002246 anc_head = (struct anc_header *)(fw->data);
2247 anc_ptr = (u32 *)((u32)fw->data + sizeof(struct anc_header));
2248 anc_size_remaining = fw->size - sizeof(struct anc_header);
2249 num_anc_slots = anc_head->num_anc_slots;
Bradley Rubin229c6a52011-07-12 16:18:48 -07002250
Bradley Rubina7096d02011-08-03 18:29:02 -07002251 if (tabla->anc_slot >= num_anc_slots) {
2252 dev_err(codec->dev, "Invalid ANC slot selected\n");
2253 release_firmware(fw);
2254 return -EINVAL;
2255 }
2256
2257 for (i = 0; i < num_anc_slots; i++) {
2258
2259 if (anc_size_remaining < TABLA_PACKED_REG_SIZE) {
2260 dev_err(codec->dev, "Invalid register format\n");
2261 release_firmware(fw);
2262 return -EINVAL;
2263 }
2264 anc_writes_size = (u32)(*anc_ptr);
2265 anc_size_remaining -= sizeof(u32);
2266 anc_ptr += 1;
2267
2268 if (anc_writes_size * TABLA_PACKED_REG_SIZE
2269 > anc_size_remaining) {
2270 dev_err(codec->dev, "Invalid register format\n");
2271 release_firmware(fw);
2272 return -ENOMEM;
2273 }
2274
2275 if (tabla->anc_slot == i)
2276 break;
2277
2278 anc_size_remaining -= (anc_writes_size *
2279 TABLA_PACKED_REG_SIZE);
Bradley Rubin939ff3f2011-08-26 17:19:34 -07002280 anc_ptr += anc_writes_size;
Bradley Rubina7096d02011-08-03 18:29:02 -07002281 }
2282 if (i == num_anc_slots) {
2283 dev_err(codec->dev, "Selected ANC slot not present\n");
Bradley Rubin229c6a52011-07-12 16:18:48 -07002284 release_firmware(fw);
2285 return -ENOMEM;
2286 }
2287
Bradley Rubina7096d02011-08-03 18:29:02 -07002288 for (i = 0; i < anc_writes_size; i++) {
2289 TABLA_CODEC_UNPACK_ENTRY(anc_ptr[i], reg,
Bradley Rubin229c6a52011-07-12 16:18:48 -07002290 mask, val);
2291 old_val = snd_soc_read(codec, reg);
Bradley Rubin4283a4c2011-07-29 16:18:54 -07002292 snd_soc_write(codec, reg, (old_val & ~mask) |
2293 (val & mask));
Bradley Rubin229c6a52011-07-12 16:18:48 -07002294 }
2295 release_firmware(fw);
Bradley Rubin229c6a52011-07-12 16:18:48 -07002296
Joonwoo Park743d0e32012-05-17 15:11:58 -07002297 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
2298 /* if MBHC polling is active, set TX7_MBHC_EN bit 7 */
2299 if (tabla->mbhc_polling_active)
2300 snd_soc_update_bits(codec, TABLA_A_TX_7_MBHC_EN, 0x80,
2301 0x80);
2302 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
Bradley Rubin229c6a52011-07-12 16:18:48 -07002303 break;
2304 case SND_SOC_DAPM_POST_PMD:
Joonwoo Park743d0e32012-05-17 15:11:58 -07002305 /* unset TX7_MBHC_EN bit 7 */
2306 snd_soc_update_bits(codec, TABLA_A_TX_7_MBHC_EN, 0x80, 0x00);
2307
Bradley Rubin229c6a52011-07-12 16:18:48 -07002308 snd_soc_write(codec, TABLA_A_CDC_CLK_ANC_RESET_CTL, 0xFF);
2309 snd_soc_write(codec, TABLA_A_CDC_CLK_ANC_CLK_EN_CTL, 0);
2310 break;
2311 }
2312 return 0;
2313}
2314
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002315/* called under codec_resource_lock acquisition */
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002316static void tabla_codec_start_hs_polling(struct snd_soc_codec *codec)
2317{
Bradley Rubincb3950a2011-08-18 13:07:26 -07002318 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Park03324832012-03-19 19:36:16 -07002319 struct wcd9xxx *tabla_core = dev_get_drvdata(codec->dev->parent);
2320 int mbhc_state = tabla->mbhc_state;
Bradley Rubincb3950a2011-08-18 13:07:26 -07002321
Joonwoo Park03324832012-03-19 19:36:16 -07002322 pr_debug("%s: enter\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002323 if (!tabla->mbhc_polling_active) {
2324 pr_debug("Polling is not active, do not start polling\n");
2325 return;
Bradley Rubincb3950a2011-08-18 13:07:26 -07002326 }
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002327 snd_soc_write(codec, TABLA_A_MBHC_SCALING_MUX_1, 0x84);
Joonwoo Park03324832012-03-19 19:36:16 -07002328
Joonwoo Park5bbcb0c2012-08-07 17:25:52 -07002329 if (tabla->no_mic_headset_override) {
2330 pr_debug("%s setting button threshold to min", __func__);
2331 /* set to min */
2332 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B4_CTL, 0x80);
2333 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B3_CTL, 0x00);
2334 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B6_CTL, 0x80);
2335 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B5_CTL, 0x00);
2336 } else if (unlikely(mbhc_state == MBHC_STATE_POTENTIAL)) {
2337 pr_debug("%s recovering MBHC state machine\n", __func__);
2338 tabla->mbhc_state = MBHC_STATE_POTENTIAL_RECOVERY;
2339 /* set to max button press threshold */
2340 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B2_CTL, 0x7F);
2341 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B1_CTL, 0xFF);
2342 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B4_CTL,
2343 (TABLA_IS_1_X(tabla_core->version) ?
2344 0x07 : 0x7F));
2345 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B3_CTL, 0xFF);
2346 /* set to max */
2347 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B6_CTL, 0x7F);
2348 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B5_CTL, 0xFF);
Joonwoo Park03324832012-03-19 19:36:16 -07002349 }
2350
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002351 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x1);
2352 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x0);
2353 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x1);
Joonwoo Park03324832012-03-19 19:36:16 -07002354 pr_debug("%s: leave\n", __func__);
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002355}
2356
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002357/* called under codec_resource_lock acquisition */
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002358static void tabla_codec_pause_hs_polling(struct snd_soc_codec *codec)
2359{
Bradley Rubincb3950a2011-08-18 13:07:26 -07002360 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
2361
Joonwoo Park03324832012-03-19 19:36:16 -07002362 pr_debug("%s: enter\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002363 if (!tabla->mbhc_polling_active) {
2364 pr_debug("polling not active, nothing to pause\n");
2365 return;
Bradley Rubincb3950a2011-08-18 13:07:26 -07002366 }
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002367
2368 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x8);
Joonwoo Park03324832012-03-19 19:36:16 -07002369 pr_debug("%s: leave\n", __func__);
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002370}
2371
Joonwoo Park03324832012-03-19 19:36:16 -07002372static void tabla_codec_switch_cfilt_mode(struct snd_soc_codec *codec, int mode)
Bhalchandra Gajare19d9c132011-11-18 14:57:08 -08002373{
2374 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
2375 u8 reg_mode_val, cur_mode_val;
2376 bool mbhc_was_polling = false;
2377
2378 if (mode)
2379 reg_mode_val = TABLA_CFILT_FAST_MODE;
2380 else
2381 reg_mode_val = TABLA_CFILT_SLOW_MODE;
2382
2383 cur_mode_val = snd_soc_read(codec,
2384 tabla->mbhc_bias_regs.cfilt_ctl) & 0x40;
2385
2386 if (cur_mode_val != reg_mode_val) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002387 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Bhalchandra Gajare19d9c132011-11-18 14:57:08 -08002388 if (tabla->mbhc_polling_active) {
2389 tabla_codec_pause_hs_polling(codec);
2390 mbhc_was_polling = true;
2391 }
2392 snd_soc_update_bits(codec,
2393 tabla->mbhc_bias_regs.cfilt_ctl, 0x40, reg_mode_val);
2394 if (mbhc_was_polling)
2395 tabla_codec_start_hs_polling(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002396 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
Bhalchandra Gajare19d9c132011-11-18 14:57:08 -08002397 pr_debug("%s: CFILT mode change (%x to %x)\n", __func__,
2398 cur_mode_val, reg_mode_val);
2399 } else {
2400 pr_debug("%s: CFILT Value is already %x\n",
2401 __func__, cur_mode_val);
2402 }
2403}
2404
2405static void tabla_codec_update_cfilt_usage(struct snd_soc_codec *codec,
2406 u8 cfilt_sel, int inc)
2407{
2408 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
2409 u32 *cfilt_cnt_ptr = NULL;
2410 u16 micb_cfilt_reg;
2411
2412 switch (cfilt_sel) {
2413 case TABLA_CFILT1_SEL:
2414 cfilt_cnt_ptr = &tabla->cfilt1_cnt;
2415 micb_cfilt_reg = TABLA_A_MICB_CFILT_1_CTL;
2416 break;
2417 case TABLA_CFILT2_SEL:
2418 cfilt_cnt_ptr = &tabla->cfilt2_cnt;
2419 micb_cfilt_reg = TABLA_A_MICB_CFILT_2_CTL;
2420 break;
2421 case TABLA_CFILT3_SEL:
2422 cfilt_cnt_ptr = &tabla->cfilt3_cnt;
2423 micb_cfilt_reg = TABLA_A_MICB_CFILT_3_CTL;
2424 break;
2425 default:
2426 return; /* should not happen */
2427 }
2428
2429 if (inc) {
2430 if (!(*cfilt_cnt_ptr)++) {
2431 /* Switch CFILT to slow mode if MBHC CFILT being used */
2432 if (cfilt_sel == tabla->mbhc_bias_regs.cfilt_sel)
2433 tabla_codec_switch_cfilt_mode(codec, 0);
2434
2435 snd_soc_update_bits(codec, micb_cfilt_reg, 0x80, 0x80);
2436 }
2437 } else {
2438 /* check if count not zero, decrement
2439 * then check if zero, go ahead disable cfilter
2440 */
2441 if ((*cfilt_cnt_ptr) && !--(*cfilt_cnt_ptr)) {
2442 snd_soc_update_bits(codec, micb_cfilt_reg, 0x80, 0);
2443
2444 /* Switch CFILT to fast mode if MBHC CFILT being used */
2445 if (cfilt_sel == tabla->mbhc_bias_regs.cfilt_sel)
2446 tabla_codec_switch_cfilt_mode(codec, 1);
2447 }
2448 }
2449}
2450
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002451static int tabla_find_k_value(unsigned int ldoh_v, unsigned int cfilt_mv)
2452{
2453 int rc = -EINVAL;
2454 unsigned min_mv, max_mv;
2455
2456 switch (ldoh_v) {
2457 case TABLA_LDOH_1P95_V:
2458 min_mv = 160;
2459 max_mv = 1800;
2460 break;
2461 case TABLA_LDOH_2P35_V:
2462 min_mv = 200;
2463 max_mv = 2200;
2464 break;
2465 case TABLA_LDOH_2P75_V:
2466 min_mv = 240;
2467 max_mv = 2600;
2468 break;
2469 case TABLA_LDOH_2P85_V:
2470 min_mv = 250;
2471 max_mv = 2700;
2472 break;
2473 default:
2474 goto done;
2475 }
2476
2477 if (cfilt_mv < min_mv || cfilt_mv > max_mv)
2478 goto done;
2479
2480 for (rc = 4; rc <= 44; rc++) {
2481 min_mv = max_mv * (rc) / 44;
2482 if (min_mv >= cfilt_mv) {
2483 rc -= 4;
2484 break;
2485 }
2486 }
2487done:
2488 return rc;
2489}
2490
2491static bool tabla_is_hph_pa_on(struct snd_soc_codec *codec)
2492{
2493 u8 hph_reg_val = 0;
2494 hph_reg_val = snd_soc_read(codec, TABLA_A_RX_HPH_CNP_EN);
2495
2496 return (hph_reg_val & 0x30) ? true : false;
2497}
2498
Joonwoo Parka9444452011-12-08 18:48:27 -08002499static bool tabla_is_hph_dac_on(struct snd_soc_codec *codec, int left)
2500{
2501 u8 hph_reg_val = 0;
2502 if (left)
2503 hph_reg_val = snd_soc_read(codec,
2504 TABLA_A_RX_HPH_L_DAC_CTL);
2505 else
2506 hph_reg_val = snd_soc_read(codec,
2507 TABLA_A_RX_HPH_R_DAC_CTL);
2508
2509 return (hph_reg_val & 0xC0) ? true : false;
2510}
2511
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002512static void tabla_turn_onoff_override(struct snd_soc_codec *codec, bool on)
2513{
2514 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x04, on << 2);
2515}
2516
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002517/* called under codec_resource_lock acquisition */
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002518static void tabla_codec_drive_v_to_micbias(struct snd_soc_codec *codec,
2519 int usec)
2520{
2521 int cfilt_k_val;
2522 bool set = true;
2523 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
2524
2525 if (tabla->mbhc_data.micb_mv != VDDIO_MICBIAS_MV &&
2526 tabla->mbhc_micbias_switched) {
2527 pr_debug("%s: set mic V to micbias V\n", __func__);
2528 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x2, 0x2);
2529 tabla_turn_onoff_override(codec, true);
2530 while (1) {
2531 cfilt_k_val = tabla_find_k_value(
2532 tabla->pdata->micbias.ldoh_v,
2533 set ? tabla->mbhc_data.micb_mv :
2534 VDDIO_MICBIAS_MV);
2535 snd_soc_update_bits(codec,
2536 tabla->mbhc_bias_regs.cfilt_val,
2537 0xFC, (cfilt_k_val << 2));
2538 if (!set)
2539 break;
2540 usleep_range(usec, usec);
2541 set = false;
2542 }
2543 tabla_turn_onoff_override(codec, false);
2544 }
2545}
2546
2547/* called under codec_resource_lock acquisition */
2548static void __tabla_codec_switch_micbias(struct snd_soc_codec *codec,
2549 int vddio_switch, bool restartpolling,
2550 bool checkpolling)
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002551{
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002552 int cfilt_k_val;
Joonwoo Park41956722012-04-18 13:13:07 -07002553 bool override;
2554 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002555
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002556 if (vddio_switch && !tabla->mbhc_micbias_switched &&
2557 (!checkpolling || tabla->mbhc_polling_active)) {
2558 if (restartpolling)
Bhalchandra Gajarec1e19c42011-11-18 11:22:56 -08002559 tabla_codec_pause_hs_polling(codec);
Joonwoo Park41956722012-04-18 13:13:07 -07002560 override = snd_soc_read(codec, TABLA_A_CDC_MBHC_B1_CTL) & 0x04;
2561 if (!override)
2562 tabla_turn_onoff_override(codec, true);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002563 /* Adjust threshold if Mic Bias voltage changes */
2564 if (tabla->mbhc_data.micb_mv != VDDIO_MICBIAS_MV) {
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002565 cfilt_k_val = tabla_find_k_value(
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002566 tabla->pdata->micbias.ldoh_v,
2567 VDDIO_MICBIAS_MV);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002568 snd_soc_update_bits(codec,
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002569 tabla->mbhc_bias_regs.cfilt_val,
2570 0xFC, (cfilt_k_val << 2));
Joonwoo Parkc1c67a92012-08-07 16:05:36 -07002571 usleep_range(cfilt_adjust_ms * 1000,
2572 cfilt_adjust_ms * 1000);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002573 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B1_CTL,
2574 tabla->mbhc_data.adj_v_ins_hu & 0xFF);
2575 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B2_CTL,
2576 (tabla->mbhc_data.adj_v_ins_hu >> 8) &
2577 0xFF);
2578 pr_debug("%s: Programmed MBHC thresholds to VDDIO\n",
2579 __func__);
2580 }
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002581
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002582 /* enable MIC BIAS Switch to VDDIO */
2583 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg,
2584 0x80, 0x80);
2585 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg,
2586 0x10, 0x00);
Joonwoo Park41956722012-04-18 13:13:07 -07002587 if (!override)
2588 tabla_turn_onoff_override(codec, false);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002589 if (restartpolling)
Bhalchandra Gajarec1e19c42011-11-18 11:22:56 -08002590 tabla_codec_start_hs_polling(codec);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002591
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002592 tabla->mbhc_micbias_switched = true;
2593 pr_debug("%s: VDDIO switch enabled\n", __func__);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002594 } else if (!vddio_switch && tabla->mbhc_micbias_switched) {
2595 if ((!checkpolling || tabla->mbhc_polling_active) &&
2596 restartpolling)
2597 tabla_codec_pause_hs_polling(codec);
2598 /* Reprogram thresholds */
2599 if (tabla->mbhc_data.micb_mv != VDDIO_MICBIAS_MV) {
2600 cfilt_k_val = tabla_find_k_value(
2601 tabla->pdata->micbias.ldoh_v,
2602 tabla->mbhc_data.micb_mv);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002603 snd_soc_update_bits(codec,
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002604 tabla->mbhc_bias_regs.cfilt_val,
2605 0xFC, (cfilt_k_val << 2));
Joonwoo Parkc1c67a92012-08-07 16:05:36 -07002606 usleep_range(cfilt_adjust_ms * 1000,
2607 cfilt_adjust_ms * 1000);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002608 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B1_CTL,
2609 tabla->mbhc_data.v_ins_hu & 0xFF);
2610 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B2_CTL,
2611 (tabla->mbhc_data.v_ins_hu >> 8) & 0xFF);
2612 pr_debug("%s: Programmed MBHC thresholds to MICBIAS\n",
2613 __func__);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002614 }
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002615
2616 /* Disable MIC BIAS Switch to VDDIO */
2617 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg,
2618 0x80, 0x00);
2619 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg,
2620 0x10, 0x00);
2621
2622 if ((!checkpolling || tabla->mbhc_polling_active) &&
2623 restartpolling)
2624 tabla_codec_start_hs_polling(codec);
2625
2626 tabla->mbhc_micbias_switched = false;
2627 pr_debug("%s: VDDIO switch disabled\n", __func__);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002628 }
2629}
2630
Joonwoo Parkcf473b42012-03-29 19:48:16 -07002631static void tabla_codec_switch_micbias(struct snd_soc_codec *codec,
2632 int vddio_switch)
2633{
2634 return __tabla_codec_switch_micbias(codec, vddio_switch, true, true);
2635}
2636
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002637static int tabla_codec_enable_micbias(struct snd_soc_dapm_widget *w,
2638 struct snd_kcontrol *kcontrol, int event)
2639{
2640 struct snd_soc_codec *codec = w->codec;
Patrick Lai3043fba2011-08-01 14:15:57 -07002641 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
2642 u16 micb_int_reg;
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002643 int micb_line;
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07002644 u8 cfilt_sel_val = 0;
Bradley Rubin229c6a52011-07-12 16:18:48 -07002645 char *internal1_text = "Internal1";
2646 char *internal2_text = "Internal2";
2647 char *internal3_text = "Internal3";
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002648
2649 pr_debug("%s %d\n", __func__, event);
2650 switch (w->reg) {
2651 case TABLA_A_MICB_1_CTL:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002652 micb_int_reg = TABLA_A_MICB_1_INT_RBIAS;
Patrick Lai3043fba2011-08-01 14:15:57 -07002653 cfilt_sel_val = tabla->pdata->micbias.bias1_cfilt_sel;
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002654 micb_line = TABLA_MICBIAS1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002655 break;
2656 case TABLA_A_MICB_2_CTL:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002657 micb_int_reg = TABLA_A_MICB_2_INT_RBIAS;
Patrick Lai3043fba2011-08-01 14:15:57 -07002658 cfilt_sel_val = tabla->pdata->micbias.bias2_cfilt_sel;
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002659 micb_line = TABLA_MICBIAS2;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002660 break;
2661 case TABLA_A_MICB_3_CTL:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002662 micb_int_reg = TABLA_A_MICB_3_INT_RBIAS;
Patrick Lai3043fba2011-08-01 14:15:57 -07002663 cfilt_sel_val = tabla->pdata->micbias.bias3_cfilt_sel;
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002664 micb_line = TABLA_MICBIAS3;
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07002665 break;
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08002666 case TABLA_1_A_MICB_4_CTL:
2667 case TABLA_2_A_MICB_4_CTL:
2668 micb_int_reg = tabla->reg_addr.micb_4_int_rbias;
Patrick Lai3043fba2011-08-01 14:15:57 -07002669 cfilt_sel_val = tabla->pdata->micbias.bias4_cfilt_sel;
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002670 micb_line = TABLA_MICBIAS4;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002671 break;
2672 default:
2673 pr_err("%s: Error, invalid micbias register\n", __func__);
2674 return -EINVAL;
2675 }
2676
2677 switch (event) {
2678 case SND_SOC_DAPM_PRE_PMU:
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002679 /* Decide whether to switch the micbias for MBHC */
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002680 if (w->reg == tabla->mbhc_bias_regs.ctl_reg) {
2681 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002682 tabla_codec_switch_micbias(codec, 0);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002683 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
2684 }
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002685
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07002686 snd_soc_update_bits(codec, w->reg, 0x0E, 0x0A);
Patrick Lai3043fba2011-08-01 14:15:57 -07002687 tabla_codec_update_cfilt_usage(codec, cfilt_sel_val, 1);
Bradley Rubin229c6a52011-07-12 16:18:48 -07002688
2689 if (strnstr(w->name, internal1_text, 30))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002690 snd_soc_update_bits(codec, micb_int_reg, 0xE0, 0xE0);
Bradley Rubin229c6a52011-07-12 16:18:48 -07002691 else if (strnstr(w->name, internal2_text, 30))
2692 snd_soc_update_bits(codec, micb_int_reg, 0x1C, 0x1C);
2693 else if (strnstr(w->name, internal3_text, 30))
2694 snd_soc_update_bits(codec, micb_int_reg, 0x3, 0x3);
2695
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002696 break;
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002697 case SND_SOC_DAPM_POST_PMU:
Kiran Kandid8cf5212012-03-02 15:34:53 -08002698
2699 usleep_range(20000, 20000);
2700
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002701 if (tabla->mbhc_polling_active &&
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002702 tabla->mbhc_cfg.micbias == micb_line) {
2703 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002704 tabla_codec_pause_hs_polling(codec);
2705 tabla_codec_start_hs_polling(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002706 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
Bradley Rubin4d09cf42011-08-17 17:59:16 -07002707 }
2708 break;
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002709
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002710 case SND_SOC_DAPM_POST_PMD:
Joonwoo Park03324832012-03-19 19:36:16 -07002711 if ((w->reg == tabla->mbhc_bias_regs.ctl_reg) &&
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002712 tabla_is_hph_pa_on(codec)) {
2713 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002714 tabla_codec_switch_micbias(codec, 1);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002715 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
2716 }
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07002717
Bradley Rubin229c6a52011-07-12 16:18:48 -07002718 if (strnstr(w->name, internal1_text, 30))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002719 snd_soc_update_bits(codec, micb_int_reg, 0x80, 0x00);
Bradley Rubin229c6a52011-07-12 16:18:48 -07002720 else if (strnstr(w->name, internal2_text, 30))
2721 snd_soc_update_bits(codec, micb_int_reg, 0x10, 0x00);
2722 else if (strnstr(w->name, internal3_text, 30))
2723 snd_soc_update_bits(codec, micb_int_reg, 0x2, 0x0);
2724
Patrick Lai3043fba2011-08-01 14:15:57 -07002725 tabla_codec_update_cfilt_usage(codec, cfilt_sel_val, 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002726 break;
2727 }
2728
2729 return 0;
2730}
2731
Kiran Kandid8cf5212012-03-02 15:34:53 -08002732
2733static void tx_hpf_corner_freq_callback(struct work_struct *work)
2734{
2735 struct delayed_work *hpf_delayed_work;
2736 struct hpf_work *hpf_work;
2737 struct tabla_priv *tabla;
2738 struct snd_soc_codec *codec;
2739 u16 tx_mux_ctl_reg;
2740 u8 hpf_cut_of_freq;
2741
2742 hpf_delayed_work = to_delayed_work(work);
2743 hpf_work = container_of(hpf_delayed_work, struct hpf_work, dwork);
2744 tabla = hpf_work->tabla;
2745 codec = hpf_work->tabla->codec;
2746 hpf_cut_of_freq = hpf_work->tx_hpf_cut_of_freq;
2747
2748 tx_mux_ctl_reg = TABLA_A_CDC_TX1_MUX_CTL +
2749 (hpf_work->decimator - 1) * 8;
2750
2751 pr_debug("%s(): decimator %u hpf_cut_of_freq 0x%x\n", __func__,
2752 hpf_work->decimator, (unsigned int)hpf_cut_of_freq);
2753
2754 snd_soc_update_bits(codec, tx_mux_ctl_reg, 0x30, hpf_cut_of_freq << 4);
2755}
2756
2757#define TX_MUX_CTL_CUT_OFF_FREQ_MASK 0x30
2758#define CF_MIN_3DB_4HZ 0x0
2759#define CF_MIN_3DB_75HZ 0x1
2760#define CF_MIN_3DB_150HZ 0x2
2761
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002762static int tabla_codec_enable_dec(struct snd_soc_dapm_widget *w,
2763 struct snd_kcontrol *kcontrol, int event)
2764{
2765 struct snd_soc_codec *codec = w->codec;
Kiran Kandid8cf5212012-03-02 15:34:53 -08002766 unsigned int decimator;
2767 char *dec_name = NULL;
2768 char *widget_name = NULL;
2769 char *temp;
2770 int ret = 0;
2771 u16 dec_reset_reg, tx_vol_ctl_reg, tx_mux_ctl_reg;
2772 u8 dec_hpf_cut_of_freq;
Kuirong Wange9c8a222012-03-28 16:24:09 -07002773 int offset;
2774
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002775
2776 pr_debug("%s %d\n", __func__, event);
2777
Kiran Kandid8cf5212012-03-02 15:34:53 -08002778 widget_name = kstrndup(w->name, 15, GFP_KERNEL);
2779 if (!widget_name)
2780 return -ENOMEM;
2781 temp = widget_name;
2782
2783 dec_name = strsep(&widget_name, " ");
2784 widget_name = temp;
2785 if (!dec_name) {
2786 pr_err("%s: Invalid decimator = %s\n", __func__, w->name);
2787 ret = -EINVAL;
2788 goto out;
2789 }
2790
2791 ret = kstrtouint(strpbrk(dec_name, "123456789"), 10, &decimator);
2792 if (ret < 0) {
2793 pr_err("%s: Invalid decimator = %s\n", __func__, dec_name);
2794 ret = -EINVAL;
2795 goto out;
2796 }
2797
2798 pr_debug("%s(): widget = %s dec_name = %s decimator = %u\n", __func__,
2799 w->name, dec_name, decimator);
2800
Kuirong Wange9c8a222012-03-28 16:24:09 -07002801 if (w->reg == TABLA_A_CDC_CLK_TX_CLK_EN_B1_CTL) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002802 dec_reset_reg = TABLA_A_CDC_CLK_TX_RESET_B1_CTL;
Kuirong Wange9c8a222012-03-28 16:24:09 -07002803 offset = 0;
2804 } else if (w->reg == TABLA_A_CDC_CLK_TX_CLK_EN_B2_CTL) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002805 dec_reset_reg = TABLA_A_CDC_CLK_TX_RESET_B2_CTL;
Kuirong Wange9c8a222012-03-28 16:24:09 -07002806 offset = 8;
2807 } else {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002808 pr_err("%s: Error, incorrect dec\n", __func__);
2809 return -EINVAL;
2810 }
2811
Kiran Kandid8cf5212012-03-02 15:34:53 -08002812 tx_vol_ctl_reg = TABLA_A_CDC_TX1_VOL_CTL_CFG + 8 * (decimator -1);
2813 tx_mux_ctl_reg = TABLA_A_CDC_TX1_MUX_CTL + 8 * (decimator - 1);
2814
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002815 switch (event) {
2816 case SND_SOC_DAPM_PRE_PMU:
Kiran Kandid8cf5212012-03-02 15:34:53 -08002817
2818 // Enableable TX digital mute */
2819 snd_soc_update_bits(codec, tx_vol_ctl_reg, 0x01, 0x01);
2820
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002821 snd_soc_update_bits(codec, dec_reset_reg, 1 << w->shift,
2822 1 << w->shift);
2823 snd_soc_update_bits(codec, dec_reset_reg, 1 << w->shift, 0x0);
Kiran Kandid8cf5212012-03-02 15:34:53 -08002824
2825 dec_hpf_cut_of_freq = snd_soc_read(codec, tx_mux_ctl_reg);
2826
2827 dec_hpf_cut_of_freq = (dec_hpf_cut_of_freq & 0x30) >> 4;
2828
2829 tx_hpf_work[decimator - 1].tx_hpf_cut_of_freq =
2830 dec_hpf_cut_of_freq;
2831
2832 if ((dec_hpf_cut_of_freq != CF_MIN_3DB_150HZ)) {
2833
2834 /* set cut of freq to CF_MIN_3DB_150HZ (0x1); */
2835 snd_soc_update_bits(codec, tx_mux_ctl_reg, 0x30,
2836 CF_MIN_3DB_150HZ << 4);
2837 }
2838
2839 /* enable HPF */
2840 snd_soc_update_bits(codec, tx_mux_ctl_reg , 0x08, 0x00);
2841
2842 break;
2843
2844 case SND_SOC_DAPM_POST_PMU:
2845
2846 /* Disable TX digital mute */
2847 snd_soc_update_bits(codec, tx_vol_ctl_reg, 0x01, 0x00);
2848
2849 if (tx_hpf_work[decimator - 1].tx_hpf_cut_of_freq !=
2850 CF_MIN_3DB_150HZ) {
2851
2852 schedule_delayed_work(&tx_hpf_work[decimator - 1].dwork,
2853 msecs_to_jiffies(300));
2854 }
Kuirong Wange9c8a222012-03-28 16:24:09 -07002855 /* apply the digital gain after the decimator is enabled*/
2856 if ((w->shift) < ARRAY_SIZE(rx_digital_gain_reg))
2857 snd_soc_write(codec,
2858 tx_digital_gain_reg[w->shift + offset],
2859 snd_soc_read(codec,
2860 tx_digital_gain_reg[w->shift + offset])
2861 );
2862
Kiran Kandid8cf5212012-03-02 15:34:53 -08002863 break;
2864
2865 case SND_SOC_DAPM_PRE_PMD:
2866
2867 snd_soc_update_bits(codec, tx_vol_ctl_reg, 0x01, 0x01);
2868 cancel_delayed_work_sync(&tx_hpf_work[decimator - 1].dwork);
2869 break;
2870
2871 case SND_SOC_DAPM_POST_PMD:
2872
2873 snd_soc_update_bits(codec, tx_mux_ctl_reg, 0x08, 0x08);
2874 snd_soc_update_bits(codec, tx_mux_ctl_reg, 0x30,
2875 (tx_hpf_work[decimator - 1].tx_hpf_cut_of_freq) << 4);
2876
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002877 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002878 }
Kiran Kandid8cf5212012-03-02 15:34:53 -08002879out:
2880 kfree(widget_name);
2881 return ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002882}
2883
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07002884static int tabla_codec_reset_interpolator(struct snd_soc_dapm_widget *w,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002885 struct snd_kcontrol *kcontrol, int event)
2886{
2887 struct snd_soc_codec *codec = w->codec;
2888
Kiran Kandi8b3a8302011-09-27 16:13:28 -07002889 pr_debug("%s %d %s\n", __func__, event, w->name);
2890
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002891 switch (event) {
2892 case SND_SOC_DAPM_PRE_PMU:
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07002893 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_RX_RESET_CTL,
2894 1 << w->shift, 1 << w->shift);
2895 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_RX_RESET_CTL,
2896 1 << w->shift, 0x0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002897 break;
Kuirong Wange9c8a222012-03-28 16:24:09 -07002898 case SND_SOC_DAPM_POST_PMU:
2899 /* apply the digital gain after the interpolator is enabled*/
2900 if ((w->shift) < ARRAY_SIZE(rx_digital_gain_reg))
2901 snd_soc_write(codec,
2902 rx_digital_gain_reg[w->shift],
2903 snd_soc_read(codec,
2904 rx_digital_gain_reg[w->shift])
2905 );
2906 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002907 }
2908 return 0;
2909}
2910
Bradley Rubin229c6a52011-07-12 16:18:48 -07002911static int tabla_codec_enable_ldo_h(struct snd_soc_dapm_widget *w,
2912 struct snd_kcontrol *kcontrol, int event)
2913{
2914 switch (event) {
2915 case SND_SOC_DAPM_POST_PMU:
2916 case SND_SOC_DAPM_POST_PMD:
2917 usleep_range(1000, 1000);
2918 break;
2919 }
2920 return 0;
2921}
2922
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07002923static int tabla_codec_enable_rx_bias(struct snd_soc_dapm_widget *w,
2924 struct snd_kcontrol *kcontrol, int event)
2925{
2926 struct snd_soc_codec *codec = w->codec;
2927
2928 pr_debug("%s %d\n", __func__, event);
2929
2930 switch (event) {
2931 case SND_SOC_DAPM_PRE_PMU:
2932 tabla_enable_rx_bias(codec, 1);
2933 break;
2934 case SND_SOC_DAPM_POST_PMD:
2935 tabla_enable_rx_bias(codec, 0);
2936 break;
2937 }
2938 return 0;
2939}
Kiran Kandi8b3a8302011-09-27 16:13:28 -07002940static int tabla_hphr_dac_event(struct snd_soc_dapm_widget *w,
2941 struct snd_kcontrol *kcontrol, int event)
2942{
2943 struct snd_soc_codec *codec = w->codec;
2944
2945 pr_debug("%s %s %d\n", __func__, w->name, event);
2946
2947 switch (event) {
2948 case SND_SOC_DAPM_PRE_PMU:
2949 snd_soc_update_bits(codec, w->reg, 0x40, 0x40);
2950 break;
2951 case SND_SOC_DAPM_POST_PMD:
2952 snd_soc_update_bits(codec, w->reg, 0x40, 0x00);
2953 break;
2954 }
2955 return 0;
2956}
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07002957
Joonwoo Park8b1f0982011-12-08 17:12:45 -08002958static void tabla_snd_soc_jack_report(struct tabla_priv *tabla,
2959 struct snd_soc_jack *jack, int status,
2960 int mask)
2961{
2962 /* XXX: wake_lock_timeout()? */
Joonwoo Park03324832012-03-19 19:36:16 -07002963 snd_soc_jack_report_no_dapm(jack, status, mask);
Joonwoo Park8b1f0982011-12-08 17:12:45 -08002964}
2965
Patrick Lai49efeac2011-11-03 11:01:12 -07002966static void hphocp_off_report(struct tabla_priv *tabla,
2967 u32 jack_status, int irq)
2968{
2969 struct snd_soc_codec *codec;
Joonwoo Park03324832012-03-19 19:36:16 -07002970 if (!tabla) {
2971 pr_err("%s: Bad tabla private data\n", __func__);
2972 return;
2973 }
Patrick Lai49efeac2011-11-03 11:01:12 -07002974
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002975 pr_debug("%s: clear ocp status %x\n", __func__, jack_status);
Joonwoo Park03324832012-03-19 19:36:16 -07002976 codec = tabla->codec;
2977 if (tabla->hph_status & jack_status) {
Patrick Lai49efeac2011-11-03 11:01:12 -07002978 tabla->hph_status &= ~jack_status;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07002979 if (tabla->mbhc_cfg.headset_jack)
2980 tabla_snd_soc_jack_report(tabla,
2981 tabla->mbhc_cfg.headset_jack,
Joonwoo Park8b1f0982011-12-08 17:12:45 -08002982 tabla->hph_status,
2983 TABLA_JACK_MASK);
Joonwoo Park0976d012011-12-22 11:48:18 -08002984 snd_soc_update_bits(codec, TABLA_A_RX_HPH_OCP_CTL, 0x10, 0x00);
2985 snd_soc_update_bits(codec, TABLA_A_RX_HPH_OCP_CTL, 0x10, 0x10);
Patrick Laic7cae882011-11-18 11:52:49 -08002986 /* reset retry counter as PA is turned off signifying
2987 * start of new OCP detection session
2988 */
2989 if (TABLA_IRQ_HPH_PA_OCPL_FAULT)
2990 tabla->hphlocp_cnt = 0;
2991 else
2992 tabla->hphrocp_cnt = 0;
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05302993 wcd9xxx_enable_irq(codec->control_data, irq);
Patrick Lai49efeac2011-11-03 11:01:12 -07002994 }
2995}
2996
2997static void hphlocp_off_report(struct work_struct *work)
2998{
2999 struct tabla_priv *tabla = container_of(work, struct tabla_priv,
3000 hphlocp_work);
3001 hphocp_off_report(tabla, SND_JACK_OC_HPHL, TABLA_IRQ_HPH_PA_OCPL_FAULT);
3002}
3003
3004static void hphrocp_off_report(struct work_struct *work)
3005{
3006 struct tabla_priv *tabla = container_of(work, struct tabla_priv,
3007 hphrocp_work);
3008 hphocp_off_report(tabla, SND_JACK_OC_HPHR, TABLA_IRQ_HPH_PA_OCPR_FAULT);
3009}
3010
Kiran Kandibf0b1ff2011-09-15 13:55:21 -07003011static int tabla_hph_pa_event(struct snd_soc_dapm_widget *w,
3012 struct snd_kcontrol *kcontrol, int event)
3013{
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07003014 struct snd_soc_codec *codec = w->codec;
3015 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
3016 u8 mbhc_micb_ctl_val;
Kiran Kandibf0b1ff2011-09-15 13:55:21 -07003017 pr_debug("%s: event = %d\n", __func__, event);
3018
3019 switch (event) {
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07003020 case SND_SOC_DAPM_PRE_PMU:
3021 mbhc_micb_ctl_val = snd_soc_read(codec,
3022 tabla->mbhc_bias_regs.ctl_reg);
3023
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07003024 if (!(mbhc_micb_ctl_val & 0x80)) {
3025 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07003026 tabla_codec_switch_micbias(codec, 1);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07003027 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
3028 }
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07003029 break;
3030
Kiran Kandibf0b1ff2011-09-15 13:55:21 -07003031 case SND_SOC_DAPM_POST_PMD:
Patrick Lai49efeac2011-11-03 11:01:12 -07003032 /* schedule work is required because at the time HPH PA DAPM
3033 * event callback is called by DAPM framework, CODEC dapm mutex
3034 * would have been locked while snd_soc_jack_report also
3035 * attempts to acquire same lock.
3036 */
Joonwoo Parka9444452011-12-08 18:48:27 -08003037 if (w->shift == 5) {
3038 clear_bit(TABLA_HPHL_PA_OFF_ACK,
3039 &tabla->hph_pa_dac_state);
3040 clear_bit(TABLA_HPHL_DAC_OFF_ACK,
3041 &tabla->hph_pa_dac_state);
3042 if (tabla->hph_status & SND_JACK_OC_HPHL)
3043 schedule_work(&tabla->hphlocp_work);
3044 } else if (w->shift == 4) {
3045 clear_bit(TABLA_HPHR_PA_OFF_ACK,
3046 &tabla->hph_pa_dac_state);
3047 clear_bit(TABLA_HPHR_DAC_OFF_ACK,
3048 &tabla->hph_pa_dac_state);
3049 if (tabla->hph_status & SND_JACK_OC_HPHR)
3050 schedule_work(&tabla->hphrocp_work);
3051 }
3052
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07003053 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Joonwoo Park03324832012-03-19 19:36:16 -07003054 tabla_codec_switch_micbias(codec, 0);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07003055 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07003056
Kiran Kandibf0b1ff2011-09-15 13:55:21 -07003057 pr_debug("%s: sleep 10 ms after %s PA disable.\n", __func__,
3058 w->name);
3059 usleep_range(10000, 10000);
Kiran Kandibf0b1ff2011-09-15 13:55:21 -07003060 break;
3061 }
3062 return 0;
3063}
3064
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003065static void tabla_get_mbhc_micbias_regs(struct snd_soc_codec *codec,
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08003066 struct mbhc_micbias_regs *micbias_regs)
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003067{
3068 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003069 unsigned int cfilt;
3070
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07003071 switch (tabla->mbhc_cfg.micbias) {
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003072 case TABLA_MICBIAS1:
3073 cfilt = tabla->pdata->micbias.bias1_cfilt_sel;
3074 micbias_regs->mbhc_reg = TABLA_A_MICB_1_MBHC;
3075 micbias_regs->int_rbias = TABLA_A_MICB_1_INT_RBIAS;
3076 micbias_regs->ctl_reg = TABLA_A_MICB_1_CTL;
3077 break;
3078 case TABLA_MICBIAS2:
3079 cfilt = tabla->pdata->micbias.bias2_cfilt_sel;
3080 micbias_regs->mbhc_reg = TABLA_A_MICB_2_MBHC;
3081 micbias_regs->int_rbias = TABLA_A_MICB_2_INT_RBIAS;
3082 micbias_regs->ctl_reg = TABLA_A_MICB_2_CTL;
3083 break;
3084 case TABLA_MICBIAS3:
3085 cfilt = tabla->pdata->micbias.bias3_cfilt_sel;
3086 micbias_regs->mbhc_reg = TABLA_A_MICB_3_MBHC;
3087 micbias_regs->int_rbias = TABLA_A_MICB_3_INT_RBIAS;
3088 micbias_regs->ctl_reg = TABLA_A_MICB_3_CTL;
3089 break;
3090 case TABLA_MICBIAS4:
3091 cfilt = tabla->pdata->micbias.bias4_cfilt_sel;
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08003092 micbias_regs->mbhc_reg = tabla->reg_addr.micb_4_mbhc;
3093 micbias_regs->int_rbias = tabla->reg_addr.micb_4_int_rbias;
3094 micbias_regs->ctl_reg = tabla->reg_addr.micb_4_ctl;
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003095 break;
3096 default:
3097 /* Should never reach here */
3098 pr_err("%s: Invalid MIC BIAS for MBHC\n", __func__);
Jordan Crouse239d8412011-11-23 11:47:02 -07003099 return;
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003100 }
3101
Bhalchandra Gajare19d9c132011-11-18 14:57:08 -08003102 micbias_regs->cfilt_sel = cfilt;
3103
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003104 switch (cfilt) {
3105 case TABLA_CFILT1_SEL:
3106 micbias_regs->cfilt_val = TABLA_A_MICB_CFILT_1_VAL;
3107 micbias_regs->cfilt_ctl = TABLA_A_MICB_CFILT_1_CTL;
Joonwoo Park0976d012011-12-22 11:48:18 -08003108 tabla->mbhc_data.micb_mv = tabla->pdata->micbias.cfilt1_mv;
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003109 break;
3110 case TABLA_CFILT2_SEL:
3111 micbias_regs->cfilt_val = TABLA_A_MICB_CFILT_2_VAL;
3112 micbias_regs->cfilt_ctl = TABLA_A_MICB_CFILT_2_CTL;
Joonwoo Park0976d012011-12-22 11:48:18 -08003113 tabla->mbhc_data.micb_mv = tabla->pdata->micbias.cfilt2_mv;
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003114 break;
3115 case TABLA_CFILT3_SEL:
3116 micbias_regs->cfilt_val = TABLA_A_MICB_CFILT_3_VAL;
3117 micbias_regs->cfilt_ctl = TABLA_A_MICB_CFILT_3_CTL;
Joonwoo Park0976d012011-12-22 11:48:18 -08003118 tabla->mbhc_data.micb_mv = tabla->pdata->micbias.cfilt3_mv;
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003119 break;
3120 }
3121}
Santosh Mardie15e2302011-11-15 10:39:23 +05303122static const struct snd_soc_dapm_widget tabla_dapm_i2s_widgets[] = {
3123 SND_SOC_DAPM_SUPPLY("RX_I2S_CLK", TABLA_A_CDC_CLK_RX_I2S_CTL,
3124 4, 0, NULL, 0),
3125 SND_SOC_DAPM_SUPPLY("TX_I2S_CLK", TABLA_A_CDC_CLK_TX_I2S_CTL, 4,
3126 0, NULL, 0),
3127};
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07003128
Kiran Kandi8b3a8302011-09-27 16:13:28 -07003129static int tabla_lineout_dac_event(struct snd_soc_dapm_widget *w,
3130 struct snd_kcontrol *kcontrol, int event)
3131{
3132 struct snd_soc_codec *codec = w->codec;
3133
3134 pr_debug("%s %s %d\n", __func__, w->name, event);
3135
3136 switch (event) {
3137 case SND_SOC_DAPM_PRE_PMU:
3138 snd_soc_update_bits(codec, w->reg, 0x40, 0x40);
3139 break;
3140
3141 case SND_SOC_DAPM_POST_PMD:
3142 snd_soc_update_bits(codec, w->reg, 0x40, 0x00);
3143 break;
3144 }
3145 return 0;
3146}
3147
Damir Didjusto52900462012-08-16 21:22:29 -07003148static int tabla_ear_pa_event(struct snd_soc_dapm_widget *w,
3149 struct snd_kcontrol *kcontrol, int event)
3150{
3151 struct snd_soc_codec *codec = w->codec;
3152
3153 pr_debug("%s %d\n", __func__, event);
3154
3155 switch (event) {
3156 case SND_SOC_DAPM_PRE_PMU:
3157 snd_soc_update_bits(codec, TABLA_A_RX_EAR_EN, 0x50, 0x50);
3158 break;
3159
3160 case SND_SOC_DAPM_PRE_PMD:
3161 snd_soc_update_bits(codec, TABLA_A_RX_EAR_EN, 0x10, 0x00);
3162 snd_soc_update_bits(codec, TABLA_A_RX_EAR_EN, 0x40, 0x00);
3163 break;
3164 }
3165 return 0;
3166}
3167
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08003168static const struct snd_soc_dapm_widget tabla_1_x_dapm_widgets[] = {
3169 SND_SOC_DAPM_MICBIAS_E("MIC BIAS4 External", TABLA_1_A_MICB_4_CTL, 7,
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05303170 0, tabla_codec_enable_micbias,
3171 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
3172 SND_SOC_DAPM_POST_PMD),
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08003173};
3174
3175static const struct snd_soc_dapm_widget tabla_2_higher_dapm_widgets[] = {
3176 SND_SOC_DAPM_MICBIAS_E("MIC BIAS4 External", TABLA_2_A_MICB_4_CTL, 7,
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05303177 0, tabla_codec_enable_micbias,
3178 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
3179 SND_SOC_DAPM_POST_PMD),
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08003180};
3181
Santosh Mardie15e2302011-11-15 10:39:23 +05303182static const struct snd_soc_dapm_route audio_i2s_map[] = {
3183 {"RX_I2S_CLK", NULL, "CDC_CONN"},
3184 {"SLIM RX1", NULL, "RX_I2S_CLK"},
3185 {"SLIM RX2", NULL, "RX_I2S_CLK"},
3186 {"SLIM RX3", NULL, "RX_I2S_CLK"},
3187 {"SLIM RX4", NULL, "RX_I2S_CLK"},
3188
3189 {"SLIM TX7", NULL, "TX_I2S_CLK"},
3190 {"SLIM TX8", NULL, "TX_I2S_CLK"},
3191 {"SLIM TX9", NULL, "TX_I2S_CLK"},
3192 {"SLIM TX10", NULL, "TX_I2S_CLK"},
3193};
3194
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003195static const struct snd_soc_dapm_route audio_map[] = {
3196 /* SLIMBUS Connections */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003197
3198 {"SLIM TX1", NULL, "SLIM TX1 MUX"},
3199 {"SLIM TX1 MUX", "DEC1", "DEC1 MUX"},
3200
Kiran Kandi1e6371d2012-03-29 11:48:57 -07003201 {"SLIM TX2", NULL, "SLIM TX2 MUX"},
3202 {"SLIM TX2 MUX", "DEC2", "DEC2 MUX"},
3203
3204 {"SLIM TX3", NULL, "SLIM TX3 MUX"},
3205 {"SLIM TX3 MUX", "DEC3", "DEC3 MUX"},
Neema Shetty3fb1b802012-04-27 13:53:24 -07003206 {"SLIM TX3 MUX", "RMIX1", "RX1 MIX1"},
3207 {"SLIM TX3 MUX", "RMIX2", "RX2 MIX1"},
3208 {"SLIM TX3 MUX", "RMIX3", "RX3 MIX1"},
3209 {"SLIM TX3 MUX", "RMIX4", "RX4 MIX1"},
3210 {"SLIM TX3 MUX", "RMIX5", "RX5 MIX1"},
3211 {"SLIM TX3 MUX", "RMIX6", "RX6 MIX1"},
3212 {"SLIM TX3 MUX", "RMIX7", "RX7 MIX1"},
Kiran Kandi1e6371d2012-03-29 11:48:57 -07003213
3214 {"SLIM TX4", NULL, "SLIM TX4 MUX"},
3215 {"SLIM TX4 MUX", "DEC4", "DEC4 MUX"},
3216
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003217 {"SLIM TX5", NULL, "SLIM TX5 MUX"},
3218 {"SLIM TX5 MUX", "DEC5", "DEC5 MUX"},
Neema Shetty3fb1b802012-04-27 13:53:24 -07003219 {"SLIM TX5 MUX", "RMIX1", "RX1 MIX1"},
3220 {"SLIM TX5 MUX", "RMIX2", "RX2 MIX1"},
3221 {"SLIM TX5 MUX", "RMIX3", "RX3 MIX1"},
3222 {"SLIM TX5 MUX", "RMIX4", "RX4 MIX1"},
3223 {"SLIM TX5 MUX", "RMIX5", "RX5 MIX1"},
3224 {"SLIM TX5 MUX", "RMIX6", "RX6 MIX1"},
3225 {"SLIM TX5 MUX", "RMIX7", "RX7 MIX1"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003226
3227 {"SLIM TX6", NULL, "SLIM TX6 MUX"},
3228 {"SLIM TX6 MUX", "DEC6", "DEC6 MUX"},
3229
3230 {"SLIM TX7", NULL, "SLIM TX7 MUX"},
3231 {"SLIM TX7 MUX", "DEC1", "DEC1 MUX"},
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07003232 {"SLIM TX7 MUX", "DEC2", "DEC2 MUX"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003233 {"SLIM TX7 MUX", "DEC3", "DEC3 MUX"},
3234 {"SLIM TX7 MUX", "DEC4", "DEC4 MUX"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003235 {"SLIM TX7 MUX", "DEC5", "DEC5 MUX"},
3236 {"SLIM TX7 MUX", "DEC6", "DEC6 MUX"},
Bhalchandra Gajare0d77e1b2011-07-08 10:54:14 -07003237 {"SLIM TX7 MUX", "DEC7", "DEC7 MUX"},
3238 {"SLIM TX7 MUX", "DEC8", "DEC8 MUX"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003239 {"SLIM TX7 MUX", "DEC9", "DEC9 MUX"},
3240 {"SLIM TX7 MUX", "DEC10", "DEC10 MUX"},
Neema Shetty3fb1b802012-04-27 13:53:24 -07003241 {"SLIM TX7 MUX", "RMIX1", "RX1 MIX1"},
3242 {"SLIM TX7 MUX", "RMIX2", "RX2 MIX1"},
3243 {"SLIM TX7 MUX", "RMIX3", "RX3 MIX1"},
3244 {"SLIM TX7 MUX", "RMIX4", "RX4 MIX1"},
3245 {"SLIM TX7 MUX", "RMIX5", "RX5 MIX1"},
3246 {"SLIM TX7 MUX", "RMIX6", "RX6 MIX1"},
3247 {"SLIM TX7 MUX", "RMIX7", "RX7 MIX1"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003248
3249 {"SLIM TX8", NULL, "SLIM TX8 MUX"},
Kiran Kandicf45f6a2011-07-17 21:10:19 -07003250 {"SLIM TX8 MUX", "DEC1", "DEC1 MUX"},
3251 {"SLIM TX8 MUX", "DEC2", "DEC2 MUX"},
3252 {"SLIM TX8 MUX", "DEC3", "DEC3 MUX"},
Bhalchandra Gajare9ec83cd2011-09-23 17:25:07 -07003253 {"SLIM TX8 MUX", "DEC4", "DEC4 MUX"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003254 {"SLIM TX8 MUX", "DEC5", "DEC5 MUX"},
3255 {"SLIM TX8 MUX", "DEC6", "DEC6 MUX"},
Kiran Kandi1e6371d2012-03-29 11:48:57 -07003256 {"SLIM TX8 MUX", "DEC7", "DEC7 MUX"},
3257 {"SLIM TX8 MUX", "DEC8", "DEC8 MUX"},
3258 {"SLIM TX8 MUX", "DEC9", "DEC9 MUX"},
3259 {"SLIM TX8 MUX", "DEC10", "DEC10 MUX"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003260
Kiran Kandi3426e512011-09-13 22:50:10 -07003261 {"SLIM TX9", NULL, "SLIM TX9 MUX"},
3262 {"SLIM TX9 MUX", "DEC1", "DEC1 MUX"},
3263 {"SLIM TX9 MUX", "DEC2", "DEC2 MUX"},
3264 {"SLIM TX9 MUX", "DEC3", "DEC3 MUX"},
3265 {"SLIM TX9 MUX", "DEC4", "DEC4 MUX"},
3266 {"SLIM TX9 MUX", "DEC5", "DEC5 MUX"},
3267 {"SLIM TX9 MUX", "DEC6", "DEC6 MUX"},
3268 {"SLIM TX9 MUX", "DEC7", "DEC7 MUX"},
3269 {"SLIM TX9 MUX", "DEC8", "DEC8 MUX"},
3270 {"SLIM TX9 MUX", "DEC9", "DEC9 MUX"},
3271 {"SLIM TX9 MUX", "DEC10", "DEC10 MUX"},
3272
3273 {"SLIM TX10", NULL, "SLIM TX10 MUX"},
3274 {"SLIM TX10 MUX", "DEC1", "DEC1 MUX"},
3275 {"SLIM TX10 MUX", "DEC2", "DEC2 MUX"},
3276 {"SLIM TX10 MUX", "DEC3", "DEC3 MUX"},
3277 {"SLIM TX10 MUX", "DEC4", "DEC4 MUX"},
3278 {"SLIM TX10 MUX", "DEC5", "DEC5 MUX"},
3279 {"SLIM TX10 MUX", "DEC6", "DEC6 MUX"},
3280 {"SLIM TX10 MUX", "DEC7", "DEC7 MUX"},
3281 {"SLIM TX10 MUX", "DEC8", "DEC8 MUX"},
3282 {"SLIM TX10 MUX", "DEC9", "DEC9 MUX"},
3283 {"SLIM TX10 MUX", "DEC10", "DEC10 MUX"},
3284
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003285 /* Earpiece (RX MIX1) */
3286 {"EAR", NULL, "EAR PA"},
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08003287 {"EAR PA", NULL, "EAR_PA_MIXER"},
3288 {"EAR_PA_MIXER", NULL, "DAC1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003289 {"DAC1", NULL, "CP"},
3290
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08003291 {"ANC1 FB MUX", "EAR_HPH_L", "RX1 MIX2"},
3292 {"ANC1 FB MUX", "EAR_LINE_1", "RX2 MIX2"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003293 {"ANC", NULL, "ANC1 FB MUX"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003294
3295 /* Headset (RX MIX1 and RX MIX2) */
3296 {"HEADPHONE", NULL, "HPHL"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003297 {"HEADPHONE", NULL, "HPHR"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003298
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08003299 {"HPHL", NULL, "HPHL_PA_MIXER"},
3300 {"HPHL_PA_MIXER", NULL, "HPHL DAC"},
3301
3302 {"HPHR", NULL, "HPHR_PA_MIXER"},
3303 {"HPHR_PA_MIXER", NULL, "HPHR DAC"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003304
3305 {"HPHL DAC", NULL, "CP"},
3306 {"HPHR DAC", NULL, "CP"},
3307
3308 {"ANC", NULL, "ANC1 MUX"},
3309 {"ANC", NULL, "ANC2 MUX"},
3310 {"ANC1 MUX", "ADC1", "ADC1"},
3311 {"ANC1 MUX", "ADC2", "ADC2"},
3312 {"ANC1 MUX", "ADC3", "ADC3"},
3313 {"ANC1 MUX", "ADC4", "ADC4"},
3314 {"ANC2 MUX", "ADC1", "ADC1"},
3315 {"ANC2 MUX", "ADC2", "ADC2"},
3316 {"ANC2 MUX", "ADC3", "ADC3"},
3317 {"ANC2 MUX", "ADC4", "ADC4"},
3318
Bradley Rubine1d08622011-07-20 18:01:35 -07003319 {"ANC", NULL, "CDC_CONN"},
3320
Bradley Rubin229c6a52011-07-12 16:18:48 -07003321 {"DAC1", "Switch", "RX1 CHAIN"},
3322 {"HPHL DAC", "Switch", "RX1 CHAIN"},
Kiran Kandi8b3a8302011-09-27 16:13:28 -07003323 {"HPHR DAC", NULL, "RX2 CHAIN"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003324
Kiran Kandidb0a4b02011-08-23 09:32:09 -07003325 {"LINEOUT1", NULL, "LINEOUT1 PA"},
3326 {"LINEOUT2", NULL, "LINEOUT2 PA"},
3327 {"LINEOUT3", NULL, "LINEOUT3 PA"},
3328 {"LINEOUT4", NULL, "LINEOUT4 PA"},
3329 {"LINEOUT5", NULL, "LINEOUT5 PA"},
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07003330
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08003331 {"LINEOUT1 PA", NULL, "LINEOUT1_PA_MIXER"},
3332 {"LINEOUT1_PA_MIXER", NULL, "LINEOUT1 DAC"},
3333 {"LINEOUT2 PA", NULL, "LINEOUT2_PA_MIXER"},
3334 {"LINEOUT2_PA_MIXER", NULL, "LINEOUT2 DAC"},
3335 {"LINEOUT3 PA", NULL, "LINEOUT3_PA_MIXER"},
3336 {"LINEOUT3_PA_MIXER", NULL, "LINEOUT3 DAC"},
3337 {"LINEOUT4 PA", NULL, "LINEOUT4_PA_MIXER"},
3338 {"LINEOUT4_PA_MIXER", NULL, "LINEOUT4 DAC"},
3339 {"LINEOUT5 PA", NULL, "LINEOUT5_PA_MIXER"},
3340 {"LINEOUT5_PA_MIXER", NULL, "LINEOUT5 DAC"},
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07003341
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08003342 {"LINEOUT1 DAC", NULL, "RX3 MIX2"},
Kiran Kandi8b3a8302011-09-27 16:13:28 -07003343 {"LINEOUT5 DAC", NULL, "RX7 MIX1"},
3344
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08003345 {"RX1 CHAIN", NULL, "RX1 MIX2"},
3346 {"RX2 CHAIN", NULL, "RX2 MIX2"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003347 {"RX1 CHAIN", NULL, "ANC"},
3348 {"RX2 CHAIN", NULL, "ANC"},
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07003349
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003350 {"CP", NULL, "RX_BIAS"},
3351 {"LINEOUT1 DAC", NULL, "RX_BIAS"},
3352 {"LINEOUT2 DAC", NULL, "RX_BIAS"},
3353 {"LINEOUT3 DAC", NULL, "RX_BIAS"},
3354 {"LINEOUT4 DAC", NULL, "RX_BIAS"},
Kiran Kandi8b3a8302011-09-27 16:13:28 -07003355 {"LINEOUT5 DAC", NULL, "RX_BIAS"},
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003356
Kuirong Wang0f8ade32012-02-27 16:29:45 -08003357 {"RX1 MIX1", NULL, "COMP1_CLK"},
3358 {"RX2 MIX1", NULL, "COMP1_CLK"},
3359 {"RX3 MIX1", NULL, "COMP2_CLK"},
3360 {"RX5 MIX1", NULL, "COMP2_CLK"},
3361
3362
Bradley Rubin229c6a52011-07-12 16:18:48 -07003363 {"RX1 MIX1", NULL, "RX1 MIX1 INP1"},
3364 {"RX1 MIX1", NULL, "RX1 MIX1 INP2"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003365 {"RX1 MIX1", NULL, "RX1 MIX1 INP3"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003366 {"RX2 MIX1", NULL, "RX2 MIX1 INP1"},
3367 {"RX2 MIX1", NULL, "RX2 MIX1 INP2"},
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07003368 {"RX3 MIX1", NULL, "RX3 MIX1 INP1"},
3369 {"RX3 MIX1", NULL, "RX3 MIX1 INP2"},
3370 {"RX4 MIX1", NULL, "RX4 MIX1 INP1"},
3371 {"RX4 MIX1", NULL, "RX4 MIX1 INP2"},
3372 {"RX5 MIX1", NULL, "RX5 MIX1 INP1"},
3373 {"RX5 MIX1", NULL, "RX5 MIX1 INP2"},
3374 {"RX6 MIX1", NULL, "RX6 MIX1 INP1"},
3375 {"RX6 MIX1", NULL, "RX6 MIX1 INP2"},
Bhalchandra Gajare0a8ad172011-08-12 13:32:22 -07003376 {"RX7 MIX1", NULL, "RX7 MIX1 INP1"},
3377 {"RX7 MIX1", NULL, "RX7 MIX1 INP2"},
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08003378 {"RX1 MIX2", NULL, "RX1 MIX1"},
3379 {"RX1 MIX2", NULL, "RX1 MIX2 INP1"},
3380 {"RX1 MIX2", NULL, "RX1 MIX2 INP2"},
3381 {"RX2 MIX2", NULL, "RX2 MIX1"},
3382 {"RX2 MIX2", NULL, "RX2 MIX2 INP1"},
3383 {"RX2 MIX2", NULL, "RX2 MIX2 INP2"},
3384 {"RX3 MIX2", NULL, "RX3 MIX1"},
3385 {"RX3 MIX2", NULL, "RX3 MIX2 INP1"},
3386 {"RX3 MIX2", NULL, "RX3 MIX2 INP2"},
Bradley Rubin74a9b4a2011-06-13 15:03:43 -07003387
Bradley Rubin229c6a52011-07-12 16:18:48 -07003388 {"RX1 MIX1 INP1", "RX1", "SLIM RX1"},
3389 {"RX1 MIX1 INP1", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303390 {"RX1 MIX1 INP1", "RX3", "SLIM RX3"},
3391 {"RX1 MIX1 INP1", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003392 {"RX1 MIX1 INP1", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003393 {"RX1 MIX1 INP1", "RX6", "SLIM RX6"},
3394 {"RX1 MIX1 INP1", "RX7", "SLIM RX7"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003395 {"RX1 MIX1 INP1", "IIR1", "IIR1"},
3396 {"RX1 MIX1 INP2", "RX1", "SLIM RX1"},
3397 {"RX1 MIX1 INP2", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303398 {"RX1 MIX1 INP2", "RX3", "SLIM RX3"},
3399 {"RX1 MIX1 INP2", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003400 {"RX1 MIX1 INP2", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003401 {"RX1 MIX1 INP2", "RX6", "SLIM RX6"},
3402 {"RX1 MIX1 INP2", "RX7", "SLIM RX7"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003403 {"RX1 MIX1 INP2", "IIR1", "IIR1"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003404 {"RX1 MIX1 INP3", "RX1", "SLIM RX1"},
3405 {"RX1 MIX1 INP3", "RX2", "SLIM RX2"},
3406 {"RX1 MIX1 INP3", "RX3", "SLIM RX3"},
3407 {"RX1 MIX1 INP3", "RX4", "SLIM RX4"},
3408 {"RX1 MIX1 INP3", "RX5", "SLIM RX5"},
3409 {"RX1 MIX1 INP3", "RX6", "SLIM RX6"},
3410 {"RX1 MIX1 INP3", "RX7", "SLIM RX7"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003411 {"RX2 MIX1 INP1", "RX1", "SLIM RX1"},
3412 {"RX2 MIX1 INP1", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303413 {"RX2 MIX1 INP1", "RX3", "SLIM RX3"},
3414 {"RX2 MIX1 INP1", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003415 {"RX2 MIX1 INP1", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003416 {"RX2 MIX1 INP1", "RX6", "SLIM RX6"},
3417 {"RX2 MIX1 INP1", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003418 {"RX2 MIX1 INP1", "IIR1", "IIR1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003419 {"RX2 MIX1 INP2", "RX1", "SLIM RX1"},
3420 {"RX2 MIX1 INP2", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303421 {"RX2 MIX1 INP2", "RX3", "SLIM RX3"},
3422 {"RX2 MIX1 INP2", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003423 {"RX2 MIX1 INP2", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003424 {"RX2 MIX1 INP2", "RX6", "SLIM RX6"},
3425 {"RX2 MIX1 INP2", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003426 {"RX2 MIX1 INP2", "IIR1", "IIR1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003427 {"RX3 MIX1 INP1", "RX1", "SLIM RX1"},
3428 {"RX3 MIX1 INP1", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303429 {"RX3 MIX1 INP1", "RX3", "SLIM RX3"},
3430 {"RX3 MIX1 INP1", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003431 {"RX3 MIX1 INP1", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003432 {"RX3 MIX1 INP1", "RX6", "SLIM RX6"},
3433 {"RX3 MIX1 INP1", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003434 {"RX3 MIX1 INP1", "IIR1", "IIR1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003435 {"RX3 MIX1 INP2", "RX1", "SLIM RX1"},
3436 {"RX3 MIX1 INP2", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303437 {"RX3 MIX1 INP2", "RX3", "SLIM RX3"},
3438 {"RX3 MIX1 INP2", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003439 {"RX3 MIX1 INP2", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003440 {"RX3 MIX1 INP2", "RX6", "SLIM RX6"},
3441 {"RX3 MIX1 INP2", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003442 {"RX3 MIX1 INP2", "IIR1", "IIR1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003443 {"RX4 MIX1 INP1", "RX1", "SLIM RX1"},
3444 {"RX4 MIX1 INP1", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303445 {"RX4 MIX1 INP1", "RX3", "SLIM RX3"},
3446 {"RX4 MIX1 INP1", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003447 {"RX4 MIX1 INP1", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003448 {"RX4 MIX1 INP1", "RX6", "SLIM RX6"},
3449 {"RX4 MIX1 INP1", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003450 {"RX4 MIX1 INP1", "IIR1", "IIR1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003451 {"RX4 MIX1 INP2", "RX1", "SLIM RX1"},
3452 {"RX4 MIX1 INP2", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303453 {"RX4 MIX1 INP2", "RX3", "SLIM RX3"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003454 {"RX4 MIX1 INP2", "RX5", "SLIM RX5"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303455 {"RX4 MIX1 INP2", "RX4", "SLIM RX4"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003456 {"RX4 MIX1 INP2", "RX6", "SLIM RX6"},
3457 {"RX4 MIX1 INP2", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003458 {"RX4 MIX1 INP2", "IIR1", "IIR1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003459 {"RX5 MIX1 INP1", "RX1", "SLIM RX1"},
3460 {"RX5 MIX1 INP1", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303461 {"RX5 MIX1 INP1", "RX3", "SLIM RX3"},
3462 {"RX5 MIX1 INP1", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003463 {"RX5 MIX1 INP1", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003464 {"RX5 MIX1 INP1", "RX6", "SLIM RX6"},
3465 {"RX5 MIX1 INP1", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003466 {"RX5 MIX1 INP1", "IIR1", "IIR1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003467 {"RX5 MIX1 INP2", "RX1", "SLIM RX1"},
3468 {"RX5 MIX1 INP2", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303469 {"RX5 MIX1 INP2", "RX3", "SLIM RX3"},
3470 {"RX5 MIX1 INP2", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003471 {"RX5 MIX1 INP2", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003472 {"RX5 MIX1 INP2", "RX6", "SLIM RX6"},
3473 {"RX5 MIX1 INP2", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003474 {"RX5 MIX1 INP2", "IIR1", "IIR1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003475 {"RX6 MIX1 INP1", "RX1", "SLIM RX1"},
3476 {"RX6 MIX1 INP1", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303477 {"RX6 MIX1 INP1", "RX3", "SLIM RX3"},
3478 {"RX6 MIX1 INP1", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003479 {"RX6 MIX1 INP1", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003480 {"RX6 MIX1 INP1", "RX6", "SLIM RX6"},
3481 {"RX6 MIX1 INP1", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003482 {"RX6 MIX1 INP1", "IIR1", "IIR1"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003483 {"RX6 MIX1 INP2", "RX1", "SLIM RX1"},
3484 {"RX6 MIX1 INP2", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303485 {"RX6 MIX1 INP2", "RX3", "SLIM RX3"},
3486 {"RX6 MIX1 INP2", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003487 {"RX6 MIX1 INP2", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003488 {"RX6 MIX1 INP2", "RX6", "SLIM RX6"},
3489 {"RX6 MIX1 INP2", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003490 {"RX6 MIX1 INP2", "IIR1", "IIR1"},
Bhalchandra Gajare0a8ad172011-08-12 13:32:22 -07003491 {"RX7 MIX1 INP1", "RX1", "SLIM RX1"},
3492 {"RX7 MIX1 INP1", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303493 {"RX7 MIX1 INP1", "RX3", "SLIM RX3"},
3494 {"RX7 MIX1 INP1", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003495 {"RX7 MIX1 INP1", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003496 {"RX7 MIX1 INP1", "RX6", "SLIM RX6"},
3497 {"RX7 MIX1 INP1", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003498 {"RX7 MIX1 INP1", "IIR1", "IIR1"},
Bhalchandra Gajare0a8ad172011-08-12 13:32:22 -07003499 {"RX7 MIX1 INP2", "RX1", "SLIM RX1"},
3500 {"RX7 MIX1 INP2", "RX2", "SLIM RX2"},
Santosh Mardie15e2302011-11-15 10:39:23 +05303501 {"RX7 MIX1 INP2", "RX3", "SLIM RX3"},
3502 {"RX7 MIX1 INP2", "RX4", "SLIM RX4"},
Kiran Kandia9fffe92012-05-20 23:42:30 -07003503 {"RX7 MIX1 INP2", "RX5", "SLIM RX5"},
Neema Shettyd3a89262012-02-16 10:23:50 -08003504 {"RX7 MIX1 INP2", "RX6", "SLIM RX6"},
3505 {"RX7 MIX1 INP2", "RX7", "SLIM RX7"},
Patrick Lai16261e82011-09-30 13:25:52 -07003506 {"RX7 MIX1 INP2", "IIR1", "IIR1"},
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08003507 {"RX1 MIX2 INP1", "IIR1", "IIR1"},
3508 {"RX1 MIX2 INP2", "IIR1", "IIR1"},
3509 {"RX2 MIX2 INP1", "IIR1", "IIR1"},
3510 {"RX2 MIX2 INP2", "IIR1", "IIR1"},
3511 {"RX3 MIX2 INP1", "IIR1", "IIR1"},
3512 {"RX3 MIX2 INP2", "IIR1", "IIR1"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003513
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003514 /* Decimator Inputs */
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003515 {"DEC1 MUX", "DMIC1", "DMIC1"},
Kiran Kandicf45f6a2011-07-17 21:10:19 -07003516 {"DEC1 MUX", "ADC6", "ADC6"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003517 {"DEC1 MUX", NULL, "CDC_CONN"},
Kiran Kandicf45f6a2011-07-17 21:10:19 -07003518 {"DEC2 MUX", "DMIC2", "DMIC2"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003519 {"DEC2 MUX", "ADC5", "ADC5"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003520 {"DEC2 MUX", NULL, "CDC_CONN"},
Kiran Kandicf45f6a2011-07-17 21:10:19 -07003521 {"DEC3 MUX", "DMIC3", "DMIC3"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003522 {"DEC3 MUX", "ADC4", "ADC4"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003523 {"DEC3 MUX", NULL, "CDC_CONN"},
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07003524 {"DEC4 MUX", "DMIC4", "DMIC4"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003525 {"DEC4 MUX", "ADC3", "ADC3"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003526 {"DEC4 MUX", NULL, "CDC_CONN"},
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07003527 {"DEC5 MUX", "DMIC5", "DMIC5"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003528 {"DEC5 MUX", "ADC2", "ADC2"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003529 {"DEC5 MUX", NULL, "CDC_CONN"},
Bhalchandra Gajare7cf018e2011-08-11 18:58:32 -07003530 {"DEC6 MUX", "DMIC6", "DMIC6"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003531 {"DEC6 MUX", "ADC1", "ADC1"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003532 {"DEC6 MUX", NULL, "CDC_CONN"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003533 {"DEC7 MUX", "DMIC1", "DMIC1"},
Kiran Kandi1e6371d2012-03-29 11:48:57 -07003534 {"DEC7 MUX", "DMIC6", "DMIC6"},
3535 {"DEC7 MUX", "ADC1", "ADC1"},
Kiran Kandicf45f6a2011-07-17 21:10:19 -07003536 {"DEC7 MUX", "ADC6", "ADC6"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003537 {"DEC7 MUX", NULL, "CDC_CONN"},
Kiran Kandi1e6371d2012-03-29 11:48:57 -07003538 {"DEC8 MUX", "DMIC2", "DMIC2"},
3539 {"DEC8 MUX", "DMIC5", "DMIC5"},
3540 {"DEC8 MUX", "ADC2", "ADC2"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003541 {"DEC8 MUX", "ADC5", "ADC5"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003542 {"DEC8 MUX", NULL, "CDC_CONN"},
Kiran Kandi1e6371d2012-03-29 11:48:57 -07003543 {"DEC9 MUX", "DMIC4", "DMIC4"},
3544 {"DEC9 MUX", "DMIC5", "DMIC5"},
3545 {"DEC9 MUX", "ADC2", "ADC2"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003546 {"DEC9 MUX", "ADC3", "ADC3"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003547 {"DEC9 MUX", NULL, "CDC_CONN"},
Kiran Kandi1e6371d2012-03-29 11:48:57 -07003548 {"DEC10 MUX", "DMIC3", "DMIC3"},
3549 {"DEC10 MUX", "DMIC6", "DMIC6"},
3550 {"DEC10 MUX", "ADC1", "ADC1"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003551 {"DEC10 MUX", "ADC4", "ADC4"},
Bradley Rubine1d08622011-07-20 18:01:35 -07003552 {"DEC10 MUX", NULL, "CDC_CONN"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003553
3554 /* ADC Connections */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003555 {"ADC1", NULL, "AMIC1"},
3556 {"ADC2", NULL, "AMIC2"},
Bhalchandra Gajarecc6ffa02011-07-14 18:35:41 -07003557 {"ADC3", NULL, "AMIC3"},
3558 {"ADC4", NULL, "AMIC4"},
3559 {"ADC5", NULL, "AMIC5"},
3560 {"ADC6", NULL, "AMIC6"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003561
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08003562 /* AUX PGA Connections */
3563 {"HPHL_PA_MIXER", "AUX_PGA_L Switch", "AUX_PGA_Left"},
3564 {"HPHL_PA_MIXER", "AUX_PGA_R Switch", "AUX_PGA_Right"},
3565 {"HPHL_PA_MIXER", "AUX_PGA_L_INV Switch", "AUX_PGA_Left"},
3566 {"HPHL_PA_MIXER", "AUX_PGA_R_INV Switch", "AUX_PGA_Right"},
3567 {"HPHR_PA_MIXER", "AUX_PGA_L Switch", "AUX_PGA_Left"},
3568 {"HPHR_PA_MIXER", "AUX_PGA_R Switch", "AUX_PGA_Right"},
3569 {"HPHR_PA_MIXER", "AUX_PGA_L_INV Switch", "AUX_PGA_Left"},
3570 {"HPHR_PA_MIXER", "AUX_PGA_R_INV Switch", "AUX_PGA_Right"},
3571 {"LINEOUT1_PA_MIXER", "AUX_PGA_L Switch", "AUX_PGA_Left"},
3572 {"LINEOUT1_PA_MIXER", "AUX_PGA_R Switch", "AUX_PGA_Right"},
3573 {"LINEOUT1_PA_MIXER", "AUX_PGA_L_INV Switch", "AUX_PGA_Left"},
3574 {"LINEOUT1_PA_MIXER", "AUX_PGA_R_INV Switch", "AUX_PGA_Right"},
3575 {"LINEOUT2_PA_MIXER", "AUX_PGA_L Switch", "AUX_PGA_Left"},
3576 {"LINEOUT2_PA_MIXER", "AUX_PGA_R Switch", "AUX_PGA_Right"},
3577 {"LINEOUT2_PA_MIXER", "AUX_PGA_L_INV Switch", "AUX_PGA_Left"},
3578 {"LINEOUT2_PA_MIXER", "AUX_PGA_R_INV Switch", "AUX_PGA_Right"},
3579 {"LINEOUT3_PA_MIXER", "AUX_PGA_L Switch", "AUX_PGA_Left"},
3580 {"LINEOUT3_PA_MIXER", "AUX_PGA_R Switch", "AUX_PGA_Right"},
3581 {"LINEOUT3_PA_MIXER", "AUX_PGA_L_INV Switch", "AUX_PGA_Left"},
3582 {"LINEOUT3_PA_MIXER", "AUX_PGA_R_INV Switch", "AUX_PGA_Right"},
3583 {"LINEOUT4_PA_MIXER", "AUX_PGA_L Switch", "AUX_PGA_Left"},
3584 {"LINEOUT4_PA_MIXER", "AUX_PGA_R Switch", "AUX_PGA_Right"},
3585 {"LINEOUT4_PA_MIXER", "AUX_PGA_L_INV Switch", "AUX_PGA_Left"},
3586 {"LINEOUT4_PA_MIXER", "AUX_PGA_R_INV Switch", "AUX_PGA_Right"},
3587 {"LINEOUT5_PA_MIXER", "AUX_PGA_L Switch", "AUX_PGA_Left"},
3588 {"LINEOUT5_PA_MIXER", "AUX_PGA_R Switch", "AUX_PGA_Right"},
3589 {"LINEOUT5_PA_MIXER", "AUX_PGA_L_INV Switch", "AUX_PGA_Left"},
3590 {"LINEOUT5_PA_MIXER", "AUX_PGA_R_INV Switch", "AUX_PGA_Right"},
3591 {"EAR_PA_MIXER", "AUX_PGA_L Switch", "AUX_PGA_Left"},
3592 {"EAR_PA_MIXER", "AUX_PGA_R Switch", "AUX_PGA_Right"},
3593 {"EAR_PA_MIXER", "AUX_PGA_L_INV Switch", "AUX_PGA_Left"},
3594 {"EAR_PA_MIXER", "AUX_PGA_R_INV Switch", "AUX_PGA_Right"},
3595 {"AUX_PGA_Left", NULL, "AMIC5"},
3596 {"AUX_PGA_Right", NULL, "AMIC6"},
3597
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003598 {"IIR1", NULL, "IIR1 INP1 MUX"},
Patrick Lai16261e82011-09-30 13:25:52 -07003599 {"IIR1 INP1 MUX", "DEC1", "DEC1 MUX"},
3600 {"IIR1 INP1 MUX", "DEC2", "DEC2 MUX"},
3601 {"IIR1 INP1 MUX", "DEC3", "DEC3 MUX"},
3602 {"IIR1 INP1 MUX", "DEC4", "DEC4 MUX"},
3603 {"IIR1 INP1 MUX", "DEC5", "DEC5 MUX"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003604 {"IIR1 INP1 MUX", "DEC6", "DEC6 MUX"},
Patrick Lai16261e82011-09-30 13:25:52 -07003605 {"IIR1 INP1 MUX", "DEC7", "DEC7 MUX"},
3606 {"IIR1 INP1 MUX", "DEC8", "DEC8 MUX"},
3607 {"IIR1 INP1 MUX", "DEC9", "DEC9 MUX"},
3608 {"IIR1 INP1 MUX", "DEC10", "DEC10 MUX"},
Bradley Rubin229c6a52011-07-12 16:18:48 -07003609
3610 {"MIC BIAS1 Internal1", NULL, "LDO_H"},
3611 {"MIC BIAS1 Internal2", NULL, "LDO_H"},
3612 {"MIC BIAS1 External", NULL, "LDO_H"},
3613 {"MIC BIAS2 Internal1", NULL, "LDO_H"},
3614 {"MIC BIAS2 Internal2", NULL, "LDO_H"},
3615 {"MIC BIAS2 Internal3", NULL, "LDO_H"},
3616 {"MIC BIAS2 External", NULL, "LDO_H"},
3617 {"MIC BIAS3 Internal1", NULL, "LDO_H"},
3618 {"MIC BIAS3 Internal2", NULL, "LDO_H"},
3619 {"MIC BIAS3 External", NULL, "LDO_H"},
3620 {"MIC BIAS4 External", NULL, "LDO_H"},
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003621};
3622
Kiran Kandi8b3a8302011-09-27 16:13:28 -07003623static const struct snd_soc_dapm_route tabla_1_x_lineout_2_to_4_map[] = {
3624
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08003625 {"RX4 DSM MUX", "DSM_INV", "RX3 MIX2"},
Kiran Kandi8b3a8302011-09-27 16:13:28 -07003626 {"RX4 DSM MUX", "CIC_OUT", "RX4 MIX1"},
3627
3628 {"LINEOUT2 DAC", NULL, "RX4 DSM MUX"},
3629
3630 {"LINEOUT3 DAC", NULL, "RX5 MIX1"},
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08003631 {"LINEOUT3 DAC GROUND", "Switch", "RX3 MIX2"},
Kiran Kandi8b3a8302011-09-27 16:13:28 -07003632 {"LINEOUT3 DAC", NULL, "LINEOUT3 DAC GROUND"},
3633
3634 {"RX6 DSM MUX", "DSM_INV", "RX5 MIX1"},
3635 {"RX6 DSM MUX", "CIC_OUT", "RX6 MIX1"},
3636
3637 {"LINEOUT4 DAC", NULL, "RX6 DSM MUX"},
3638 {"LINEOUT4 DAC GROUND", "Switch", "RX4 DSM MUX"},
3639 {"LINEOUT4 DAC", NULL, "LINEOUT4 DAC GROUND"},
3640};
3641
Kiran Kandi7a9fd902011-11-14 13:51:45 -08003642
3643static const struct snd_soc_dapm_route tabla_2_x_lineout_2_to_4_map[] = {
3644
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08003645 {"RX4 DSM MUX", "DSM_INV", "RX3 MIX2"},
Kiran Kandi7a9fd902011-11-14 13:51:45 -08003646 {"RX4 DSM MUX", "CIC_OUT", "RX4 MIX1"},
3647
3648 {"LINEOUT3 DAC", NULL, "RX4 DSM MUX"},
3649
3650 {"LINEOUT2 DAC", NULL, "RX5 MIX1"},
3651
3652 {"RX6 DSM MUX", "DSM_INV", "RX5 MIX1"},
3653 {"RX6 DSM MUX", "CIC_OUT", "RX6 MIX1"},
3654
3655 {"LINEOUT4 DAC", NULL, "RX6 DSM MUX"},
3656};
3657
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003658static int tabla_readable(struct snd_soc_codec *ssc, unsigned int reg)
3659{
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08003660 int i;
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05303661 struct wcd9xxx *tabla_core = dev_get_drvdata(ssc->dev->parent);
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08003662
3663 if (TABLA_IS_1_X(tabla_core->version)) {
3664 for (i = 0; i < ARRAY_SIZE(tabla_1_reg_readable); i++) {
3665 if (tabla_1_reg_readable[i] == reg)
3666 return 1;
3667 }
3668 } else {
3669 for (i = 0; i < ARRAY_SIZE(tabla_2_reg_readable); i++) {
3670 if (tabla_2_reg_readable[i] == reg)
3671 return 1;
3672 }
3673 }
3674
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003675 return tabla_reg_readable[reg];
3676}
Kuirong Wange9c8a222012-03-28 16:24:09 -07003677static bool tabla_is_digital_gain_register(unsigned int reg)
3678{
3679 bool rtn = false;
3680 switch (reg) {
3681 case TABLA_A_CDC_RX1_VOL_CTL_B2_CTL:
3682 case TABLA_A_CDC_RX2_VOL_CTL_B2_CTL:
3683 case TABLA_A_CDC_RX3_VOL_CTL_B2_CTL:
3684 case TABLA_A_CDC_RX4_VOL_CTL_B2_CTL:
3685 case TABLA_A_CDC_RX5_VOL_CTL_B2_CTL:
3686 case TABLA_A_CDC_RX6_VOL_CTL_B2_CTL:
3687 case TABLA_A_CDC_RX7_VOL_CTL_B2_CTL:
3688 case TABLA_A_CDC_TX1_VOL_CTL_GAIN:
3689 case TABLA_A_CDC_TX2_VOL_CTL_GAIN:
3690 case TABLA_A_CDC_TX3_VOL_CTL_GAIN:
3691 case TABLA_A_CDC_TX4_VOL_CTL_GAIN:
3692 case TABLA_A_CDC_TX5_VOL_CTL_GAIN:
3693 case TABLA_A_CDC_TX6_VOL_CTL_GAIN:
3694 case TABLA_A_CDC_TX7_VOL_CTL_GAIN:
3695 case TABLA_A_CDC_TX8_VOL_CTL_GAIN:
3696 case TABLA_A_CDC_TX9_VOL_CTL_GAIN:
3697 case TABLA_A_CDC_TX10_VOL_CTL_GAIN:
3698 rtn = true;
3699 break;
3700 default:
3701 break;
3702 }
3703 return rtn;
3704}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003705static int tabla_volatile(struct snd_soc_codec *ssc, unsigned int reg)
3706{
3707 /* Registers lower than 0x100 are top level registers which can be
3708 * written by the Tabla core driver.
3709 */
3710
3711 if ((reg >= TABLA_A_CDC_MBHC_EN_CTL) || (reg < 0x100))
3712 return 1;
3713
Ben Romberger1f045a72011-11-04 10:14:57 -07003714 /* IIR Coeff registers are not cacheable */
3715 if ((reg >= TABLA_A_CDC_IIR1_COEF_B1_CTL) &&
3716 (reg <= TABLA_A_CDC_IIR2_COEF_B5_CTL))
3717 return 1;
3718
Kuirong Wange9c8a222012-03-28 16:24:09 -07003719 /* Digital gain register is not cacheable so we have to write
3720 * the setting even it is the same
3721 */
3722 if (tabla_is_digital_gain_register(reg))
3723 return 1;
3724
Joonwoo Parkab2c5872012-05-03 15:16:02 -07003725 /* HPH status registers */
3726 if (reg == TABLA_A_RX_HPH_L_STATUS || reg == TABLA_A_RX_HPH_R_STATUS)
3727 return 1;
3728
Kuirong Wangbc5ac042012-06-26 15:35:22 -07003729 if (reg == TABLA_A_CDC_COMP1_SHUT_DOWN_STATUS ||
3730 reg == TABLA_A_CDC_COMP2_SHUT_DOWN_STATUS)
3731 return 1;
3732
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003733 return 0;
3734}
3735
3736#define TABLA_FORMATS (SNDRV_PCM_FMTBIT_S16_LE)
3737static int tabla_write(struct snd_soc_codec *codec, unsigned int reg,
3738 unsigned int value)
3739{
3740 int ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003741 BUG_ON(reg > TABLA_MAX_REGISTER);
3742
3743 if (!tabla_volatile(codec, reg)) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003744 ret = snd_soc_cache_write(codec, reg, value);
3745 if (ret != 0)
3746 dev_err(codec->dev, "Cache write to %x failed: %d\n",
3747 reg, ret);
3748 }
3749
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05303750 return wcd9xxx_reg_write(codec->control_data, reg, value);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003751}
3752static unsigned int tabla_read(struct snd_soc_codec *codec,
3753 unsigned int reg)
3754{
3755 unsigned int val;
3756 int ret;
3757
3758 BUG_ON(reg > TABLA_MAX_REGISTER);
3759
3760 if (!tabla_volatile(codec, reg) && tabla_readable(codec, reg) &&
3761 reg < codec->driver->reg_cache_size) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003762 ret = snd_soc_cache_read(codec, reg, &val);
3763 if (ret >= 0) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003764 return val;
3765 } else
3766 dev_err(codec->dev, "Cache read from %x failed: %d\n",
3767 reg, ret);
3768 }
3769
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05303770 val = wcd9xxx_reg_read(codec->control_data, reg);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003771 return val;
3772}
3773
Joonwoo Parkcf473b42012-03-29 19:48:16 -07003774static s16 tabla_get_current_v_ins(struct tabla_priv *tabla, bool hu)
3775{
3776 s16 v_ins;
3777 if ((tabla->mbhc_data.micb_mv != VDDIO_MICBIAS_MV) &&
3778 tabla->mbhc_micbias_switched)
3779 v_ins = hu ? (s16)tabla->mbhc_data.adj_v_ins_hu :
3780 (s16)tabla->mbhc_data.adj_v_ins_h;
3781 else
3782 v_ins = hu ? (s16)tabla->mbhc_data.v_ins_hu :
3783 (s16)tabla->mbhc_data.v_ins_h;
3784 return v_ins;
3785}
3786
3787static s16 tabla_get_current_v_hs_max(struct tabla_priv *tabla)
3788{
3789 s16 v_hs_max;
3790 struct tabla_mbhc_plug_type_cfg *plug_type;
3791
3792 plug_type = TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla->mbhc_cfg.calibration);
3793 if ((tabla->mbhc_data.micb_mv != VDDIO_MICBIAS_MV) &&
3794 tabla->mbhc_micbias_switched)
3795 v_hs_max = tabla->mbhc_data.adj_v_hs_max;
3796 else
3797 v_hs_max = plug_type->v_hs_max;
3798 return v_hs_max;
3799}
3800
Joonwoo Parkdd9d2962012-07-23 19:24:20 -07003801static void tabla_codec_calibrate_rel(struct snd_soc_codec *codec)
Bradley Rubincb1e2732011-06-23 16:49:20 -07003802{
Joonwoo Park0976d012011-12-22 11:48:18 -08003803 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Bradley Rubincb1e2732011-06-23 16:49:20 -07003804
Joonwoo Park0976d012011-12-22 11:48:18 -08003805 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B3_CTL,
3806 tabla->mbhc_data.v_b1_hu & 0xFF);
3807 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B4_CTL,
3808 (tabla->mbhc_data.v_b1_hu >> 8) & 0xFF);
3809
3810 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B5_CTL,
3811 tabla->mbhc_data.v_b1_h & 0xFF);
3812 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B6_CTL,
3813 (tabla->mbhc_data.v_b1_h >> 8) & 0xFF);
3814
3815 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B9_CTL,
3816 tabla->mbhc_data.v_brh & 0xFF);
3817 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B10_CTL,
3818 (tabla->mbhc_data.v_brh >> 8) & 0xFF);
3819
3820 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B11_CTL,
3821 tabla->mbhc_data.v_brl & 0xFF);
3822 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B12_CTL,
3823 (tabla->mbhc_data.v_brl >> 8) & 0xFF);
Joonwoo Parkdd9d2962012-07-23 19:24:20 -07003824}
3825
3826static void tabla_codec_calibrate_hs_polling(struct snd_soc_codec *codec)
3827{
3828 u8 *n_ready, *n_cic;
3829 struct tabla_mbhc_btn_detect_cfg *btn_det;
3830 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
3831 const s16 v_ins_hu = tabla_get_current_v_ins(tabla, true);
3832
3833 btn_det = TABLA_MBHC_CAL_BTN_DET_PTR(tabla->mbhc_cfg.calibration);
3834
3835 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B1_CTL,
3836 v_ins_hu & 0xFF);
3837 snd_soc_write(codec, TABLA_A_CDC_MBHC_VOLT_B2_CTL,
3838 (v_ins_hu >> 8) & 0xFF);
3839
3840 tabla_codec_calibrate_rel(codec);
Joonwoo Park0976d012011-12-22 11:48:18 -08003841
Joonwoo Parkc0672392012-01-11 11:03:14 -08003842 n_ready = tabla_mbhc_cal_btn_det_mp(btn_det, TABLA_BTN_DET_N_READY);
Joonwoo Park0976d012011-12-22 11:48:18 -08003843 snd_soc_write(codec, TABLA_A_CDC_MBHC_TIMER_B1_CTL,
Joonwoo Parkc0672392012-01-11 11:03:14 -08003844 n_ready[tabla_codec_mclk_index(tabla)]);
Joonwoo Park0976d012011-12-22 11:48:18 -08003845 snd_soc_write(codec, TABLA_A_CDC_MBHC_TIMER_B2_CTL,
3846 tabla->mbhc_data.npoll);
3847 snd_soc_write(codec, TABLA_A_CDC_MBHC_TIMER_B3_CTL,
3848 tabla->mbhc_data.nbounce_wait);
Joonwoo Park0976d012011-12-22 11:48:18 -08003849 n_cic = tabla_mbhc_cal_btn_det_mp(btn_det, TABLA_BTN_DET_N_CIC);
Joonwoo Park107edf02012-01-11 11:42:24 -08003850 snd_soc_write(codec, TABLA_A_CDC_MBHC_TIMER_B6_CTL,
3851 n_cic[tabla_codec_mclk_index(tabla)]);
Bradley Rubincb1e2732011-06-23 16:49:20 -07003852}
3853
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003854static int tabla_startup(struct snd_pcm_substream *substream,
3855 struct snd_soc_dai *dai)
3856{
Kuirong Wanga545e722012-02-06 19:12:54 -08003857 struct wcd9xxx *tabla_core = dev_get_drvdata(dai->codec->dev->parent);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003858 pr_debug("%s(): substream = %s stream = %d\n" , __func__,
3859 substream->name, substream->stream);
Kuirong Wanga545e722012-02-06 19:12:54 -08003860 if ((tabla_core != NULL) &&
3861 (tabla_core->dev != NULL) &&
3862 (tabla_core->dev->parent != NULL))
3863 pm_runtime_get_sync(tabla_core->dev->parent);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003864
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003865 return 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003866}
3867
Ajay Dudani506f81c2012-08-23 16:01:50 -07003868static void tabla_shutdown(struct snd_pcm_substream *substream,
3869 struct snd_soc_dai *dai)
3870{
3871 struct wcd9xxx *tabla_core = dev_get_drvdata(dai->codec->dev->parent);
3872 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(dai->codec);
3873 u32 active = 0;
3874
3875 pr_debug("%s(): substream = %s stream = %d\n" , __func__,
3876 substream->name, substream->stream);
3877 if (tabla->intf_type != WCD9XXX_INTERFACE_TYPE_SLIMBUS)
3878 return;
3879
3880 if (dai->id <= NUM_CODEC_DAIS) {
3881 if (tabla->dai[dai->id-1].ch_mask) {
3882 active = 1;
3883 pr_debug("%s(): Codec DAI: chmask[%d] = 0x%x\n",
3884 __func__, dai->id-1, tabla->dai[dai->id-1].ch_mask);
3885 }
3886 }
3887
3888 if ((tabla_core != NULL) &&
3889 (tabla_core->dev != NULL) &&
3890 (tabla_core->dev->parent != NULL) &&
3891 (active == 0)) {
3892 pm_runtime_mark_last_busy(tabla_core->dev->parent);
3893 pm_runtime_put(tabla_core->dev->parent);
3894 }
3895}
3896
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07003897int tabla_mclk_enable(struct snd_soc_codec *codec, int mclk_enable, bool dapm)
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003898{
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003899 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
3900
Joonwoo Parkcf473b42012-03-29 19:48:16 -07003901 pr_debug("%s: mclk_enable = %u, dapm = %d\n", __func__, mclk_enable,
3902 dapm);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07003903 if (dapm)
3904 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003905 if (mclk_enable) {
3906 tabla->mclk_enabled = true;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003907
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07003908 if (tabla->mbhc_polling_active) {
Bradley Rubincb1e2732011-06-23 16:49:20 -07003909 tabla_codec_pause_hs_polling(codec);
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07003910 tabla_codec_disable_clock_block(codec);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003911 tabla_codec_enable_bandgap(codec,
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07003912 TABLA_BANDGAP_AUDIO_MODE);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003913 tabla_codec_enable_clock_block(codec, 0);
Bradley Rubincb1e2732011-06-23 16:49:20 -07003914 tabla_codec_calibrate_hs_polling(codec);
3915 tabla_codec_start_hs_polling(codec);
Asish Bhattacharya486745a2012-01-20 06:41:53 +05303916 } else {
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07003917 tabla_codec_disable_clock_block(codec);
Asish Bhattacharya486745a2012-01-20 06:41:53 +05303918 tabla_codec_enable_bandgap(codec,
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07003919 TABLA_BANDGAP_AUDIO_MODE);
Asish Bhattacharya486745a2012-01-20 06:41:53 +05303920 tabla_codec_enable_clock_block(codec, 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003921 }
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003922 } else {
3923
3924 if (!tabla->mclk_enabled) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07003925 if (dapm)
3926 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003927 pr_err("Error, MCLK already diabled\n");
3928 return -EINVAL;
3929 }
3930 tabla->mclk_enabled = false;
3931
3932 if (tabla->mbhc_polling_active) {
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07003933 tabla_codec_pause_hs_polling(codec);
3934 tabla_codec_disable_clock_block(codec);
3935 tabla_codec_enable_bandgap(codec,
3936 TABLA_BANDGAP_MBHC_MODE);
3937 tabla_enable_rx_bias(codec, 1);
3938 tabla_codec_enable_clock_block(codec, 1);
3939 tabla_codec_calibrate_hs_polling(codec);
3940 tabla_codec_start_hs_polling(codec);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003941 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN1,
3942 0x05, 0x01);
Asish Bhattacharya486745a2012-01-20 06:41:53 +05303943 } else {
3944 tabla_codec_disable_clock_block(codec);
3945 tabla_codec_enable_bandgap(codec,
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07003946 TABLA_BANDGAP_OFF);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003947 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003948 }
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07003949 if (dapm)
3950 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07003951 return 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003952}
3953
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003954static int tabla_set_dai_sysclk(struct snd_soc_dai *dai,
3955 int clk_id, unsigned int freq, int dir)
3956{
3957 pr_debug("%s\n", __func__);
3958 return 0;
3959}
3960
3961static int tabla_set_dai_fmt(struct snd_soc_dai *dai, unsigned int fmt)
3962{
Santosh Mardie15e2302011-11-15 10:39:23 +05303963 u8 val = 0;
3964 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(dai->codec);
3965
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003966 pr_debug("%s\n", __func__);
Santosh Mardie15e2302011-11-15 10:39:23 +05303967 switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
3968 case SND_SOC_DAIFMT_CBS_CFS:
3969 /* CPU is master */
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05303970 if (tabla->intf_type == WCD9XXX_INTERFACE_TYPE_I2C) {
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08003971 if (dai->id == AIF1_CAP)
Santosh Mardie15e2302011-11-15 10:39:23 +05303972 snd_soc_update_bits(dai->codec,
3973 TABLA_A_CDC_CLK_TX_I2S_CTL,
3974 TABLA_I2S_MASTER_MODE_MASK, 0);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08003975 else if (dai->id == AIF1_PB)
Santosh Mardie15e2302011-11-15 10:39:23 +05303976 snd_soc_update_bits(dai->codec,
3977 TABLA_A_CDC_CLK_RX_I2S_CTL,
3978 TABLA_I2S_MASTER_MODE_MASK, 0);
3979 }
3980 break;
3981 case SND_SOC_DAIFMT_CBM_CFM:
3982 /* CPU is slave */
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05303983 if (tabla->intf_type == WCD9XXX_INTERFACE_TYPE_I2C) {
Santosh Mardie15e2302011-11-15 10:39:23 +05303984 val = TABLA_I2S_MASTER_MODE_MASK;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08003985 if (dai->id == AIF1_CAP)
Santosh Mardie15e2302011-11-15 10:39:23 +05303986 snd_soc_update_bits(dai->codec,
3987 TABLA_A_CDC_CLK_TX_I2S_CTL, val, val);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08003988 else if (dai->id == AIF1_PB)
Santosh Mardie15e2302011-11-15 10:39:23 +05303989 snd_soc_update_bits(dai->codec,
3990 TABLA_A_CDC_CLK_RX_I2S_CTL, val, val);
3991 }
3992 break;
3993 default:
3994 return -EINVAL;
3995 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003996 return 0;
3997}
3998
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08003999static int tabla_set_channel_map(struct snd_soc_dai *dai,
4000 unsigned int tx_num, unsigned int *tx_slot,
4001 unsigned int rx_num, unsigned int *rx_slot)
4002
4003{
4004 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(dai->codec);
4005 u32 i = 0;
4006 if (!tx_slot && !rx_slot) {
4007 pr_err("%s: Invalid\n", __func__);
4008 return -EINVAL;
4009 }
Kiran Kandia9fffe92012-05-20 23:42:30 -07004010 pr_debug("%s(): dai_name = %s DAI-ID %x tx_ch %d rx_ch %d\n",
4011 __func__, dai->name, dai->id, tx_num, rx_num);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004012
Kiran Kandia9fffe92012-05-20 23:42:30 -07004013 if (dai->id == AIF1_PB || dai->id == AIF2_PB || dai->id == AIF3_PB) {
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004014 for (i = 0; i < rx_num; i++) {
4015 tabla->dai[dai->id - 1].ch_num[i] = rx_slot[i];
4016 tabla->dai[dai->id - 1].ch_act = 0;
4017 tabla->dai[dai->id - 1].ch_tot = rx_num;
4018 }
Neema Shetty3fb1b802012-04-27 13:53:24 -07004019 } else if (dai->id == AIF1_CAP || dai->id == AIF2_CAP ||
4020 dai->id == AIF3_CAP) {
Swaminathan Sathappan8644fb32012-08-01 12:53:03 -07004021 tabla->dai[dai->id - 1].ch_tot = tx_num;
4022 /* All channels are already active.
4023 * do not reset ch_act flag
4024 */
4025 if ((tabla->dai[dai->id - 1].ch_tot != 0)
4026 && (tabla->dai[dai->id - 1].ch_act ==
4027 tabla->dai[dai->id - 1].ch_tot)) {
4028 pr_info("%s: ch_act = %d, ch_tot = %d\n", __func__,
4029 tabla->dai[dai->id - 1].ch_act,
4030 tabla->dai[dai->id - 1].ch_tot);
4031 return 0;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004032 }
Swaminathan Sathappan8644fb32012-08-01 12:53:03 -07004033
4034 tabla->dai[dai->id - 1].ch_act = 0;
4035 for (i = 0; i < tx_num; i++)
4036 tabla->dai[dai->id - 1].ch_num[i] = tx_slot[i];
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004037 }
4038 return 0;
4039}
4040
4041static int tabla_get_channel_map(struct snd_soc_dai *dai,
4042 unsigned int *tx_num, unsigned int *tx_slot,
4043 unsigned int *rx_num, unsigned int *rx_slot)
4044
4045{
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304046 struct wcd9xxx *tabla = dev_get_drvdata(dai->codec->control_data);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004047
4048 u32 cnt = 0;
4049 u32 tx_ch[SLIM_MAX_TX_PORTS];
4050 u32 rx_ch[SLIM_MAX_RX_PORTS];
4051
4052 if (!rx_slot && !tx_slot) {
4053 pr_err("%s: Invalid\n", __func__);
4054 return -EINVAL;
4055 }
Kiran Kandia9fffe92012-05-20 23:42:30 -07004056
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004057 /* for virtual port, codec driver needs to do
4058 * housekeeping, for now should be ok
4059 */
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304060 wcd9xxx_get_channel(tabla, rx_ch, tx_ch);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004061 if (dai->id == AIF1_PB) {
4062 *rx_num = tabla_dai[dai->id - 1].playback.channels_max;
4063 while (cnt < *rx_num) {
4064 rx_slot[cnt] = rx_ch[cnt];
4065 cnt++;
4066 }
4067 } else if (dai->id == AIF1_CAP) {
4068 *tx_num = tabla_dai[dai->id - 1].capture.channels_max;
4069 while (cnt < *tx_num) {
4070 tx_slot[cnt] = tx_ch[6 + cnt];
4071 cnt++;
4072 }
Neema Shettyd3a89262012-02-16 10:23:50 -08004073 } else if (dai->id == AIF2_PB) {
4074 *rx_num = tabla_dai[dai->id - 1].playback.channels_max;
4075 while (cnt < *rx_num) {
4076 rx_slot[cnt] = rx_ch[5 + cnt];
4077 cnt++;
4078 }
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004079 } else if (dai->id == AIF2_CAP) {
4080 *tx_num = tabla_dai[dai->id - 1].capture.channels_max;
4081 tx_slot[0] = tx_ch[cnt];
4082 tx_slot[1] = tx_ch[1 + cnt];
Kiran Kandi323d7102012-04-18 19:56:14 -07004083 tx_slot[2] = tx_ch[5 + cnt];
Kiran Kandie408b842012-05-17 19:48:04 -07004084 tx_slot[3] = tx_ch[3 + cnt];
Kiran Kandia9fffe92012-05-20 23:42:30 -07004085
4086 } else if (dai->id == AIF3_PB) {
4087 *rx_num = tabla_dai[dai->id - 1].playback.channels_max;
4088 rx_slot[0] = rx_ch[3];
4089 rx_slot[1] = rx_ch[4];
4090
Neema Shetty3fb1b802012-04-27 13:53:24 -07004091 } else if (dai->id == AIF3_CAP) {
4092 *tx_num = tabla_dai[dai->id - 1].capture.channels_max;
4093 tx_slot[cnt] = tx_ch[2 + cnt];
4094 tx_slot[cnt + 1] = tx_ch[4 + cnt];
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004095 }
Kiran Kandia9fffe92012-05-20 23:42:30 -07004096 pr_debug("%s(): dai_name = %s DAI-ID %x tx_ch %d rx_ch %d\n",
4097 __func__, dai->name, dai->id, *tx_num, *rx_num);
4098
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004099
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004100 return 0;
4101}
4102
Kiran Kandi93923902012-06-20 17:00:25 -07004103
4104static struct snd_soc_dapm_widget tabla_dapm_aif_in_widgets[] = {
4105
4106 SND_SOC_DAPM_AIF_IN_E("SLIM RX1", "AIF1 Playback", 0, SND_SOC_NOPM, 1,
4107 0, tabla_codec_enable_slimrx,
4108 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4109
4110 SND_SOC_DAPM_AIF_IN_E("SLIM RX2", "AIF1 Playback", 0, SND_SOC_NOPM, 2,
4111 0, tabla_codec_enable_slimrx,
4112 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4113
4114 SND_SOC_DAPM_AIF_IN_E("SLIM RX3", "AIF1 Playback", 0, SND_SOC_NOPM, 3,
4115 0, tabla_codec_enable_slimrx,
4116 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4117
4118 SND_SOC_DAPM_AIF_IN_E("SLIM RX4", "AIF3 Playback", 0, SND_SOC_NOPM, 4,
4119 0, tabla_codec_enable_slimrx,
4120 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4121
4122 SND_SOC_DAPM_AIF_IN_E("SLIM RX5", "AIF3 Playback", 0, SND_SOC_NOPM, 5,
4123 0, tabla_codec_enable_slimrx,
4124 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4125
4126 SND_SOC_DAPM_AIF_IN_E("SLIM RX6", "AIF2 Playback", 0, SND_SOC_NOPM, 6,
4127 0, tabla_codec_enable_slimrx,
4128 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4129
4130 SND_SOC_DAPM_AIF_IN_E("SLIM RX7", "AIF2 Playback", 0, SND_SOC_NOPM, 7,
4131 0, tabla_codec_enable_slimrx,
4132 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4133};
4134
4135static struct snd_soc_dapm_widget tabla_dapm_aif_out_widgets[] = {
4136
4137 SND_SOC_DAPM_AIF_OUT_E("SLIM TX1", "AIF2 Capture", 0, SND_SOC_NOPM, 1,
4138 0, tabla_codec_enable_slimtx,
4139 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4140
4141 SND_SOC_DAPM_AIF_OUT_E("SLIM TX2", "AIF2 Capture", 0, SND_SOC_NOPM, 2,
4142 0, tabla_codec_enable_slimtx,
4143 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4144
4145 SND_SOC_DAPM_AIF_OUT_E("SLIM TX3", "AIF3 Capture", 0, SND_SOC_NOPM, 3,
4146 0, tabla_codec_enable_slimtx,
4147 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4148
4149 SND_SOC_DAPM_AIF_OUT_E("SLIM TX4", "AIF2 Capture", 0, SND_SOC_NOPM, 4,
4150 0, tabla_codec_enable_slimtx,
4151 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4152
4153 SND_SOC_DAPM_AIF_OUT_E("SLIM TX5", "AIF3 Capture", 0, SND_SOC_NOPM, 5,
4154 0, tabla_codec_enable_slimtx,
4155 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4156
4157 SND_SOC_DAPM_AIF_OUT_E("SLIM TX6", "AIF2 Capture", 0, SND_SOC_NOPM, 6,
4158 0, tabla_codec_enable_slimtx,
4159 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4160
4161 SND_SOC_DAPM_AIF_OUT_E("SLIM TX7", "AIF1 Capture", 0, SND_SOC_NOPM, 7,
4162 0, tabla_codec_enable_slimtx,
4163 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4164
4165 SND_SOC_DAPM_AIF_OUT_E("SLIM TX8", "AIF1 Capture", 0, SND_SOC_NOPM, 8,
4166 0, tabla_codec_enable_slimtx,
4167 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4168
4169 SND_SOC_DAPM_AIF_OUT_E("SLIM TX9", "AIF1 Capture", 0, SND_SOC_NOPM, 9,
4170 0, tabla_codec_enable_slimtx,
4171 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4172
4173 SND_SOC_DAPM_AIF_OUT_E("SLIM TX10", "AIF1 Capture", 0, SND_SOC_NOPM, 10,
4174 0, tabla_codec_enable_slimtx,
4175 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4176};
4177
4178static int tabla_set_interpolator_rate(struct snd_soc_dai *dai,
4179 u8 rx_fs_rate_reg_val, u32 compander_fs, u32 sample_rate)
4180{
4181 u32 i, j;
4182 u8 rx_mix1_inp;
4183 u16 rx_mix_1_reg_1, rx_mix_1_reg_2;
4184 u16 rx_fs_reg;
4185 u8 rx_mix_1_reg_1_val, rx_mix_1_reg_2_val;
4186 struct snd_soc_codec *codec = dai->codec;
4187 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
4188 struct snd_soc_dapm_widget *w = tabla_dapm_aif_in_widgets;
4189
4190 for (i = 0; i < ARRAY_SIZE(tabla_dapm_aif_in_widgets); i++) {
4191
4192 if (strncmp(dai->driver->playback.stream_name, w[i].sname, 13))
4193 continue;
4194
4195 rx_mix1_inp = w[i].shift + 4;
4196
4197 if ((rx_mix1_inp < 0x5) || (rx_mix1_inp > 0xB)) {
4198
4199 pr_err("%s: Invalid SLIM RX%u port. widget = %s\n",
4200 __func__, rx_mix1_inp - 4 , w[i].name);
4201 return -EINVAL;
4202 }
4203
4204 rx_mix_1_reg_1 = TABLA_A_CDC_CONN_RX1_B1_CTL;
4205
4206 for (j = 0; j < NUM_INTERPOLATORS; j++) {
4207
4208 rx_mix_1_reg_2 = rx_mix_1_reg_1 + 1;
4209
4210 rx_mix_1_reg_1_val = snd_soc_read(codec,
4211 rx_mix_1_reg_1);
4212 rx_mix_1_reg_2_val = snd_soc_read(codec,
4213 rx_mix_1_reg_2);
4214
4215 if (((rx_mix_1_reg_1_val & 0x0F) == rx_mix1_inp) ||
4216 (((rx_mix_1_reg_1_val >> 4) & 0x0F) == rx_mix1_inp)
4217 || ((rx_mix_1_reg_2_val & 0x0F) == rx_mix1_inp)) {
4218
4219 rx_fs_reg = TABLA_A_CDC_RX1_B5_CTL + 8 * j;
4220
4221 pr_debug("%s: %s connected to RX%u\n", __func__,
4222 w[i].name, j + 1);
4223
4224 pr_debug("%s: set RX%u sample rate to %u\n",
4225 __func__, j + 1, sample_rate);
4226
4227 snd_soc_update_bits(codec, rx_fs_reg,
4228 0xE0, rx_fs_rate_reg_val);
4229
4230 if (comp_rx_path[j] < COMPANDER_MAX)
4231 tabla->comp_fs[comp_rx_path[j]]
4232 = compander_fs;
4233 }
4234 if (j <= 2)
4235 rx_mix_1_reg_1 += 3;
4236 else
4237 rx_mix_1_reg_1 += 2;
4238 }
4239 }
4240 return 0;
4241}
4242
4243static int tabla_set_decimator_rate(struct snd_soc_dai *dai,
4244 u8 tx_fs_rate_reg_val, u32 sample_rate)
4245{
4246 struct snd_soc_codec *codec = dai->codec;
4247 struct snd_soc_dapm_widget *w = tabla_dapm_aif_out_widgets;
4248
4249 u32 i, tx_port;
4250 u16 tx_port_reg, tx_fs_reg;
4251 u8 tx_port_reg_val;
4252 s8 decimator;
4253
4254 for (i = 0; i < ARRAY_SIZE(tabla_dapm_aif_out_widgets); i++) {
4255
4256 if (strncmp(dai->driver->capture.stream_name, w[i].sname, 12))
4257 continue;
4258
4259 tx_port = w[i].shift;
4260
4261 if ((tx_port < 1) || (tx_port > NUM_DECIMATORS)) {
4262 pr_err("%s: Invalid SLIM TX%u port. widget = %s\n",
4263 __func__, tx_port, w[i].name);
4264 return -EINVAL;
4265 }
4266
4267 tx_port_reg = TABLA_A_CDC_CONN_TX_SB_B1_CTL + (tx_port - 1);
4268 tx_port_reg_val = snd_soc_read(codec, tx_port_reg);
4269
4270 decimator = 0;
4271
4272 if ((tx_port >= 1) && (tx_port <= 6)) {
4273
4274 tx_port_reg_val = tx_port_reg_val & 0x0F;
4275 if (tx_port_reg_val == 0x8)
4276 decimator = tx_port;
4277
4278 } else if ((tx_port >= 7) && (tx_port <= NUM_DECIMATORS)) {
4279
4280 tx_port_reg_val = tx_port_reg_val & 0x1F;
4281
4282 if ((tx_port_reg_val >= 0x8) &&
4283 (tx_port_reg_val <= 0x11)) {
4284
4285 decimator = (tx_port_reg_val - 0x8) + 1;
4286 }
4287 }
4288
4289 if (decimator) { /* SLIM_TX port has a DEC as input */
4290
4291 tx_fs_reg = TABLA_A_CDC_TX1_CLK_FS_CTL +
4292 8 * (decimator - 1);
4293
4294 pr_debug("%s: set DEC%u (-> SLIM_TX%u) rate to %u\n",
4295 __func__, decimator, tx_port, sample_rate);
4296
4297 snd_soc_update_bits(codec, tx_fs_reg, 0x07,
4298 tx_fs_rate_reg_val);
4299
4300 } else {
4301 if ((tx_port_reg_val >= 0x1) &&
4302 (tx_port_reg_val <= 0x7)) {
4303
4304 pr_debug("%s: RMIX%u going to SLIM TX%u\n",
4305 __func__, tx_port_reg_val, tx_port);
4306
4307 } else if ((tx_port_reg_val >= 0x8) &&
4308 (tx_port_reg_val <= 0x11)) {
4309
4310 pr_err("%s: ERROR: Should not be here\n",
4311 __func__);
4312 pr_err("%s: ERROR: DEC connected to SLIM TX%u\n"
4313 , __func__, tx_port);
4314 return -EINVAL;
4315
4316 } else if (tx_port_reg_val == 0) {
4317 pr_debug("%s: no signal to SLIM TX%u\n",
4318 __func__, tx_port);
4319 } else {
4320 pr_err("%s: ERROR: wrong signal to SLIM TX%u\n"
4321 , __func__, tx_port);
4322 pr_err("%s: ERROR: wrong signal = %u\n"
4323 , __func__, tx_port_reg_val);
4324 return -EINVAL;
4325 }
4326 }
4327 }
4328 return 0;
4329}
4330
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004331static int tabla_hw_params(struct snd_pcm_substream *substream,
Kiran Kandi93923902012-06-20 17:00:25 -07004332 struct snd_pcm_hw_params *params,
4333 struct snd_soc_dai *dai)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004334{
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004335 struct snd_soc_codec *codec = dai->codec;
Santosh Mardie15e2302011-11-15 10:39:23 +05304336 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(dai->codec);
Kiran Kandi93923902012-06-20 17:00:25 -07004337 u8 tx_fs_rate_reg_val, rx_fs_rate_reg_val;
Kuirong Wang0f8ade32012-02-27 16:29:45 -08004338 u32 compander_fs;
Kiran Kandi93923902012-06-20 17:00:25 -07004339 int ret;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004340
Kiran Kandia9fffe92012-05-20 23:42:30 -07004341 pr_debug("%s: dai_name = %s DAI-ID %x rate %d num_ch %d\n", __func__,
Kiran Kandi93923902012-06-20 17:00:25 -07004342 dai->name, dai->id, params_rate(params),
4343 params_channels(params));
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004344
4345 switch (params_rate(params)) {
4346 case 8000:
Kiran Kandi93923902012-06-20 17:00:25 -07004347 tx_fs_rate_reg_val = 0x00;
4348 rx_fs_rate_reg_val = 0x00;
Kuirong Wang0f8ade32012-02-27 16:29:45 -08004349 compander_fs = COMPANDER_FS_8KHZ;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004350 break;
4351 case 16000:
Kiran Kandi93923902012-06-20 17:00:25 -07004352 tx_fs_rate_reg_val = 0x01;
4353 rx_fs_rate_reg_val = 0x20;
Kuirong Wang0f8ade32012-02-27 16:29:45 -08004354 compander_fs = COMPANDER_FS_16KHZ;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004355 break;
4356 case 32000:
Kiran Kandi93923902012-06-20 17:00:25 -07004357 tx_fs_rate_reg_val = 0x02;
4358 rx_fs_rate_reg_val = 0x40;
Kuirong Wang0f8ade32012-02-27 16:29:45 -08004359 compander_fs = COMPANDER_FS_32KHZ;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004360 break;
4361 case 48000:
Kiran Kandi93923902012-06-20 17:00:25 -07004362 tx_fs_rate_reg_val = 0x03;
4363 rx_fs_rate_reg_val = 0x60;
Kuirong Wang0f8ade32012-02-27 16:29:45 -08004364 compander_fs = COMPANDER_FS_48KHZ;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004365 break;
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004366 case 96000:
Kiran Kandi93923902012-06-20 17:00:25 -07004367 tx_fs_rate_reg_val = 0x04;
4368 rx_fs_rate_reg_val = 0x80;
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004369 compander_fs = COMPANDER_FS_96KHZ;
4370 break;
4371 case 192000:
Kiran Kandi93923902012-06-20 17:00:25 -07004372 tx_fs_rate_reg_val = 0x05;
4373 rx_fs_rate_reg_val = 0xA0;
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004374 compander_fs = COMPANDER_FS_192KHZ;
4375 break;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004376 default:
4377 pr_err("%s: Invalid sampling rate %d\n", __func__,
4378 params_rate(params));
4379 return -EINVAL;
4380 }
4381
Kiran Kandi93923902012-06-20 17:00:25 -07004382 switch (substream->stream) {
4383 case SNDRV_PCM_STREAM_CAPTURE:
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004384
Kiran Kandi93923902012-06-20 17:00:25 -07004385 ret = tabla_set_decimator_rate(dai, tx_fs_rate_reg_val,
4386 params_rate(params));
4387 if (ret < 0) {
4388 pr_err("%s: set decimator rate failed %d\n", __func__,
4389 ret);
4390 return ret;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004391 }
Kiran Kandi93923902012-06-20 17:00:25 -07004392
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304393 if (tabla->intf_type == WCD9XXX_INTERFACE_TYPE_I2C) {
Santosh Mardie15e2302011-11-15 10:39:23 +05304394 switch (params_format(params)) {
4395 case SNDRV_PCM_FORMAT_S16_LE:
4396 snd_soc_update_bits(codec,
Kiran Kandi93923902012-06-20 17:00:25 -07004397 TABLA_A_CDC_CLK_TX_I2S_CTL, 0x20, 0x20);
Santosh Mardie15e2302011-11-15 10:39:23 +05304398 break;
4399 case SNDRV_PCM_FORMAT_S32_LE:
4400 snd_soc_update_bits(codec,
Kiran Kandi93923902012-06-20 17:00:25 -07004401 TABLA_A_CDC_CLK_TX_I2S_CTL, 0x20, 0x00);
Santosh Mardie15e2302011-11-15 10:39:23 +05304402 break;
4403 default:
Kiran Kandi93923902012-06-20 17:00:25 -07004404 pr_err("%s: invalid TX format %u\n", __func__,
4405 params_format(params));
4406 return -EINVAL;
Santosh Mardie15e2302011-11-15 10:39:23 +05304407 }
4408 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_TX_I2S_CTL,
Kiran Kandi93923902012-06-20 17:00:25 -07004409 0x07, tx_fs_rate_reg_val);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004410 } else {
4411 tabla->dai[dai->id - 1].rate = params_rate(params);
Santosh Mardie15e2302011-11-15 10:39:23 +05304412 }
Kiran Kandi93923902012-06-20 17:00:25 -07004413 break;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004414
Kiran Kandi93923902012-06-20 17:00:25 -07004415 case SNDRV_PCM_STREAM_PLAYBACK:
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004416
Kiran Kandi93923902012-06-20 17:00:25 -07004417 ret = tabla_set_interpolator_rate(dai, rx_fs_rate_reg_val,
4418 compander_fs, params_rate(params));
4419 if (ret < 0) {
4420 pr_err("%s: set decimator rate failed %d\n", __func__,
4421 ret);
4422 return ret;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004423 }
Kiran Kandi93923902012-06-20 17:00:25 -07004424
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304425 if (tabla->intf_type == WCD9XXX_INTERFACE_TYPE_I2C) {
Santosh Mardie15e2302011-11-15 10:39:23 +05304426 switch (params_format(params)) {
4427 case SNDRV_PCM_FORMAT_S16_LE:
4428 snd_soc_update_bits(codec,
Kiran Kandi93923902012-06-20 17:00:25 -07004429 TABLA_A_CDC_CLK_RX_I2S_CTL, 0x20, 0x20);
Santosh Mardie15e2302011-11-15 10:39:23 +05304430 break;
4431 case SNDRV_PCM_FORMAT_S32_LE:
4432 snd_soc_update_bits(codec,
Kiran Kandi93923902012-06-20 17:00:25 -07004433 TABLA_A_CDC_CLK_RX_I2S_CTL, 0x20, 0x00);
Santosh Mardie15e2302011-11-15 10:39:23 +05304434 break;
4435 default:
Kiran Kandi93923902012-06-20 17:00:25 -07004436 pr_err("%s: invalid RX format %u\n", __func__,
4437 params_format(params));
4438 return -EINVAL;
Santosh Mardie15e2302011-11-15 10:39:23 +05304439 }
4440 snd_soc_update_bits(codec, TABLA_A_CDC_CLK_RX_I2S_CTL,
Kiran Kandi93923902012-06-20 17:00:25 -07004441 0x03, (rx_fs_rate_reg_val >> 0x05));
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004442 } else {
4443 tabla->dai[dai->id - 1].rate = params_rate(params);
Santosh Mardie15e2302011-11-15 10:39:23 +05304444 }
Kiran Kandi93923902012-06-20 17:00:25 -07004445 break;
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004446
Kiran Kandi93923902012-06-20 17:00:25 -07004447 default:
4448 pr_err("%s: Invalid stream type %d\n", __func__,
4449 substream->stream);
4450 return -EINVAL;
4451 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004452 return 0;
4453}
4454
4455static struct snd_soc_dai_ops tabla_dai_ops = {
4456 .startup = tabla_startup,
Ajay Dudani506f81c2012-08-23 16:01:50 -07004457 .shutdown = tabla_shutdown,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004458 .hw_params = tabla_hw_params,
4459 .set_sysclk = tabla_set_dai_sysclk,
4460 .set_fmt = tabla_set_dai_fmt,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004461 .set_channel_map = tabla_set_channel_map,
4462 .get_channel_map = tabla_get_channel_map,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004463};
4464
4465static struct snd_soc_dai_driver tabla_dai[] = {
4466 {
4467 .name = "tabla_rx1",
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004468 .id = AIF1_PB,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004469 .playback = {
4470 .stream_name = "AIF1 Playback",
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004471 .rates = WCD9310_RATES,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004472 .formats = TABLA_FORMATS,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004473 .rate_max = 192000,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004474 .rate_min = 8000,
4475 .channels_min = 1,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004476 .channels_max = 2,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004477 },
4478 .ops = &tabla_dai_ops,
4479 },
4480 {
4481 .name = "tabla_tx1",
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004482 .id = AIF1_CAP,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004483 .capture = {
4484 .stream_name = "AIF1 Capture",
Bhalchandra Gajare9a901fd2011-08-01 10:07:15 -07004485 .rates = WCD9310_RATES,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004486 .formats = TABLA_FORMATS,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004487 .rate_max = 192000,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004488 .rate_min = 8000,
4489 .channels_min = 1,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004490 .channels_max = 4,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004491 },
4492 .ops = &tabla_dai_ops,
4493 },
Neema Shettyd3a89262012-02-16 10:23:50 -08004494 {
4495 .name = "tabla_rx2",
4496 .id = AIF2_PB,
4497 .playback = {
4498 .stream_name = "AIF2 Playback",
4499 .rates = WCD9310_RATES,
4500 .formats = TABLA_FORMATS,
4501 .rate_min = 8000,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004502 .rate_max = 192000,
Neema Shettyd3a89262012-02-16 10:23:50 -08004503 .channels_min = 1,
4504 .channels_max = 2,
4505 },
4506 .ops = &tabla_dai_ops,
4507 },
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004508 {
4509 .name = "tabla_tx2",
4510 .id = AIF2_CAP,
4511 .capture = {
4512 .stream_name = "AIF2 Capture",
4513 .rates = WCD9310_RATES,
4514 .formats = TABLA_FORMATS,
4515 .rate_max = 192000,
4516 .rate_min = 8000,
4517 .channels_min = 1,
4518 .channels_max = 4,
4519 },
4520 .ops = &tabla_dai_ops,
4521 },
Neema Shetty3fb1b802012-04-27 13:53:24 -07004522 {
4523 .name = "tabla_tx3",
4524 .id = AIF3_CAP,
4525 .capture = {
4526 .stream_name = "AIF3 Capture",
4527 .rates = WCD9310_RATES,
4528 .formats = TABLA_FORMATS,
4529 .rate_max = 48000,
4530 .rate_min = 8000,
4531 .channels_min = 1,
4532 .channels_max = 2,
4533 },
4534 .ops = &tabla_dai_ops,
4535 },
Kiran Kandia9fffe92012-05-20 23:42:30 -07004536 {
4537 .name = "tabla_rx3",
4538 .id = AIF3_PB,
4539 .playback = {
4540 .stream_name = "AIF3 Playback",
4541 .rates = WCD9310_RATES,
4542 .formats = TABLA_FORMATS,
4543 .rate_min = 8000,
4544 .rate_max = 192000,
4545 .channels_min = 1,
4546 .channels_max = 2,
4547 },
4548 .ops = &tabla_dai_ops,
4549 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07004550};
Santosh Mardie15e2302011-11-15 10:39:23 +05304551
4552static struct snd_soc_dai_driver tabla_i2s_dai[] = {
4553 {
4554 .name = "tabla_i2s_rx1",
4555 .id = 1,
4556 .playback = {
4557 .stream_name = "AIF1 Playback",
4558 .rates = WCD9310_RATES,
4559 .formats = TABLA_FORMATS,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004560 .rate_max = 192000,
Santosh Mardie15e2302011-11-15 10:39:23 +05304561 .rate_min = 8000,
4562 .channels_min = 1,
4563 .channels_max = 4,
4564 },
4565 .ops = &tabla_dai_ops,
4566 },
4567 {
4568 .name = "tabla_i2s_tx1",
4569 .id = 2,
4570 .capture = {
4571 .stream_name = "AIF1 Capture",
4572 .rates = WCD9310_RATES,
4573 .formats = TABLA_FORMATS,
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004574 .rate_max = 192000,
Santosh Mardie15e2302011-11-15 10:39:23 +05304575 .rate_min = 8000,
4576 .channels_min = 1,
4577 .channels_max = 4,
4578 },
4579 .ops = &tabla_dai_ops,
4580 },
4581};
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004582
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07004583static int tabla_codec_enable_chmask(struct tabla_priv *tabla_p,
4584 int event, int index)
4585{
4586 int ret = 0;
4587 u32 k = 0;
4588 switch (event) {
4589 case SND_SOC_DAPM_POST_PMU:
4590 for (k = 0; k < tabla_p->dai[index].ch_tot; k++) {
4591 ret = wcd9xxx_get_slave_port(
4592 tabla_p->dai[index].ch_num[k]);
4593 if (ret < 0) {
4594 pr_err("%s: Invalid slave port ID: %d\n",
4595 __func__, ret);
4596 ret = -EINVAL;
4597 break;
4598 }
4599 tabla_p->dai[index].ch_mask |= 1 << ret;
4600 }
4601 ret = 0;
4602 break;
4603 case SND_SOC_DAPM_POST_PMD:
4604 ret = wait_event_timeout(tabla_p->dai[index].dai_wait,
4605 (tabla_p->dai[index].ch_mask == 0),
4606 msecs_to_jiffies(SLIM_CLOSE_TIMEOUT));
4607 if (!ret) {
4608 pr_err("%s: Slim close tx/rx wait timeout\n",
4609 __func__);
4610 ret = -EINVAL;
SathishKumar Mani58253c72012-10-06 13:24:38 -07004611 } else
4612 ret = 0;
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07004613 break;
4614 }
4615 return ret;
4616}
4617
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004618static int tabla_codec_enable_slimrx(struct snd_soc_dapm_widget *w,
4619 struct snd_kcontrol *kcontrol, int event)
4620{
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304621 struct wcd9xxx *tabla;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004622 struct snd_soc_codec *codec = w->codec;
4623 struct tabla_priv *tabla_p = snd_soc_codec_get_drvdata(codec);
4624 u32 j = 0;
SathishKumar Mani58253c72012-10-06 13:24:38 -07004625 int ret = 0;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004626 codec->control_data = dev_get_drvdata(codec->dev->parent);
4627 tabla = codec->control_data;
Swaminathan Sathappanb74caaa2012-07-10 17:28:54 -07004628
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004629 /* Execute the callback only if interface type is slimbus */
Swaminathan Sathappanb74caaa2012-07-10 17:28:54 -07004630 if (tabla_p->intf_type != WCD9XXX_INTERFACE_TYPE_SLIMBUS) {
4631 if (event == SND_SOC_DAPM_POST_PMD && (tabla != NULL) &&
4632 (tabla->dev != NULL) &&
4633 (tabla->dev->parent != NULL)) {
4634 pm_runtime_mark_last_busy(tabla->dev->parent);
4635 pm_runtime_put(tabla->dev->parent);
4636 }
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004637 return 0;
Swaminathan Sathappanb74caaa2012-07-10 17:28:54 -07004638 }
Kiran Kandia9fffe92012-05-20 23:42:30 -07004639
4640 pr_debug("%s: %s %d\n", __func__, w->name, event);
4641
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004642 switch (event) {
4643 case SND_SOC_DAPM_POST_PMU:
4644 for (j = 0; j < ARRAY_SIZE(tabla_dai); j++) {
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004645 if ((tabla_dai[j].id == AIF1_CAP) ||
Neema Shetty3fb1b802012-04-27 13:53:24 -07004646 (tabla_dai[j].id == AIF2_CAP) ||
4647 (tabla_dai[j].id == AIF3_CAP))
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004648 continue;
4649 if (!strncmp(w->sname,
4650 tabla_dai[j].playback.stream_name, 13)) {
4651 ++tabla_p->dai[j].ch_act;
4652 break;
4653 }
4654 }
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07004655 if (tabla_p->dai[j].ch_act == tabla_p->dai[j].ch_tot) {
4656 ret = tabla_codec_enable_chmask(tabla_p,
4657 SND_SOC_DAPM_POST_PMU,
4658 j);
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304659 ret = wcd9xxx_cfg_slim_sch_rx(tabla,
4660 tabla_p->dai[j].ch_num,
4661 tabla_p->dai[j].ch_tot,
4662 tabla_p->dai[j].rate);
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07004663 }
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004664 break;
4665 case SND_SOC_DAPM_POST_PMD:
4666 for (j = 0; j < ARRAY_SIZE(tabla_dai); j++) {
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004667 if ((tabla_dai[j].id == AIF1_CAP) ||
Neema Shetty3fb1b802012-04-27 13:53:24 -07004668 (tabla_dai[j].id == AIF2_CAP) ||
4669 (tabla_dai[j].id == AIF3_CAP))
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004670 continue;
4671 if (!strncmp(w->sname,
4672 tabla_dai[j].playback.stream_name, 13)) {
Helen Zeng2aaa3c12012-06-08 20:29:21 -07004673 if (tabla_p->dai[j].ch_act)
4674 --tabla_p->dai[j].ch_act;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004675 break;
4676 }
4677 }
4678 if (!tabla_p->dai[j].ch_act) {
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304679 ret = wcd9xxx_close_slim_sch_rx(tabla,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004680 tabla_p->dai[j].ch_num,
4681 tabla_p->dai[j].ch_tot);
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07004682 ret = tabla_codec_enable_chmask(tabla_p,
4683 SND_SOC_DAPM_POST_PMD,
4684 j);
SathishKumar Mani58253c72012-10-06 13:24:38 -07004685 if (ret < 0) {
4686 ret = wcd9xxx_disconnect_port(tabla,
4687 tabla_p->dai[j].ch_num,
4688 tabla_p->dai[j].ch_tot,
4689 1);
4690 pr_info("%s: Disconnect RX port ret = %d\n",
4691 __func__, ret);
4692 }
4693 tabla_p->dai[j].rate = 0;
4694 memset(tabla_p->dai[j].ch_num, 0, (sizeof(u32)*
4695 tabla_p->dai[j].ch_tot));
4696 tabla_p->dai[j].ch_tot = 0;
4697
Swaminathan Sathappanb74caaa2012-07-10 17:28:54 -07004698 if ((tabla != NULL) &&
4699 (tabla->dev != NULL) &&
4700 (tabla->dev->parent != NULL)) {
4701 pm_runtime_mark_last_busy(tabla->dev->parent);
4702 pm_runtime_put(tabla->dev->parent);
4703 }
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004704 }
4705 }
4706 return ret;
4707}
4708
4709static int tabla_codec_enable_slimtx(struct snd_soc_dapm_widget *w,
4710 struct snd_kcontrol *kcontrol, int event)
4711{
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304712 struct wcd9xxx *tabla;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004713 struct snd_soc_codec *codec = w->codec;
4714 struct tabla_priv *tabla_p = snd_soc_codec_get_drvdata(codec);
4715 /* index to the DAI ID, for now hardcoding */
4716 u32 j = 0;
SathishKumar Mani58253c72012-10-06 13:24:38 -07004717 int ret = 0;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004718
4719 codec->control_data = dev_get_drvdata(codec->dev->parent);
4720 tabla = codec->control_data;
4721
4722 /* Execute the callback only if interface type is slimbus */
Swaminathan Sathappanb74caaa2012-07-10 17:28:54 -07004723 if (tabla_p->intf_type != WCD9XXX_INTERFACE_TYPE_SLIMBUS) {
4724 if (event == SND_SOC_DAPM_POST_PMD && (tabla != NULL) &&
4725 (tabla->dev != NULL) &&
4726 (tabla->dev->parent != NULL)) {
4727 pm_runtime_mark_last_busy(tabla->dev->parent);
4728 pm_runtime_put(tabla->dev->parent);
4729 }
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004730 return 0;
Swaminathan Sathappanb74caaa2012-07-10 17:28:54 -07004731 }
Kiran Kandi1e6371d2012-03-29 11:48:57 -07004732
4733 pr_debug("%s(): %s %d\n", __func__, w->name, event);
4734
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004735 switch (event) {
4736 case SND_SOC_DAPM_POST_PMU:
4737 for (j = 0; j < ARRAY_SIZE(tabla_dai); j++) {
Neema Shettyd3a89262012-02-16 10:23:50 -08004738 if (tabla_dai[j].id == AIF1_PB ||
Kiran Kandia9fffe92012-05-20 23:42:30 -07004739 tabla_dai[j].id == AIF2_PB ||
4740 tabla_dai[j].id == AIF3_PB)
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004741 continue;
4742 if (!strncmp(w->sname,
4743 tabla_dai[j].capture.stream_name, 13)) {
4744 ++tabla_p->dai[j].ch_act;
4745 break;
4746 }
4747 }
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07004748 if (tabla_p->dai[j].ch_act == tabla_p->dai[j].ch_tot) {
4749 ret = tabla_codec_enable_chmask(tabla_p,
4750 SND_SOC_DAPM_POST_PMU,
4751 j);
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304752 ret = wcd9xxx_cfg_slim_sch_tx(tabla,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004753 tabla_p->dai[j].ch_num,
4754 tabla_p->dai[j].ch_tot,
4755 tabla_p->dai[j].rate);
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07004756 }
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004757 break;
4758 case SND_SOC_DAPM_POST_PMD:
4759 for (j = 0; j < ARRAY_SIZE(tabla_dai); j++) {
Neema Shettyd3a89262012-02-16 10:23:50 -08004760 if (tabla_dai[j].id == AIF1_PB ||
Kiran Kandia9fffe92012-05-20 23:42:30 -07004761 tabla_dai[j].id == AIF2_PB ||
4762 tabla_dai[j].id == AIF3_PB)
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004763 continue;
4764 if (!strncmp(w->sname,
4765 tabla_dai[j].capture.stream_name, 13)) {
4766 --tabla_p->dai[j].ch_act;
4767 break;
4768 }
4769 }
4770 if (!tabla_p->dai[j].ch_act) {
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304771 ret = wcd9xxx_close_slim_sch_tx(tabla,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004772 tabla_p->dai[j].ch_num,
4773 tabla_p->dai[j].ch_tot);
SathishKumar Mani58253c72012-10-06 13:24:38 -07004774 ret = tabla_codec_enable_chmask(tabla_p,
4775 SND_SOC_DAPM_POST_PMD,
4776 j);
4777 if (ret < 0) {
4778 ret = wcd9xxx_disconnect_port(tabla,
4779 tabla_p->dai[j].ch_num,
4780 tabla_p->dai[j].ch_tot, 0);
4781 pr_info("%s: Disconnect TX port, ret = %d\n",
4782 __func__, ret);
4783 }
4784
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004785 tabla_p->dai[j].rate = 0;
4786 memset(tabla_p->dai[j].ch_num, 0, (sizeof(u32)*
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05304787 tabla_p->dai[j].ch_tot));
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004788 tabla_p->dai[j].ch_tot = 0;
Swaminathan Sathappanb74caaa2012-07-10 17:28:54 -07004789 if ((tabla != NULL) &&
4790 (tabla->dev != NULL) &&
4791 (tabla->dev->parent != NULL)) {
4792 pm_runtime_mark_last_busy(tabla->dev->parent);
4793 pm_runtime_put(tabla->dev->parent);
4794 }
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004795 }
4796 }
4797 return ret;
4798}
4799
4800/* Todo: Have seperate dapm widgets for I2S and Slimbus.
4801 * Might Need to have callbacks registered only for slimbus
4802 */
4803static const struct snd_soc_dapm_widget tabla_dapm_widgets[] = {
4804 /*RX stuff */
4805 SND_SOC_DAPM_OUTPUT("EAR"),
4806
Damir Didjusto52900462012-08-16 21:22:29 -07004807 SND_SOC_DAPM_PGA_E("EAR PA", SND_SOC_NOPM, 0, 0, NULL,
4808 0, tabla_ear_pa_event, SND_SOC_DAPM_PRE_PMU |
4809 SND_SOC_DAPM_PRE_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004810
Damir Didjusto52900462012-08-16 21:22:29 -07004811 SND_SOC_DAPM_MIXER("DAC1", SND_SOC_NOPM, 0, 0, dac1_switch,
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004812 ARRAY_SIZE(dac1_switch)),
4813
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004814 /* Headphone */
4815 SND_SOC_DAPM_OUTPUT("HEADPHONE"),
4816 SND_SOC_DAPM_PGA_E("HPHL", TABLA_A_RX_HPH_CNP_EN, 5, 0, NULL, 0,
4817 tabla_hph_pa_event, SND_SOC_DAPM_PRE_PMU |
4818 SND_SOC_DAPM_POST_PMD),
4819 SND_SOC_DAPM_MIXER("HPHL DAC", TABLA_A_RX_HPH_L_DAC_CTL, 7, 0,
4820 hphl_switch, ARRAY_SIZE(hphl_switch)),
4821
4822 SND_SOC_DAPM_PGA_E("HPHR", TABLA_A_RX_HPH_CNP_EN, 4, 0, NULL, 0,
4823 tabla_hph_pa_event, SND_SOC_DAPM_PRE_PMU |
4824 SND_SOC_DAPM_POST_PMD),
4825
4826 SND_SOC_DAPM_DAC_E("HPHR DAC", NULL, TABLA_A_RX_HPH_R_DAC_CTL, 7, 0,
4827 tabla_hphr_dac_event,
4828 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
4829
4830 /* Speaker */
4831 SND_SOC_DAPM_OUTPUT("LINEOUT1"),
4832 SND_SOC_DAPM_OUTPUT("LINEOUT2"),
4833 SND_SOC_DAPM_OUTPUT("LINEOUT3"),
4834 SND_SOC_DAPM_OUTPUT("LINEOUT4"),
4835 SND_SOC_DAPM_OUTPUT("LINEOUT5"),
4836
4837 SND_SOC_DAPM_PGA_E("LINEOUT1 PA", TABLA_A_RX_LINE_CNP_EN, 0, 0, NULL,
4838 0, tabla_codec_enable_lineout, SND_SOC_DAPM_PRE_PMU |
4839 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4840 SND_SOC_DAPM_PGA_E("LINEOUT2 PA", TABLA_A_RX_LINE_CNP_EN, 1, 0, NULL,
4841 0, tabla_codec_enable_lineout, SND_SOC_DAPM_PRE_PMU |
4842 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4843 SND_SOC_DAPM_PGA_E("LINEOUT3 PA", TABLA_A_RX_LINE_CNP_EN, 2, 0, NULL,
4844 0, tabla_codec_enable_lineout, SND_SOC_DAPM_PRE_PMU |
4845 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4846 SND_SOC_DAPM_PGA_E("LINEOUT4 PA", TABLA_A_RX_LINE_CNP_EN, 3, 0, NULL,
4847 0, tabla_codec_enable_lineout, SND_SOC_DAPM_PRE_PMU |
4848 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4849 SND_SOC_DAPM_PGA_E("LINEOUT5 PA", TABLA_A_RX_LINE_CNP_EN, 4, 0, NULL, 0,
4850 tabla_codec_enable_lineout, SND_SOC_DAPM_PRE_PMU |
4851 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4852
4853 SND_SOC_DAPM_DAC_E("LINEOUT1 DAC", NULL, TABLA_A_RX_LINE_1_DAC_CTL, 7, 0
4854 , tabla_lineout_dac_event,
4855 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
4856 SND_SOC_DAPM_DAC_E("LINEOUT2 DAC", NULL, TABLA_A_RX_LINE_2_DAC_CTL, 7, 0
4857 , tabla_lineout_dac_event,
4858 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
4859 SND_SOC_DAPM_DAC_E("LINEOUT3 DAC", NULL, TABLA_A_RX_LINE_3_DAC_CTL, 7, 0
4860 , tabla_lineout_dac_event,
4861 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
4862 SND_SOC_DAPM_SWITCH("LINEOUT3 DAC GROUND", SND_SOC_NOPM, 0, 0,
4863 &lineout3_ground_switch),
4864 SND_SOC_DAPM_DAC_E("LINEOUT4 DAC", NULL, TABLA_A_RX_LINE_4_DAC_CTL, 7, 0
4865 , tabla_lineout_dac_event,
4866 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
4867 SND_SOC_DAPM_SWITCH("LINEOUT4 DAC GROUND", SND_SOC_NOPM, 0, 0,
4868 &lineout4_ground_switch),
4869 SND_SOC_DAPM_DAC_E("LINEOUT5 DAC", NULL, TABLA_A_RX_LINE_5_DAC_CTL, 7, 0
4870 , tabla_lineout_dac_event,
4871 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
4872
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08004873 SND_SOC_DAPM_MIXER_E("RX1 MIX2", TABLA_A_CDC_CLK_RX_B1_CTL, 0, 0, NULL,
Kuirong Wange9c8a222012-03-28 16:24:09 -07004874 0, tabla_codec_reset_interpolator, SND_SOC_DAPM_PRE_PMU |
4875 SND_SOC_DAPM_POST_PMU),
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08004876 SND_SOC_DAPM_MIXER_E("RX2 MIX2", TABLA_A_CDC_CLK_RX_B1_CTL, 1, 0, NULL,
Kuirong Wange9c8a222012-03-28 16:24:09 -07004877 0, tabla_codec_reset_interpolator, SND_SOC_DAPM_PRE_PMU |
4878 SND_SOC_DAPM_POST_PMU),
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08004879 SND_SOC_DAPM_MIXER_E("RX3 MIX2", TABLA_A_CDC_CLK_RX_B1_CTL, 2, 0, NULL,
Kuirong Wange9c8a222012-03-28 16:24:09 -07004880 0, tabla_codec_reset_interpolator, SND_SOC_DAPM_PRE_PMU |
4881 SND_SOC_DAPM_POST_PMU),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004882 SND_SOC_DAPM_MIXER_E("RX4 MIX1", TABLA_A_CDC_CLK_RX_B1_CTL, 3, 0, NULL,
Kuirong Wange9c8a222012-03-28 16:24:09 -07004883 0, tabla_codec_reset_interpolator, SND_SOC_DAPM_PRE_PMU |
4884 SND_SOC_DAPM_POST_PMU),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004885 SND_SOC_DAPM_MIXER_E("RX5 MIX1", TABLA_A_CDC_CLK_RX_B1_CTL, 4, 0, NULL,
Kuirong Wange9c8a222012-03-28 16:24:09 -07004886 0, tabla_codec_reset_interpolator, SND_SOC_DAPM_PRE_PMU |
4887 SND_SOC_DAPM_POST_PMU),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004888 SND_SOC_DAPM_MIXER_E("RX6 MIX1", TABLA_A_CDC_CLK_RX_B1_CTL, 5, 0, NULL,
Kuirong Wange9c8a222012-03-28 16:24:09 -07004889 0, tabla_codec_reset_interpolator, SND_SOC_DAPM_PRE_PMU |
4890 SND_SOC_DAPM_POST_PMU),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004891 SND_SOC_DAPM_MIXER_E("RX7 MIX1", TABLA_A_CDC_CLK_RX_B1_CTL, 6, 0, NULL,
Kuirong Wange9c8a222012-03-28 16:24:09 -07004892 0, tabla_codec_reset_interpolator, SND_SOC_DAPM_PRE_PMU |
4893 SND_SOC_DAPM_POST_PMU),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004894
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08004895 SND_SOC_DAPM_MIXER("RX1 MIX1", SND_SOC_NOPM, 0, 0, NULL, 0),
4896 SND_SOC_DAPM_MIXER("RX2 MIX1", SND_SOC_NOPM, 0, 0, NULL, 0),
4897 SND_SOC_DAPM_MIXER("RX3 MIX1", SND_SOC_NOPM, 0, 0, NULL, 0),
4898
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004899 SND_SOC_DAPM_MUX_E("RX4 DSM MUX", TABLA_A_CDC_CLK_RX_B1_CTL, 3, 0,
4900 &rx4_dsm_mux, tabla_codec_reset_interpolator,
4901 SND_SOC_DAPM_PRE_PMU),
4902
4903 SND_SOC_DAPM_MUX_E("RX6 DSM MUX", TABLA_A_CDC_CLK_RX_B1_CTL, 5, 0,
4904 &rx6_dsm_mux, tabla_codec_reset_interpolator,
4905 SND_SOC_DAPM_PRE_PMU),
4906
4907 SND_SOC_DAPM_MIXER("RX1 CHAIN", TABLA_A_CDC_RX1_B6_CTL, 5, 0, NULL, 0),
4908 SND_SOC_DAPM_MIXER("RX2 CHAIN", TABLA_A_CDC_RX2_B6_CTL, 5, 0, NULL, 0),
4909
4910 SND_SOC_DAPM_MUX("RX1 MIX1 INP1", SND_SOC_NOPM, 0, 0,
4911 &rx_mix1_inp1_mux),
4912 SND_SOC_DAPM_MUX("RX1 MIX1 INP2", SND_SOC_NOPM, 0, 0,
4913 &rx_mix1_inp2_mux),
Kiran Kandia9fffe92012-05-20 23:42:30 -07004914 SND_SOC_DAPM_MUX("RX1 MIX1 INP3", SND_SOC_NOPM, 0, 0,
4915 &rx_mix1_inp3_mux),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004916 SND_SOC_DAPM_MUX("RX2 MIX1 INP1", SND_SOC_NOPM, 0, 0,
4917 &rx2_mix1_inp1_mux),
4918 SND_SOC_DAPM_MUX("RX2 MIX1 INP2", SND_SOC_NOPM, 0, 0,
4919 &rx2_mix1_inp2_mux),
4920 SND_SOC_DAPM_MUX("RX3 MIX1 INP1", SND_SOC_NOPM, 0, 0,
4921 &rx3_mix1_inp1_mux),
4922 SND_SOC_DAPM_MUX("RX3 MIX1 INP2", SND_SOC_NOPM, 0, 0,
4923 &rx3_mix1_inp2_mux),
4924 SND_SOC_DAPM_MUX("RX4 MIX1 INP1", SND_SOC_NOPM, 0, 0,
4925 &rx4_mix1_inp1_mux),
4926 SND_SOC_DAPM_MUX("RX4 MIX1 INP2", SND_SOC_NOPM, 0, 0,
4927 &rx4_mix1_inp2_mux),
4928 SND_SOC_DAPM_MUX("RX5 MIX1 INP1", SND_SOC_NOPM, 0, 0,
4929 &rx5_mix1_inp1_mux),
4930 SND_SOC_DAPM_MUX("RX5 MIX1 INP2", SND_SOC_NOPM, 0, 0,
4931 &rx5_mix1_inp2_mux),
4932 SND_SOC_DAPM_MUX("RX6 MIX1 INP1", SND_SOC_NOPM, 0, 0,
4933 &rx6_mix1_inp1_mux),
4934 SND_SOC_DAPM_MUX("RX6 MIX1 INP2", SND_SOC_NOPM, 0, 0,
4935 &rx6_mix1_inp2_mux),
4936 SND_SOC_DAPM_MUX("RX7 MIX1 INP1", SND_SOC_NOPM, 0, 0,
4937 &rx7_mix1_inp1_mux),
4938 SND_SOC_DAPM_MUX("RX7 MIX1 INP2", SND_SOC_NOPM, 0, 0,
4939 &rx7_mix1_inp2_mux),
Kuirong Wangbfdd6ca2012-02-29 13:06:38 -08004940 SND_SOC_DAPM_MUX("RX1 MIX2 INP1", SND_SOC_NOPM, 0, 0,
4941 &rx1_mix2_inp1_mux),
4942 SND_SOC_DAPM_MUX("RX1 MIX2 INP2", SND_SOC_NOPM, 0, 0,
4943 &rx1_mix2_inp2_mux),
4944 SND_SOC_DAPM_MUX("RX2 MIX2 INP1", SND_SOC_NOPM, 0, 0,
4945 &rx2_mix2_inp1_mux),
4946 SND_SOC_DAPM_MUX("RX2 MIX2 INP2", SND_SOC_NOPM, 0, 0,
4947 &rx2_mix2_inp2_mux),
4948 SND_SOC_DAPM_MUX("RX3 MIX2 INP1", SND_SOC_NOPM, 0, 0,
4949 &rx3_mix2_inp1_mux),
4950 SND_SOC_DAPM_MUX("RX3 MIX2 INP2", SND_SOC_NOPM, 0, 0,
4951 &rx3_mix2_inp2_mux),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004952
4953 SND_SOC_DAPM_SUPPLY("CP", TABLA_A_CP_EN, 0, 0,
4954 tabla_codec_enable_charge_pump, SND_SOC_DAPM_POST_PMU |
4955 SND_SOC_DAPM_PRE_PMD),
4956
4957 SND_SOC_DAPM_SUPPLY("RX_BIAS", SND_SOC_NOPM, 0, 0,
4958 tabla_codec_enable_rx_bias, SND_SOC_DAPM_PRE_PMU |
4959 SND_SOC_DAPM_POST_PMD),
4960
4961 /* TX */
4962
4963 SND_SOC_DAPM_SUPPLY("CDC_CONN", TABLA_A_CDC_CLK_OTHR_CTL, 2, 0, NULL,
4964 0),
4965
4966 SND_SOC_DAPM_SUPPLY("LDO_H", TABLA_A_LDO_H_MODE_1, 7, 0,
4967 tabla_codec_enable_ldo_h, SND_SOC_DAPM_POST_PMU),
4968
Kuirong Wang0f8ade32012-02-27 16:29:45 -08004969 SND_SOC_DAPM_SUPPLY("COMP1_CLK", SND_SOC_NOPM, 0, 0,
4970 tabla_config_compander, SND_SOC_DAPM_PRE_PMU |
4971 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_POST_PMD),
4972 SND_SOC_DAPM_SUPPLY("COMP2_CLK", SND_SOC_NOPM, 1, 0,
4973 tabla_config_compander, SND_SOC_DAPM_PRE_PMU |
4974 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_POST_PMD),
4975
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08004976 SND_SOC_DAPM_INPUT("AMIC1"),
4977 SND_SOC_DAPM_MICBIAS_E("MIC BIAS1 External", TABLA_A_MICB_1_CTL, 7, 0,
4978 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
4979 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4980 SND_SOC_DAPM_MICBIAS_E("MIC BIAS1 Internal1", TABLA_A_MICB_1_CTL, 7, 0,
4981 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
4982 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4983 SND_SOC_DAPM_MICBIAS_E("MIC BIAS1 Internal2", TABLA_A_MICB_1_CTL, 7, 0,
4984 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
4985 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4986 SND_SOC_DAPM_ADC_E("ADC1", NULL, TABLA_A_TX_1_2_EN, 7, 0,
4987 tabla_codec_enable_adc, SND_SOC_DAPM_PRE_PMU |
4988 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4989
4990 SND_SOC_DAPM_INPUT("AMIC3"),
4991 SND_SOC_DAPM_ADC_E("ADC3", NULL, TABLA_A_TX_3_4_EN, 7, 0,
4992 tabla_codec_enable_adc, SND_SOC_DAPM_PRE_PMU |
4993 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4994
4995 SND_SOC_DAPM_INPUT("AMIC4"),
4996 SND_SOC_DAPM_ADC_E("ADC4", NULL, TABLA_A_TX_3_4_EN, 3, 0,
4997 tabla_codec_enable_adc, SND_SOC_DAPM_PRE_PMU |
4998 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
4999
5000 SND_SOC_DAPM_INPUT("AMIC5"),
5001 SND_SOC_DAPM_ADC_E("ADC5", NULL, TABLA_A_TX_5_6_EN, 7, 0,
5002 tabla_codec_enable_adc, SND_SOC_DAPM_POST_PMU),
5003
5004 SND_SOC_DAPM_INPUT("AMIC6"),
5005 SND_SOC_DAPM_ADC_E("ADC6", NULL, TABLA_A_TX_5_6_EN, 3, 0,
5006 tabla_codec_enable_adc, SND_SOC_DAPM_POST_PMU),
5007
5008 SND_SOC_DAPM_MUX_E("DEC1 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B1_CTL, 0, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005009 &dec1_mux, tabla_codec_enable_dec,
5010 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5011 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005012
5013 SND_SOC_DAPM_MUX_E("DEC2 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B1_CTL, 1, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005014 &dec2_mux, tabla_codec_enable_dec,
5015 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5016 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005017
5018 SND_SOC_DAPM_MUX_E("DEC3 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B1_CTL, 2, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005019 &dec3_mux, tabla_codec_enable_dec,
5020 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5021 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005022
5023 SND_SOC_DAPM_MUX_E("DEC4 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B1_CTL, 3, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005024 &dec4_mux, tabla_codec_enable_dec,
5025 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5026 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005027
5028 SND_SOC_DAPM_MUX_E("DEC5 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B1_CTL, 4, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005029 &dec5_mux, tabla_codec_enable_dec,
5030 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5031 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005032
5033 SND_SOC_DAPM_MUX_E("DEC6 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B1_CTL, 5, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005034 &dec6_mux, tabla_codec_enable_dec,
5035 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5036 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005037
5038 SND_SOC_DAPM_MUX_E("DEC7 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B1_CTL, 6, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005039 &dec7_mux, tabla_codec_enable_dec,
5040 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5041 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005042
5043 SND_SOC_DAPM_MUX_E("DEC8 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B1_CTL, 7, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005044 &dec8_mux, tabla_codec_enable_dec,
5045 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5046 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005047
5048 SND_SOC_DAPM_MUX_E("DEC9 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B2_CTL, 0, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005049 &dec9_mux, tabla_codec_enable_dec,
5050 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5051 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005052
5053 SND_SOC_DAPM_MUX_E("DEC10 MUX", TABLA_A_CDC_CLK_TX_CLK_EN_B2_CTL, 1, 0,
Kiran Kandid8cf5212012-03-02 15:34:53 -08005054 &dec10_mux, tabla_codec_enable_dec,
5055 SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
5056 SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005057
5058 SND_SOC_DAPM_MUX("ANC1 MUX", SND_SOC_NOPM, 0, 0, &anc1_mux),
5059 SND_SOC_DAPM_MUX("ANC2 MUX", SND_SOC_NOPM, 0, 0, &anc2_mux),
5060
5061 SND_SOC_DAPM_MIXER_E("ANC", SND_SOC_NOPM, 0, 0, NULL, 0,
5062 tabla_codec_enable_anc, SND_SOC_DAPM_PRE_PMU |
5063 SND_SOC_DAPM_POST_PMD),
5064
5065 SND_SOC_DAPM_MUX("ANC1 FB MUX", SND_SOC_NOPM, 0, 0, &anc1_fb_mux),
5066
5067 SND_SOC_DAPM_INPUT("AMIC2"),
5068 SND_SOC_DAPM_MICBIAS_E("MIC BIAS2 External", TABLA_A_MICB_2_CTL, 7, 0,
5069 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
5070 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
5071 SND_SOC_DAPM_MICBIAS_E("MIC BIAS2 Internal1", TABLA_A_MICB_2_CTL, 7, 0,
5072 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
5073 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
5074 SND_SOC_DAPM_MICBIAS_E("MIC BIAS2 Internal2", TABLA_A_MICB_2_CTL, 7, 0,
5075 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
5076 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
5077 SND_SOC_DAPM_MICBIAS_E("MIC BIAS2 Internal3", TABLA_A_MICB_2_CTL, 7, 0,
5078 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
5079 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
5080 SND_SOC_DAPM_MICBIAS_E("MIC BIAS3 External", TABLA_A_MICB_3_CTL, 7, 0,
5081 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
5082 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
5083 SND_SOC_DAPM_MICBIAS_E("MIC BIAS3 Internal1", TABLA_A_MICB_3_CTL, 7, 0,
5084 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
5085 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
5086 SND_SOC_DAPM_MICBIAS_E("MIC BIAS3 Internal2", TABLA_A_MICB_3_CTL, 7, 0,
5087 tabla_codec_enable_micbias, SND_SOC_DAPM_PRE_PMU |
5088 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
5089 SND_SOC_DAPM_ADC_E("ADC2", NULL, TABLA_A_TX_1_2_EN, 3, 0,
5090 tabla_codec_enable_adc, SND_SOC_DAPM_PRE_PMU |
5091 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD),
5092
5093 SND_SOC_DAPM_MUX("SLIM TX1 MUX", SND_SOC_NOPM, 0, 0, &sb_tx1_mux),
Kiran Kandi1e6371d2012-03-29 11:48:57 -07005094 SND_SOC_DAPM_MUX("SLIM TX2 MUX", SND_SOC_NOPM, 0, 0, &sb_tx2_mux),
Kiran Kandi1e6371d2012-03-29 11:48:57 -07005095 SND_SOC_DAPM_MUX("SLIM TX3 MUX", SND_SOC_NOPM, 0, 0, &sb_tx3_mux),
Kiran Kandi1e6371d2012-03-29 11:48:57 -07005096 SND_SOC_DAPM_MUX("SLIM TX4 MUX", SND_SOC_NOPM, 0, 0, &sb_tx4_mux),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005097 SND_SOC_DAPM_MUX("SLIM TX5 MUX", SND_SOC_NOPM, 0, 0, &sb_tx5_mux),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005098 SND_SOC_DAPM_MUX("SLIM TX6 MUX", SND_SOC_NOPM, 0, 0, &sb_tx6_mux),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005099 SND_SOC_DAPM_MUX("SLIM TX7 MUX", SND_SOC_NOPM, 0, 0, &sb_tx7_mux),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005100 SND_SOC_DAPM_MUX("SLIM TX8 MUX", SND_SOC_NOPM, 0, 0, &sb_tx8_mux),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005101 SND_SOC_DAPM_MUX("SLIM TX9 MUX", SND_SOC_NOPM, 0, 0, &sb_tx9_mux),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005102 SND_SOC_DAPM_MUX("SLIM TX10 MUX", SND_SOC_NOPM, 0, 0, &sb_tx10_mux),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005103
5104 /* Digital Mic Inputs */
5105 SND_SOC_DAPM_ADC_E("DMIC1", NULL, SND_SOC_NOPM, 0, 0,
5106 tabla_codec_enable_dmic, SND_SOC_DAPM_PRE_PMU |
5107 SND_SOC_DAPM_POST_PMD),
5108
5109 SND_SOC_DAPM_ADC_E("DMIC2", NULL, SND_SOC_NOPM, 0, 0,
5110 tabla_codec_enable_dmic, SND_SOC_DAPM_PRE_PMU |
5111 SND_SOC_DAPM_POST_PMD),
5112
5113 SND_SOC_DAPM_ADC_E("DMIC3", NULL, SND_SOC_NOPM, 0, 0,
5114 tabla_codec_enable_dmic, SND_SOC_DAPM_PRE_PMU |
5115 SND_SOC_DAPM_POST_PMD),
5116
5117 SND_SOC_DAPM_ADC_E("DMIC4", NULL, SND_SOC_NOPM, 0, 0,
5118 tabla_codec_enable_dmic, SND_SOC_DAPM_PRE_PMU |
5119 SND_SOC_DAPM_POST_PMD),
5120
5121 SND_SOC_DAPM_ADC_E("DMIC5", NULL, SND_SOC_NOPM, 0, 0,
5122 tabla_codec_enable_dmic, SND_SOC_DAPM_PRE_PMU |
5123 SND_SOC_DAPM_POST_PMD),
5124 SND_SOC_DAPM_ADC_E("DMIC6", NULL, SND_SOC_NOPM, 0, 0,
5125 tabla_codec_enable_dmic, SND_SOC_DAPM_PRE_PMU |
5126 SND_SOC_DAPM_POST_PMD),
5127
5128 /* Sidetone */
5129 SND_SOC_DAPM_MUX("IIR1 INP1 MUX", SND_SOC_NOPM, 0, 0, &iir1_inp1_mux),
5130 SND_SOC_DAPM_PGA("IIR1", TABLA_A_CDC_CLK_SD_CTL, 0, 0, NULL, 0),
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08005131
5132 /* AUX PGA */
5133 SND_SOC_DAPM_ADC_E("AUX_PGA_Left", NULL, TABLA_A_AUX_L_EN, 7, 0,
5134 tabla_codec_enable_aux_pga, SND_SOC_DAPM_PRE_PMU |
5135 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD |
5136 SND_SOC_DAPM_POST_PMD),
5137
5138 SND_SOC_DAPM_ADC_E("AUX_PGA_Right", NULL, TABLA_A_AUX_R_EN, 7, 0,
5139 tabla_codec_enable_aux_pga, SND_SOC_DAPM_PRE_PMU |
5140 SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD |
5141 SND_SOC_DAPM_POST_PMD),
5142
5143 /* Lineout, ear and HPH PA Mixers */
5144 SND_SOC_DAPM_MIXER("HPHL_PA_MIXER", SND_SOC_NOPM, 0, 0,
5145 hphl_pa_mix, ARRAY_SIZE(hphl_pa_mix)),
5146
5147 SND_SOC_DAPM_MIXER("HPHR_PA_MIXER", SND_SOC_NOPM, 0, 0,
5148 hphr_pa_mix, ARRAY_SIZE(hphr_pa_mix)),
5149
5150 SND_SOC_DAPM_MIXER("LINEOUT1_PA_MIXER", SND_SOC_NOPM, 0, 0,
5151 lineout1_pa_mix, ARRAY_SIZE(lineout1_pa_mix)),
5152
5153 SND_SOC_DAPM_MIXER("LINEOUT2_PA_MIXER", SND_SOC_NOPM, 0, 0,
5154 lineout2_pa_mix, ARRAY_SIZE(lineout2_pa_mix)),
5155
5156 SND_SOC_DAPM_MIXER("LINEOUT3_PA_MIXER", SND_SOC_NOPM, 0, 0,
5157 lineout3_pa_mix, ARRAY_SIZE(lineout3_pa_mix)),
5158
5159 SND_SOC_DAPM_MIXER("LINEOUT4_PA_MIXER", SND_SOC_NOPM, 0, 0,
5160 lineout4_pa_mix, ARRAY_SIZE(lineout4_pa_mix)),
5161
5162 SND_SOC_DAPM_MIXER("LINEOUT5_PA_MIXER", SND_SOC_NOPM, 0, 0,
5163 lineout5_pa_mix, ARRAY_SIZE(lineout5_pa_mix)),
5164
5165 SND_SOC_DAPM_MIXER("EAR_PA_MIXER", SND_SOC_NOPM, 0, 0,
5166 ear_pa_mix, ARRAY_SIZE(ear_pa_mix)),
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08005167};
5168
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005169static short tabla_codec_read_sta_result(struct snd_soc_codec *codec)
Bradley Rubincb1e2732011-06-23 16:49:20 -07005170{
5171 u8 bias_msb, bias_lsb;
5172 short bias_value;
5173
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005174 bias_msb = snd_soc_read(codec, TABLA_A_CDC_MBHC_B3_STATUS);
5175 bias_lsb = snd_soc_read(codec, TABLA_A_CDC_MBHC_B2_STATUS);
5176 bias_value = (bias_msb << 8) | bias_lsb;
5177 return bias_value;
5178}
5179
5180static short tabla_codec_read_dce_result(struct snd_soc_codec *codec)
5181{
5182 u8 bias_msb, bias_lsb;
5183 short bias_value;
5184
5185 bias_msb = snd_soc_read(codec, TABLA_A_CDC_MBHC_B5_STATUS);
5186 bias_lsb = snd_soc_read(codec, TABLA_A_CDC_MBHC_B4_STATUS);
5187 bias_value = (bias_msb << 8) | bias_lsb;
5188 return bias_value;
5189}
5190
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005191static void tabla_turn_onoff_rel_detection(struct snd_soc_codec *codec, bool on)
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005192{
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005193 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x02, on << 1);
5194}
5195
5196static short __tabla_codec_sta_dce(struct snd_soc_codec *codec, int dce,
5197 bool override_bypass, bool noreldetection)
5198{
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005199 short bias_value;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005200 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
5201
5202 wcd9xxx_disable_irq(codec->control_data, TABLA_IRQ_MBHC_POTENTIAL);
5203 if (noreldetection)
5204 tabla_turn_onoff_rel_detection(codec, false);
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005205
Joonwoo Park925914c2012-01-05 13:35:18 -08005206 /* Turn on the override */
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005207 if (!override_bypass)
5208 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x4, 0x4);
Bradley Rubincb1e2732011-06-23 16:49:20 -07005209 if (dce) {
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005210 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x8);
5211 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x4);
5212 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x0);
Joonwoo Park433149a2012-01-11 09:53:54 -08005213 usleep_range(tabla->mbhc_data.t_sta_dce,
5214 tabla->mbhc_data.t_sta_dce);
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005215 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x4);
Joonwoo Park0976d012011-12-22 11:48:18 -08005216 usleep_range(tabla->mbhc_data.t_dce,
5217 tabla->mbhc_data.t_dce);
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005218 bias_value = tabla_codec_read_dce_result(codec);
Bradley Rubincb1e2732011-06-23 16:49:20 -07005219 } else {
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005220 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x8);
Bradley Rubincb1e2732011-06-23 16:49:20 -07005221 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x2);
5222 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x0);
Joonwoo Park433149a2012-01-11 09:53:54 -08005223 usleep_range(tabla->mbhc_data.t_sta_dce,
5224 tabla->mbhc_data.t_sta_dce);
Joonwoo Park0976d012011-12-22 11:48:18 -08005225 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x2);
5226 usleep_range(tabla->mbhc_data.t_sta,
5227 tabla->mbhc_data.t_sta);
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005228 bias_value = tabla_codec_read_sta_result(codec);
5229 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x8);
5230 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x0);
Bradley Rubincb1e2732011-06-23 16:49:20 -07005231 }
Joonwoo Park925914c2012-01-05 13:35:18 -08005232 /* Turn off the override after measuring mic voltage */
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005233 if (!override_bypass)
5234 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x04, 0x00);
5235
5236 if (noreldetection)
5237 tabla_turn_onoff_rel_detection(codec, true);
5238 wcd9xxx_enable_irq(codec->control_data, TABLA_IRQ_MBHC_POTENTIAL);
Bradley Rubincb1e2732011-06-23 16:49:20 -07005239
Bradley Rubincb1e2732011-06-23 16:49:20 -07005240 return bias_value;
5241}
5242
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005243static short tabla_codec_sta_dce(struct snd_soc_codec *codec, int dce,
5244 bool norel)
5245{
5246 return __tabla_codec_sta_dce(codec, dce, false, norel);
5247}
5248
5249/* called only from interrupt which is under codec_resource_lock acquisition */
Bhalchandra Gajare343cbb02011-09-07 18:58:19 -07005250static short tabla_codec_setup_hs_polling(struct snd_soc_codec *codec)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005251{
5252 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Bhalchandra Gajare343cbb02011-09-07 18:58:19 -07005253 short bias_value;
Bhalchandra Gajare19d9c132011-11-18 14:57:08 -08005254 u8 cfilt_mode;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005255
Joonwoo Parkcf473b42012-03-29 19:48:16 -07005256 pr_debug("%s: enter, mclk_enabled %d\n", __func__, tabla->mclk_enabled);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005257 if (!tabla->mbhc_cfg.calibration) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005258 pr_err("Error, no tabla calibration\n");
Bradley Rubincb1e2732011-06-23 16:49:20 -07005259 return -ENODEV;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005260 }
5261
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07005262 if (!tabla->mclk_enabled) {
Joonwoo Park8a6bccc2012-05-03 15:13:13 -07005263 tabla_codec_disable_clock_block(codec);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005264 tabla_codec_enable_bandgap(codec, TABLA_BANDGAP_MBHC_MODE);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07005265 tabla_enable_rx_bias(codec, 1);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005266 tabla_codec_enable_clock_block(codec, 1);
5267 }
5268
5269 snd_soc_update_bits(codec, TABLA_A_CLK_BUFF_EN1, 0x05, 0x01);
5270
Bhalchandra Gajare19d9c132011-11-18 14:57:08 -08005271 /* Make sure CFILT is in fast mode, save current mode */
Joonwoo Parkf4267c22012-01-10 13:25:24 -08005272 cfilt_mode = snd_soc_read(codec, tabla->mbhc_bias_regs.cfilt_ctl);
5273 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.cfilt_ctl, 0x70, 0x00);
Patrick Lai3043fba2011-08-01 14:15:57 -07005274
Joonwoo Parkf4267c22012-01-10 13:25:24 -08005275 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.ctl_reg, 0x1F, 0x16);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005276
5277 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x2, 0x2);
Bradley Rubincb1e2732011-06-23 16:49:20 -07005278 snd_soc_write(codec, TABLA_A_MBHC_SCALING_MUX_1, 0x84);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005279
5280 snd_soc_update_bits(codec, TABLA_A_TX_7_MBHC_EN, 0x80, 0x80);
5281 snd_soc_update_bits(codec, TABLA_A_TX_7_MBHC_EN, 0x1F, 0x1C);
5282 snd_soc_update_bits(codec, TABLA_A_TX_7_MBHC_TEST_CTL, 0x40, 0x40);
5283
5284 snd_soc_update_bits(codec, TABLA_A_TX_7_MBHC_EN, 0x80, 0x00);
Bradley Rubincb1e2732011-06-23 16:49:20 -07005285 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x8);
5286 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x00);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005287
Joonwoo Park925914c2012-01-05 13:35:18 -08005288 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x2, 0x2);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005289 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x8, 0x8);
5290
Bradley Rubincb1e2732011-06-23 16:49:20 -07005291 tabla_codec_calibrate_hs_polling(codec);
5292
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005293 /* don't flip override */
5294 bias_value = __tabla_codec_sta_dce(codec, 1, true, true);
Joonwoo Park0976d012011-12-22 11:48:18 -08005295 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.cfilt_ctl, 0x40,
5296 cfilt_mode);
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07005297 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x13, 0x00);
Bradley Rubincb1e2732011-06-23 16:49:20 -07005298
Bhalchandra Gajare343cbb02011-09-07 18:58:19 -07005299 return bias_value;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005300}
5301
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005302static int tabla_cancel_btn_work(struct tabla_priv *tabla)
5303{
5304 int r = 0;
5305 struct wcd9xxx *core = dev_get_drvdata(tabla->codec->dev->parent);
5306
5307 if (cancel_delayed_work_sync(&tabla->mbhc_btn_dwork)) {
5308 /* if scheduled mbhc_btn_dwork is canceled from here,
5309 * we have to unlock from here instead btn_work */
5310 wcd9xxx_unlock_sleep(core);
5311 r = 1;
5312 }
5313 return r;
5314}
5315
5316/* called under codec_resource_lock acquisition */
5317void tabla_set_and_turnoff_hph_padac(struct snd_soc_codec *codec)
Joonwoo Park03324832012-03-19 19:36:16 -07005318{
5319 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005320 u8 wg_time;
5321
5322 wg_time = snd_soc_read(codec, TABLA_A_RX_HPH_CNP_WG_TIME) ;
5323 wg_time += 1;
Joonwoo Park03324832012-03-19 19:36:16 -07005324
5325 /* If headphone PA is on, check if userspace receives
5326 * removal event to sync-up PA's state */
5327 if (tabla_is_hph_pa_on(codec)) {
5328 pr_debug("%s PA is on, setting PA_OFF_ACK\n", __func__);
5329 set_bit(TABLA_HPHL_PA_OFF_ACK, &tabla->hph_pa_dac_state);
5330 set_bit(TABLA_HPHR_PA_OFF_ACK, &tabla->hph_pa_dac_state);
5331 } else {
5332 pr_debug("%s PA is off\n", __func__);
5333 }
5334
5335 if (tabla_is_hph_dac_on(codec, 1))
5336 set_bit(TABLA_HPHL_DAC_OFF_ACK, &tabla->hph_pa_dac_state);
5337 if (tabla_is_hph_dac_on(codec, 0))
5338 set_bit(TABLA_HPHR_DAC_OFF_ACK, &tabla->hph_pa_dac_state);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005339
5340 snd_soc_update_bits(codec, TABLA_A_RX_HPH_CNP_EN, 0x30, 0x00);
5341 snd_soc_update_bits(codec, TABLA_A_RX_HPH_L_DAC_CTL,
5342 0xC0, 0x00);
5343 snd_soc_update_bits(codec, TABLA_A_RX_HPH_R_DAC_CTL,
5344 0xC0, 0x00);
5345 usleep_range(wg_time * 1000, wg_time * 1000);
5346}
5347
5348static void tabla_clr_and_turnon_hph_padac(struct tabla_priv *tabla)
5349{
5350 bool pa_turned_on = false;
5351 struct snd_soc_codec *codec = tabla->codec;
5352 u8 wg_time;
5353
5354 wg_time = snd_soc_read(codec, TABLA_A_RX_HPH_CNP_WG_TIME) ;
5355 wg_time += 1;
5356
5357 if (test_and_clear_bit(TABLA_HPHR_DAC_OFF_ACK,
5358 &tabla->hph_pa_dac_state)) {
5359 pr_debug("%s: HPHR clear flag and enable DAC\n", __func__);
5360 snd_soc_update_bits(tabla->codec, TABLA_A_RX_HPH_R_DAC_CTL,
5361 0xC0, 0xC0);
5362 }
5363 if (test_and_clear_bit(TABLA_HPHL_DAC_OFF_ACK,
5364 &tabla->hph_pa_dac_state)) {
5365 pr_debug("%s: HPHL clear flag and enable DAC\n", __func__);
5366 snd_soc_update_bits(tabla->codec, TABLA_A_RX_HPH_L_DAC_CTL,
5367 0xC0, 0xC0);
5368 }
5369
5370 if (test_and_clear_bit(TABLA_HPHR_PA_OFF_ACK,
5371 &tabla->hph_pa_dac_state)) {
5372 pr_debug("%s: HPHR clear flag and enable PA\n", __func__);
5373 snd_soc_update_bits(tabla->codec, TABLA_A_RX_HPH_CNP_EN, 0x10,
5374 1 << 4);
5375 pa_turned_on = true;
5376 }
5377 if (test_and_clear_bit(TABLA_HPHL_PA_OFF_ACK,
5378 &tabla->hph_pa_dac_state)) {
5379 pr_debug("%s: HPHL clear flag and enable PA\n", __func__);
5380 snd_soc_update_bits(tabla->codec, TABLA_A_RX_HPH_CNP_EN, 0x20,
5381 1 << 5);
5382 pa_turned_on = true;
5383 }
5384
5385 if (pa_turned_on) {
5386 pr_debug("%s: PA was turned off by MBHC and not by DAPM\n",
5387 __func__);
5388 usleep_range(wg_time * 1000, wg_time * 1000);
5389 }
5390}
5391
5392/* called under codec_resource_lock acquisition */
5393static void tabla_codec_report_plug(struct snd_soc_codec *codec, int insertion,
5394 enum snd_jack_types jack_type)
5395{
5396 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07005397 pr_debug("%s: enter insertion %d hph_status %x\n",
5398 __func__, insertion, tabla->hph_status);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005399 if (!insertion) {
5400 /* Report removal */
5401 tabla->hph_status &= ~jack_type;
5402 if (tabla->mbhc_cfg.headset_jack) {
5403 /* cancel possibly scheduled btn work and
5404 * report release if we reported button press */
5405 if (tabla_cancel_btn_work(tabla)) {
5406 pr_debug("%s: button press is canceled\n",
5407 __func__);
5408 } else if (tabla->buttons_pressed) {
5409 pr_debug("%s: Reporting release for reported "
5410 "button press %d\n", __func__,
5411 jack_type);
5412 tabla_snd_soc_jack_report(tabla,
5413 tabla->mbhc_cfg.button_jack, 0,
5414 tabla->buttons_pressed);
5415 tabla->buttons_pressed &=
5416 ~TABLA_JACK_BUTTON_MASK;
5417 }
Joonwoo Park2cc13f02012-05-09 12:44:25 -07005418 pr_debug("%s: Reporting removal %d(%x)\n", __func__,
5419 jack_type, tabla->hph_status);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005420 tabla_snd_soc_jack_report(tabla,
5421 tabla->mbhc_cfg.headset_jack,
5422 tabla->hph_status,
5423 TABLA_JACK_MASK);
5424 }
5425 tabla_set_and_turnoff_hph_padac(codec);
5426 hphocp_off_report(tabla, SND_JACK_OC_HPHR,
5427 TABLA_IRQ_HPH_PA_OCPR_FAULT);
5428 hphocp_off_report(tabla, SND_JACK_OC_HPHL,
5429 TABLA_IRQ_HPH_PA_OCPL_FAULT);
5430 tabla->current_plug = PLUG_TYPE_NONE;
5431 tabla->mbhc_polling_active = false;
5432 } else {
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07005433 if (tabla->mbhc_cfg.detect_extn_cable) {
5434 /* Report removal of current jack type */
5435 if (tabla->hph_status != jack_type &&
5436 tabla->mbhc_cfg.headset_jack) {
5437 pr_debug("%s: Reporting removal (%x)\n",
5438 __func__, tabla->hph_status);
5439 tabla_snd_soc_jack_report(tabla,
5440 tabla->mbhc_cfg.headset_jack,
5441 0, TABLA_JACK_MASK);
5442 tabla->hph_status = 0;
5443 }
5444 }
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005445 /* Report insertion */
5446 tabla->hph_status |= jack_type;
5447
5448 if (jack_type == SND_JACK_HEADPHONE)
5449 tabla->current_plug = PLUG_TYPE_HEADPHONE;
Joonwoo Park2cc13f02012-05-09 12:44:25 -07005450 else if (jack_type == SND_JACK_UNSUPPORTED)
5451 tabla->current_plug = PLUG_TYPE_GND_MIC_SWAP;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005452 else if (jack_type == SND_JACK_HEADSET) {
5453 tabla->mbhc_polling_active = true;
5454 tabla->current_plug = PLUG_TYPE_HEADSET;
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07005455 } else if (jack_type == SND_JACK_LINEOUT)
5456 tabla->current_plug = PLUG_TYPE_HIGH_HPH;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005457 if (tabla->mbhc_cfg.headset_jack) {
Joonwoo Park2cc13f02012-05-09 12:44:25 -07005458 pr_debug("%s: Reporting insertion %d(%x)\n", __func__,
5459 jack_type, tabla->hph_status);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005460 tabla_snd_soc_jack_report(tabla,
5461 tabla->mbhc_cfg.headset_jack,
5462 tabla->hph_status,
5463 TABLA_JACK_MASK);
5464 }
5465 tabla_clr_and_turnon_hph_padac(tabla);
5466 }
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07005467 pr_debug("%s: leave hph_status %x\n", __func__, tabla->hph_status);
Joonwoo Park03324832012-03-19 19:36:16 -07005468}
5469
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005470static int tabla_codec_enable_hs_detect(struct snd_soc_codec *codec,
Joonwoo Park03324832012-03-19 19:36:16 -07005471 int insertion, int trigger,
5472 bool padac_off)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005473{
5474 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005475 int central_bias_enabled = 0;
Joonwoo Park0976d012011-12-22 11:48:18 -08005476 const struct tabla_mbhc_general_cfg *generic =
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005477 TABLA_MBHC_CAL_GENERAL_PTR(tabla->mbhc_cfg.calibration);
Joonwoo Park0976d012011-12-22 11:48:18 -08005478 const struct tabla_mbhc_plug_detect_cfg *plug_det =
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005479 TABLA_MBHC_CAL_PLUG_DET_PTR(tabla->mbhc_cfg.calibration);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005480
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07005481 pr_debug("%s: enter insertion(%d) trigger(0x%x)\n",
5482 __func__, insertion, trigger);
5483
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005484 if (!tabla->mbhc_cfg.calibration) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005485 pr_err("Error, no tabla calibration\n");
5486 return -EINVAL;
5487 }
5488
5489 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_INT_CTL, 0x1, 0);
5490
Joonwoo Park03324832012-03-19 19:36:16 -07005491 /* Make sure mic bias and Mic line schmitt trigger
5492 * are turned OFF
5493 */
5494 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.ctl_reg, 0x01, 0x01);
5495 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg, 0x90, 0x00);
5496
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005497 if (insertion) {
Bhalchandra Gajare18d10ee2012-08-23 13:44:07 -07005498 pr_debug("%s: setup for insertion\n", __func__);
Joonwoo Park03324832012-03-19 19:36:16 -07005499 tabla_codec_switch_micbias(codec, 0);
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005500
Joonwoo Park03324832012-03-19 19:36:16 -07005501 /* DAPM can manipulate PA/DAC bits concurrently */
5502 if (padac_off == true) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005503 tabla_set_and_turnoff_hph_padac(codec);
Joonwoo Park03324832012-03-19 19:36:16 -07005504 }
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005505
Joonwoo Parkcf473b42012-03-29 19:48:16 -07005506 if (trigger & MBHC_USE_HPHL_TRIGGER) {
Joonwoo Park03324832012-03-19 19:36:16 -07005507 /* Enable HPH Schmitt Trigger */
5508 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x11,
5509 0x11);
5510 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x0C,
5511 plug_det->hph_current << 2);
5512 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x02,
5513 0x02);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07005514 }
5515 if (trigger & MBHC_USE_MB_TRIGGER) {
Joonwoo Park03324832012-03-19 19:36:16 -07005516 /* enable the mic line schmitt trigger */
5517 snd_soc_update_bits(codec,
5518 tabla->mbhc_bias_regs.mbhc_reg,
5519 0x60, plug_det->mic_current << 5);
5520 snd_soc_update_bits(codec,
5521 tabla->mbhc_bias_regs.mbhc_reg,
5522 0x80, 0x80);
5523 usleep_range(plug_det->t_mic_pid, plug_det->t_mic_pid);
5524 snd_soc_update_bits(codec,
5525 tabla->mbhc_bias_regs.ctl_reg, 0x01,
5526 0x00);
5527 snd_soc_update_bits(codec,
5528 tabla->mbhc_bias_regs.mbhc_reg,
5529 0x10, 0x10);
5530 }
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005531
5532 /* setup for insetion detection */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005533 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_INT_CTL, 0x2, 0);
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005534 } else {
Joonwoo Park03324832012-03-19 19:36:16 -07005535 pr_debug("setup for removal detection\n");
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005536 /* Make sure the HPH schmitt trigger is OFF */
5537 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x12, 0x00);
5538
5539 /* enable the mic line schmitt trigger */
Joonwoo Park03324832012-03-19 19:36:16 -07005540 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.ctl_reg,
5541 0x01, 0x00);
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005542 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg, 0x60,
Joonwoo Park0976d012011-12-22 11:48:18 -08005543 plug_det->mic_current << 5);
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005544 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg,
5545 0x80, 0x80);
Joonwoo Park0976d012011-12-22 11:48:18 -08005546 usleep_range(plug_det->t_mic_pid, plug_det->t_mic_pid);
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005547 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg,
5548 0x10, 0x10);
5549
5550 /* Setup for low power removal detection */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005551 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_INT_CTL, 0x2, 0x2);
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07005552 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005553
5554 if (snd_soc_read(codec, TABLA_A_CDC_MBHC_B1_CTL) & 0x4) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005555 /* called called by interrupt */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005556 if (!(tabla->clock_active)) {
5557 tabla_codec_enable_config_mode(codec, 1);
5558 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL,
Bradley Rubincb1e2732011-06-23 16:49:20 -07005559 0x06, 0);
Joonwoo Park0976d012011-12-22 11:48:18 -08005560 usleep_range(generic->t_shutdown_plug_rem,
5561 generic->t_shutdown_plug_rem);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005562 tabla_codec_enable_config_mode(codec, 0);
5563 } else
5564 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL,
Bradley Rubincb1e2732011-06-23 16:49:20 -07005565 0x06, 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005566 }
5567
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07005568 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.int_rbias, 0x80, 0);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005569
5570 /* If central bandgap disabled */
5571 if (!(snd_soc_read(codec, TABLA_A_PIN_CTL_OE1) & 1)) {
5572 snd_soc_update_bits(codec, TABLA_A_PIN_CTL_OE1, 0x3, 0x3);
Joonwoo Park0976d012011-12-22 11:48:18 -08005573 usleep_range(generic->t_bg_fast_settle,
5574 generic->t_bg_fast_settle);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005575 central_bias_enabled = 1;
5576 }
5577
5578 /* If LDO_H disabled */
5579 if (snd_soc_read(codec, TABLA_A_PIN_CTL_OE0) & 0x80) {
5580 snd_soc_update_bits(codec, TABLA_A_PIN_CTL_OE0, 0x10, 0);
5581 snd_soc_update_bits(codec, TABLA_A_PIN_CTL_OE0, 0x80, 0x80);
Joonwoo Park0976d012011-12-22 11:48:18 -08005582 usleep_range(generic->t_ldoh, generic->t_ldoh);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005583 snd_soc_update_bits(codec, TABLA_A_PIN_CTL_OE0, 0x80, 0);
5584
5585 if (central_bias_enabled)
5586 snd_soc_update_bits(codec, TABLA_A_PIN_CTL_OE1, 0x1, 0);
5587 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005588
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08005589 snd_soc_update_bits(codec, tabla->reg_addr.micb_4_mbhc, 0x3,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005590 tabla->mbhc_cfg.micbias);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005591
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05305592 wcd9xxx_enable_irq(codec->control_data, TABLA_IRQ_MBHC_INSERTION);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005593 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_INT_CTL, 0x1, 0x1);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07005594 pr_debug("%s: leave\n", __func__);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07005595 return 0;
5596}
5597
Joonwoo Park0976d012011-12-22 11:48:18 -08005598static u16 tabla_codec_v_sta_dce(struct snd_soc_codec *codec, bool dce,
5599 s16 vin_mv)
5600{
Joonwoo Park0976d012011-12-22 11:48:18 -08005601 struct tabla_priv *tabla;
Joonwoo Park03324832012-03-19 19:36:16 -07005602 s16 diff, zero;
Joonwoo Park0976d012011-12-22 11:48:18 -08005603 u32 mb_mv, in;
Joonwoo Park03324832012-03-19 19:36:16 -07005604 u16 value;
Joonwoo Park0976d012011-12-22 11:48:18 -08005605
5606 tabla = snd_soc_codec_get_drvdata(codec);
5607 mb_mv = tabla->mbhc_data.micb_mv;
5608
5609 if (mb_mv == 0) {
5610 pr_err("%s: Mic Bias voltage is set to zero\n", __func__);
5611 return -EINVAL;
5612 }
5613
5614 if (dce) {
Joonwoo Park03324832012-03-19 19:36:16 -07005615 diff = (tabla->mbhc_data.dce_mb) - (tabla->mbhc_data.dce_z);
5616 zero = (tabla->mbhc_data.dce_z);
Joonwoo Park0976d012011-12-22 11:48:18 -08005617 } else {
Joonwoo Park03324832012-03-19 19:36:16 -07005618 diff = (tabla->mbhc_data.sta_mb) - (tabla->mbhc_data.sta_z);
5619 zero = (tabla->mbhc_data.sta_z);
Joonwoo Park0976d012011-12-22 11:48:18 -08005620 }
5621 in = (u32) diff * vin_mv;
5622
Joonwoo Park03324832012-03-19 19:36:16 -07005623 value = (u16) (in / mb_mv) + zero;
5624 return value;
Joonwoo Park0976d012011-12-22 11:48:18 -08005625}
5626
5627static s32 tabla_codec_sta_dce_v(struct snd_soc_codec *codec, s8 dce,
5628 u16 bias_value)
5629{
5630 struct tabla_priv *tabla;
Joonwoo Park03324832012-03-19 19:36:16 -07005631 s16 value, z, mb;
Joonwoo Park0976d012011-12-22 11:48:18 -08005632 s32 mv;
5633
5634 tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Park03324832012-03-19 19:36:16 -07005635 value = bias_value;
Joonwoo Park0976d012011-12-22 11:48:18 -08005636 if (dce) {
Joonwoo Park03324832012-03-19 19:36:16 -07005637 z = (tabla->mbhc_data.dce_z);
5638 mb = (tabla->mbhc_data.dce_mb);
5639 mv = (value - z) * (s32)tabla->mbhc_data.micb_mv / (mb - z);
Joonwoo Park0976d012011-12-22 11:48:18 -08005640 } else {
Joonwoo Park03324832012-03-19 19:36:16 -07005641 z = (tabla->mbhc_data.sta_z);
5642 mb = (tabla->mbhc_data.sta_mb);
5643 mv = (value - z) * (s32)tabla->mbhc_data.micb_mv / (mb - z);
Joonwoo Park0976d012011-12-22 11:48:18 -08005644 }
5645
5646 return mv;
5647}
5648
Joonwoo Park03324832012-03-19 19:36:16 -07005649static void btn_lpress_fn(struct work_struct *work)
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07005650{
5651 struct delayed_work *delayed_work;
5652 struct tabla_priv *tabla;
Joonwoo Park0976d012011-12-22 11:48:18 -08005653 short bias_value;
5654 int dce_mv, sta_mv;
Joonwoo Park03324832012-03-19 19:36:16 -07005655 struct wcd9xxx *core;
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07005656
5657 pr_debug("%s:\n", __func__);
5658
5659 delayed_work = to_delayed_work(work);
Joonwoo Park03324832012-03-19 19:36:16 -07005660 tabla = container_of(delayed_work, struct tabla_priv, mbhc_btn_dwork);
Joonwoo Park816b8e62012-01-23 16:03:21 -08005661 core = dev_get_drvdata(tabla->codec->dev->parent);
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07005662
5663 if (tabla) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005664 if (tabla->mbhc_cfg.button_jack) {
Joonwoo Park0976d012011-12-22 11:48:18 -08005665 bias_value = tabla_codec_read_sta_result(tabla->codec);
5666 sta_mv = tabla_codec_sta_dce_v(tabla->codec, 0,
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05305667 bias_value);
Joonwoo Park0976d012011-12-22 11:48:18 -08005668 bias_value = tabla_codec_read_dce_result(tabla->codec);
5669 dce_mv = tabla_codec_sta_dce_v(tabla->codec, 1,
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05305670 bias_value);
Joonwoo Park0976d012011-12-22 11:48:18 -08005671 pr_debug("%s: Reporting long button press event"
5672 " STA: %d, DCE: %d\n", __func__,
5673 sta_mv, dce_mv);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005674 tabla_snd_soc_jack_report(tabla,
5675 tabla->mbhc_cfg.button_jack,
Joonwoo Park03324832012-03-19 19:36:16 -07005676 tabla->buttons_pressed,
5677 tabla->buttons_pressed);
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07005678 }
5679 } else {
5680 pr_err("%s: Bad tabla private data\n", __func__);
5681 }
5682
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005683 pr_debug("%s: leave\n", __func__);
Joonwoo Park03324832012-03-19 19:36:16 -07005684 wcd9xxx_unlock_sleep(core);
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07005685}
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07005686
Joonwoo Parke067b232012-06-14 13:11:30 -07005687static u16 tabla_get_cfilt_reg(struct snd_soc_codec *codec, u8 cfilt)
5688{
5689 u16 reg;
5690
5691 switch (cfilt) {
5692 case TABLA_CFILT1_SEL:
5693 reg = TABLA_A_MICB_CFILT_1_CTL;
5694 break;
5695 case TABLA_CFILT2_SEL:
5696 reg = TABLA_A_MICB_CFILT_2_CTL;
5697 break;
5698 case TABLA_CFILT3_SEL:
5699 reg = TABLA_A_MICB_CFILT_3_CTL;
5700 break;
5701 default:
5702 BUG();
5703 }
5704 return reg;
5705}
5706
Joonwoo Park0976d012011-12-22 11:48:18 -08005707void tabla_mbhc_cal(struct snd_soc_codec *codec)
5708{
5709 struct tabla_priv *tabla;
5710 struct tabla_mbhc_btn_detect_cfg *btn_det;
Joonwoo Parke067b232012-06-14 13:11:30 -07005711 u8 cfilt_mode, micbias2_cfilt_mode, bg_mode;
Joonwoo Park0976d012011-12-22 11:48:18 -08005712 u8 ncic, nmeas, navg;
5713 u32 mclk_rate;
5714 u32 dce_wait, sta_wait;
5715 u8 *n_cic;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005716 void *calibration;
Joonwoo Parke067b232012-06-14 13:11:30 -07005717 u16 bias2_ctl;
Joonwoo Park0976d012011-12-22 11:48:18 -08005718
5719 tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005720 calibration = tabla->mbhc_cfg.calibration;
5721
5722 wcd9xxx_disable_irq(codec->control_data, TABLA_IRQ_MBHC_POTENTIAL);
5723 tabla_turn_onoff_rel_detection(codec, false);
Joonwoo Park0976d012011-12-22 11:48:18 -08005724
5725 /* First compute the DCE / STA wait times
5726 * depending on tunable parameters.
5727 * The value is computed in microseconds
5728 */
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005729 btn_det = TABLA_MBHC_CAL_BTN_DET_PTR(calibration);
Joonwoo Park0976d012011-12-22 11:48:18 -08005730 n_cic = tabla_mbhc_cal_btn_det_mp(btn_det, TABLA_BTN_DET_N_CIC);
Joonwoo Park107edf02012-01-11 11:42:24 -08005731 ncic = n_cic[tabla_codec_mclk_index(tabla)];
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005732 nmeas = TABLA_MBHC_CAL_BTN_DET_PTR(calibration)->n_meas;
5733 navg = TABLA_MBHC_CAL_GENERAL_PTR(calibration)->mbhc_navg;
5734 mclk_rate = tabla->mbhc_cfg.mclk_rate;
Joonwoo Park433149a2012-01-11 09:53:54 -08005735 dce_wait = (1000 * 512 * ncic * (nmeas + 1)) / (mclk_rate / 1000);
5736 sta_wait = (1000 * 128 * (navg + 1)) / (mclk_rate / 1000);
Joonwoo Park0976d012011-12-22 11:48:18 -08005737
5738 tabla->mbhc_data.t_dce = dce_wait;
5739 tabla->mbhc_data.t_sta = sta_wait;
5740
5741 /* LDOH and CFILT are already configured during pdata handling.
5742 * Only need to make sure CFILT and bandgap are in Fast mode.
5743 * Need to restore defaults once calculation is done.
5744 */
5745 cfilt_mode = snd_soc_read(codec, tabla->mbhc_bias_regs.cfilt_ctl);
Joonwoo Parke067b232012-06-14 13:11:30 -07005746 micbias2_cfilt_mode =
5747 snd_soc_read(codec, tabla_get_cfilt_reg(codec,
5748 tabla->pdata->micbias.bias2_cfilt_sel));
5749 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.cfilt_ctl, 0x40,
5750 TABLA_CFILT_FAST_MODE);
5751 snd_soc_update_bits(codec,
5752 tabla_get_cfilt_reg(codec,
5753 tabla->pdata->micbias.bias2_cfilt_sel),
5754 0x40, TABLA_CFILT_FAST_MODE);
5755
Joonwoo Park0976d012011-12-22 11:48:18 -08005756 bg_mode = snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x02,
5757 0x02);
5758
5759 /* Micbias, CFILT, LDOH, MBHC MUX mode settings
5760 * to perform ADC calibration
5761 */
5762 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.ctl_reg, 0x60,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005763 tabla->mbhc_cfg.micbias << 5);
Joonwoo Park0976d012011-12-22 11:48:18 -08005764 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.ctl_reg, 0x01, 0x00);
5765 snd_soc_update_bits(codec, TABLA_A_LDO_H_MODE_1, 0x60, 0x60);
5766 snd_soc_write(codec, TABLA_A_TX_7_MBHC_TEST_CTL, 0x78);
5767 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x04, 0x04);
5768
Joonwoo Parke067b232012-06-14 13:11:30 -07005769 /* MICBIAS2 routing for calibration */
5770 bias2_ctl = snd_soc_read(codec, TABLA_A_MICB_2_CTL);
5771 snd_soc_update_bits(codec, TABLA_A_MICB_1_MBHC, 0x03, TABLA_MICBIAS2);
5772 snd_soc_write(codec, TABLA_A_MICB_2_CTL,
5773 snd_soc_read(codec, tabla->mbhc_bias_regs.ctl_reg));
5774
Joonwoo Park0976d012011-12-22 11:48:18 -08005775 /* DCE measurement for 0 volts */
5776 snd_soc_write(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x0A);
5777 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x04);
5778 snd_soc_write(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x02);
Joonwoo Park0976d012011-12-22 11:48:18 -08005779 snd_soc_write(codec, TABLA_A_MBHC_SCALING_MUX_1, 0x81);
5780 usleep_range(100, 100);
5781 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x04);
5782 usleep_range(tabla->mbhc_data.t_dce, tabla->mbhc_data.t_dce);
5783 tabla->mbhc_data.dce_z = tabla_codec_read_dce_result(codec);
5784
5785 /* DCE measurment for MB voltage */
5786 snd_soc_write(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x0A);
5787 snd_soc_write(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x02);
5788 snd_soc_write(codec, TABLA_A_MBHC_SCALING_MUX_1, 0x82);
5789 usleep_range(100, 100);
5790 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x04);
5791 usleep_range(tabla->mbhc_data.t_dce, tabla->mbhc_data.t_dce);
5792 tabla->mbhc_data.dce_mb = tabla_codec_read_dce_result(codec);
5793
5794 /* Sta measuremnt for 0 volts */
5795 snd_soc_write(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x0A);
5796 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x02);
5797 snd_soc_write(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x02);
Joonwoo Park0976d012011-12-22 11:48:18 -08005798 snd_soc_write(codec, TABLA_A_MBHC_SCALING_MUX_1, 0x81);
5799 usleep_range(100, 100);
5800 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x02);
5801 usleep_range(tabla->mbhc_data.t_sta, tabla->mbhc_data.t_sta);
5802 tabla->mbhc_data.sta_z = tabla_codec_read_sta_result(codec);
5803
5804 /* STA Measurement for MB Voltage */
5805 snd_soc_write(codec, TABLA_A_MBHC_SCALING_MUX_1, 0x82);
5806 usleep_range(100, 100);
5807 snd_soc_write(codec, TABLA_A_CDC_MBHC_EN_CTL, 0x02);
5808 usleep_range(tabla->mbhc_data.t_sta, tabla->mbhc_data.t_sta);
5809 tabla->mbhc_data.sta_mb = tabla_codec_read_sta_result(codec);
5810
5811 /* Restore default settings. */
Joonwoo Parke067b232012-06-14 13:11:30 -07005812 snd_soc_write(codec, TABLA_A_MICB_2_CTL, bias2_ctl);
5813 snd_soc_update_bits(codec, TABLA_A_MICB_1_MBHC, 0x03,
5814 tabla->mbhc_cfg.micbias);
5815
Joonwoo Park0976d012011-12-22 11:48:18 -08005816 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x04, 0x00);
Joonwoo Parke067b232012-06-14 13:11:30 -07005817 snd_soc_update_bits(codec,
5818 tabla_get_cfilt_reg(codec,
5819 tabla->pdata->micbias.bias2_cfilt_sel), 0x40,
5820 micbias2_cfilt_mode);
Joonwoo Park0976d012011-12-22 11:48:18 -08005821 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.cfilt_ctl, 0x40,
5822 cfilt_mode);
5823 snd_soc_update_bits(codec, TABLA_A_BIAS_CENTRAL_BG_CTL, 0x02, bg_mode);
5824
5825 snd_soc_write(codec, TABLA_A_MBHC_SCALING_MUX_1, 0x84);
5826 usleep_range(100, 100);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005827
5828 wcd9xxx_enable_irq(codec->control_data, TABLA_IRQ_MBHC_POTENTIAL);
5829 tabla_turn_onoff_rel_detection(codec, true);
Joonwoo Park0976d012011-12-22 11:48:18 -08005830}
5831
5832void *tabla_mbhc_cal_btn_det_mp(const struct tabla_mbhc_btn_detect_cfg* btn_det,
5833 const enum tabla_mbhc_btn_det_mem mem)
5834{
5835 void *ret = &btn_det->_v_btn_low;
5836
5837 switch (mem) {
5838 case TABLA_BTN_DET_GAIN:
5839 ret += sizeof(btn_det->_n_cic);
5840 case TABLA_BTN_DET_N_CIC:
5841 ret += sizeof(btn_det->_n_ready);
Joonwoo Parkc0672392012-01-11 11:03:14 -08005842 case TABLA_BTN_DET_N_READY:
Joonwoo Park0976d012011-12-22 11:48:18 -08005843 ret += sizeof(btn_det->_v_btn_high[0]) * btn_det->num_btn;
5844 case TABLA_BTN_DET_V_BTN_HIGH:
5845 ret += sizeof(btn_det->_v_btn_low[0]) * btn_det->num_btn;
5846 case TABLA_BTN_DET_V_BTN_LOW:
5847 /* do nothing */
5848 break;
5849 default:
5850 ret = NULL;
5851 }
5852
5853 return ret;
5854}
5855
Joonwoo Parkcf473b42012-03-29 19:48:16 -07005856static s16 tabla_scale_v_micb_vddio(struct tabla_priv *tabla, int v,
5857 bool tovddio)
5858{
5859 int r;
5860 int vddio_k, mb_k;
5861 vddio_k = tabla_find_k_value(tabla->pdata->micbias.ldoh_v,
5862 VDDIO_MICBIAS_MV);
5863 mb_k = tabla_find_k_value(tabla->pdata->micbias.ldoh_v,
5864 tabla->mbhc_data.micb_mv);
5865 if (tovddio)
5866 r = v * vddio_k / mb_k;
5867 else
5868 r = v * mb_k / vddio_k;
5869 return r;
5870}
5871
Joonwoo Parkdd9d2962012-07-23 19:24:20 -07005872static void tabla_mbhc_calc_rel_thres(struct snd_soc_codec *codec, s16 mv)
5873{
5874 s16 deltamv;
5875 struct tabla_priv *tabla;
5876 struct tabla_mbhc_btn_detect_cfg *btn_det;
5877
5878 tabla = snd_soc_codec_get_drvdata(codec);
5879 btn_det = TABLA_MBHC_CAL_BTN_DET_PTR(tabla->mbhc_cfg.calibration);
5880
5881 tabla->mbhc_data.v_b1_h =
5882 tabla_codec_v_sta_dce(codec, DCE,
5883 mv + btn_det->v_btn_press_delta_cic);
5884
5885 tabla->mbhc_data.v_brh = tabla->mbhc_data.v_b1_h;
5886
5887 tabla->mbhc_data.v_brl = TABLA_MBHC_BUTTON_MIN;
5888
5889 deltamv = mv + btn_det->v_btn_press_delta_sta;
5890 tabla->mbhc_data.v_b1_hu = tabla_codec_v_sta_dce(codec, STA, deltamv);
5891
5892 deltamv = mv + btn_det->v_btn_press_delta_cic;
5893 tabla->mbhc_data.v_b1_huc = tabla_codec_v_sta_dce(codec, DCE, deltamv);
5894}
5895
5896static void tabla_mbhc_set_rel_thres(struct snd_soc_codec *codec, s16 mv)
5897{
5898 tabla_mbhc_calc_rel_thres(codec, mv);
5899 tabla_codec_calibrate_rel(codec);
5900}
5901
5902static s16 tabla_mbhc_highest_btn_mv(struct snd_soc_codec *codec)
5903{
5904 struct tabla_priv *tabla;
5905 struct tabla_mbhc_btn_detect_cfg *btn_det;
5906 u16 *btn_high;
5907
5908 tabla = snd_soc_codec_get_drvdata(codec);
5909 btn_det = TABLA_MBHC_CAL_BTN_DET_PTR(tabla->mbhc_cfg.calibration);
5910 btn_high = tabla_mbhc_cal_btn_det_mp(btn_det, TABLA_BTN_DET_V_BTN_HIGH);
5911
5912 return btn_high[btn_det->num_btn - 1];
5913}
5914
Joonwoo Park0976d012011-12-22 11:48:18 -08005915static void tabla_mbhc_calc_thres(struct snd_soc_codec *codec)
5916{
5917 struct tabla_priv *tabla;
Joonwoo Park0976d012011-12-22 11:48:18 -08005918 struct tabla_mbhc_btn_detect_cfg *btn_det;
5919 struct tabla_mbhc_plug_type_cfg *plug_type;
Joonwoo Parkc0672392012-01-11 11:03:14 -08005920 u8 *n_ready;
Joonwoo Park0976d012011-12-22 11:48:18 -08005921
5922 tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005923 btn_det = TABLA_MBHC_CAL_BTN_DET_PTR(tabla->mbhc_cfg.calibration);
5924 plug_type = TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla->mbhc_cfg.calibration);
Joonwoo Park0976d012011-12-22 11:48:18 -08005925
Joonwoo Parkc0672392012-01-11 11:03:14 -08005926 n_ready = tabla_mbhc_cal_btn_det_mp(btn_det, TABLA_BTN_DET_N_READY);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005927 if (tabla->mbhc_cfg.mclk_rate == TABLA_MCLK_RATE_12288KHZ) {
Joonwoo Park03324832012-03-19 19:36:16 -07005928 tabla->mbhc_data.npoll = 4;
Joonwoo Park0976d012011-12-22 11:48:18 -08005929 tabla->mbhc_data.nbounce_wait = 30;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005930 } else if (tabla->mbhc_cfg.mclk_rate == TABLA_MCLK_RATE_9600KHZ) {
Joonwoo Park0976d012011-12-22 11:48:18 -08005931 tabla->mbhc_data.npoll = 7;
5932 tabla->mbhc_data.nbounce_wait = 23;
Joonwoo Parkc0672392012-01-11 11:03:14 -08005933 }
Joonwoo Park0976d012011-12-22 11:48:18 -08005934
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005935 tabla->mbhc_data.t_sta_dce = ((1000 * 256) /
5936 (tabla->mbhc_cfg.mclk_rate / 1000) *
Joonwoo Parkc0672392012-01-11 11:03:14 -08005937 n_ready[tabla_codec_mclk_index(tabla)]) +
5938 10;
Joonwoo Park0976d012011-12-22 11:48:18 -08005939 tabla->mbhc_data.v_ins_hu =
5940 tabla_codec_v_sta_dce(codec, STA, plug_type->v_hs_max);
5941 tabla->mbhc_data.v_ins_h =
5942 tabla_codec_v_sta_dce(codec, DCE, plug_type->v_hs_max);
5943
Joonwoo Parkcf473b42012-03-29 19:48:16 -07005944 tabla->mbhc_data.v_inval_ins_low = TABLA_MBHC_FAKE_INSERT_LOW;
5945 if (tabla->mbhc_cfg.gpio)
5946 tabla->mbhc_data.v_inval_ins_high =
5947 TABLA_MBHC_FAKE_INSERT_HIGH;
5948 else
5949 tabla->mbhc_data.v_inval_ins_high =
5950 TABLA_MBHC_FAKE_INS_HIGH_NO_GPIO;
5951
5952 if (tabla->mbhc_data.micb_mv != VDDIO_MICBIAS_MV) {
5953 tabla->mbhc_data.adj_v_hs_max =
5954 tabla_scale_v_micb_vddio(tabla, plug_type->v_hs_max, true);
5955 tabla->mbhc_data.adj_v_ins_hu =
5956 tabla_codec_v_sta_dce(codec, STA,
5957 tabla->mbhc_data.adj_v_hs_max);
5958 tabla->mbhc_data.adj_v_ins_h =
5959 tabla_codec_v_sta_dce(codec, DCE,
5960 tabla->mbhc_data.adj_v_hs_max);
5961 tabla->mbhc_data.v_inval_ins_low =
5962 tabla_scale_v_micb_vddio(tabla,
5963 tabla->mbhc_data.v_inval_ins_low,
5964 false);
5965 tabla->mbhc_data.v_inval_ins_high =
5966 tabla_scale_v_micb_vddio(tabla,
5967 tabla->mbhc_data.v_inval_ins_high,
5968 false);
5969 }
5970
Joonwoo Parkdd9d2962012-07-23 19:24:20 -07005971 tabla_mbhc_calc_rel_thres(codec, tabla_mbhc_highest_btn_mv(codec));
Joonwoo Park0976d012011-12-22 11:48:18 -08005972
5973 tabla->mbhc_data.v_no_mic =
5974 tabla_codec_v_sta_dce(codec, STA, plug_type->v_no_mic);
5975}
5976
5977void tabla_mbhc_init(struct snd_soc_codec *codec)
5978{
5979 struct tabla_priv *tabla;
5980 struct tabla_mbhc_general_cfg *generic;
5981 struct tabla_mbhc_btn_detect_cfg *btn_det;
5982 int n;
Joonwoo Park0976d012011-12-22 11:48:18 -08005983 u8 *n_cic, *gain;
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05305984 struct wcd9xxx *tabla_core = dev_get_drvdata(codec->dev->parent);
Joonwoo Park0976d012011-12-22 11:48:18 -08005985
5986 tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07005987 generic = TABLA_MBHC_CAL_GENERAL_PTR(tabla->mbhc_cfg.calibration);
5988 btn_det = TABLA_MBHC_CAL_BTN_DET_PTR(tabla->mbhc_cfg.calibration);
Joonwoo Park0976d012011-12-22 11:48:18 -08005989
Joonwoo Park0976d012011-12-22 11:48:18 -08005990 for (n = 0; n < 8; n++) {
Kuirong Wangcd4b6da2012-01-16 22:54:45 -08005991 if ((!TABLA_IS_1_X(tabla_core->version)) || n != 7) {
Joonwoo Park0976d012011-12-22 11:48:18 -08005992 snd_soc_update_bits(codec,
5993 TABLA_A_CDC_MBHC_FEATURE_B1_CFG,
5994 0x07, n);
5995 snd_soc_write(codec, TABLA_A_CDC_MBHC_FEATURE_B2_CFG,
5996 btn_det->c[n]);
5997 }
5998 }
5999 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B2_CTL, 0x07,
6000 btn_det->nc);
6001
6002 n_cic = tabla_mbhc_cal_btn_det_mp(btn_det, TABLA_BTN_DET_N_CIC);
6003 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_TIMER_B6_CTL, 0xFF,
Joonwoo Park107edf02012-01-11 11:42:24 -08006004 n_cic[tabla_codec_mclk_index(tabla)]);
Joonwoo Park0976d012011-12-22 11:48:18 -08006005
6006 gain = tabla_mbhc_cal_btn_det_mp(btn_det, TABLA_BTN_DET_GAIN);
Joonwoo Park107edf02012-01-11 11:42:24 -08006007 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B2_CTL, 0x78,
6008 gain[tabla_codec_mclk_index(tabla)] << 3);
Joonwoo Park0976d012011-12-22 11:48:18 -08006009
6010 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_TIMER_B4_CTL, 0x70,
6011 generic->mbhc_nsa << 4);
6012
6013 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_TIMER_B4_CTL, 0x0F,
6014 btn_det->n_meas);
6015
6016 snd_soc_write(codec, TABLA_A_CDC_MBHC_TIMER_B5_CTL, generic->mbhc_navg);
6017
6018 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x80, 0x80);
6019
6020 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x78,
6021 btn_det->mbhc_nsc << 3);
6022
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08006023 snd_soc_update_bits(codec, tabla->reg_addr.micb_4_mbhc, 0x03,
6024 TABLA_MICBIAS2);
Joonwoo Park0976d012011-12-22 11:48:18 -08006025
6026 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x02, 0x02);
Joonwoo Park03324832012-03-19 19:36:16 -07006027
6028 snd_soc_update_bits(codec, TABLA_A_MBHC_SCALING_MUX_2, 0xF0, 0xF0);
Joonwoo Parke067b232012-06-14 13:11:30 -07006029
6030 /* override mbhc's micbias */
6031 snd_soc_update_bits(codec, TABLA_A_MICB_1_MBHC, 0x03,
6032 tabla->mbhc_cfg.micbias);
Joonwoo Park0976d012011-12-22 11:48:18 -08006033}
6034
Patrick Lai64b43262011-12-06 17:29:15 -08006035static bool tabla_mbhc_fw_validate(const struct firmware *fw)
6036{
6037 u32 cfg_offset;
6038 struct tabla_mbhc_imped_detect_cfg *imped_cfg;
6039 struct tabla_mbhc_btn_detect_cfg *btn_cfg;
6040
6041 if (fw->size < TABLA_MBHC_CAL_MIN_SIZE)
6042 return false;
6043
6044 /* previous check guarantees that there is enough fw data up
6045 * to num_btn
6046 */
6047 btn_cfg = TABLA_MBHC_CAL_BTN_DET_PTR(fw->data);
6048 cfg_offset = (u32) ((void *) btn_cfg - (void *) fw->data);
6049 if (fw->size < (cfg_offset + TABLA_MBHC_CAL_BTN_SZ(btn_cfg)))
6050 return false;
6051
6052 /* previous check guarantees that there is enough fw data up
6053 * to start of impedance detection configuration
6054 */
6055 imped_cfg = TABLA_MBHC_CAL_IMPED_DET_PTR(fw->data);
6056 cfg_offset = (u32) ((void *) imped_cfg - (void *) fw->data);
6057
6058 if (fw->size < (cfg_offset + TABLA_MBHC_CAL_IMPED_MIN_SZ))
6059 return false;
6060
6061 if (fw->size < (cfg_offset + TABLA_MBHC_CAL_IMPED_SZ(imped_cfg)))
6062 return false;
6063
6064 return true;
6065}
Joonwoo Park03324832012-03-19 19:36:16 -07006066
Joonwoo Parkfee17432012-04-16 16:33:55 -07006067/* called under codec_resource_lock acquisition */
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006068static int tabla_determine_button(const struct tabla_priv *priv,
Joonwoo Parkfee17432012-04-16 16:33:55 -07006069 const s32 micmv)
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006070{
6071 s16 *v_btn_low, *v_btn_high;
6072 struct tabla_mbhc_btn_detect_cfg *btn_det;
6073 int i, btn = -1;
6074
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006075 btn_det = TABLA_MBHC_CAL_BTN_DET_PTR(priv->mbhc_cfg.calibration);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006076 v_btn_low = tabla_mbhc_cal_btn_det_mp(btn_det, TABLA_BTN_DET_V_BTN_LOW);
6077 v_btn_high = tabla_mbhc_cal_btn_det_mp(btn_det,
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05306078 TABLA_BTN_DET_V_BTN_HIGH);
Joonwoo Parkfee17432012-04-16 16:33:55 -07006079
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006080 for (i = 0; i < btn_det->num_btn; i++) {
Joonwoo Parkfee17432012-04-16 16:33:55 -07006081 if ((v_btn_low[i] <= micmv) && (v_btn_high[i] >= micmv)) {
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006082 btn = i;
6083 break;
6084 }
6085 }
6086
6087 if (btn == -1)
6088 pr_debug("%s: couldn't find button number for mic mv %d\n",
Joonwoo Parkfee17432012-04-16 16:33:55 -07006089 __func__, micmv);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006090
6091 return btn;
6092}
6093
6094static int tabla_get_button_mask(const int btn)
6095{
6096 int mask = 0;
6097 switch (btn) {
6098 case 0:
6099 mask = SND_JACK_BTN_0;
6100 break;
6101 case 1:
6102 mask = SND_JACK_BTN_1;
6103 break;
6104 case 2:
6105 mask = SND_JACK_BTN_2;
6106 break;
6107 case 3:
6108 mask = SND_JACK_BTN_3;
6109 break;
6110 case 4:
6111 mask = SND_JACK_BTN_4;
6112 break;
6113 case 5:
6114 mask = SND_JACK_BTN_5;
6115 break;
6116 case 6:
6117 mask = SND_JACK_BTN_6;
6118 break;
6119 case 7:
6120 mask = SND_JACK_BTN_7;
6121 break;
6122 }
6123 return mask;
6124}
6125
Bradley Rubincb1e2732011-06-23 16:49:20 -07006126static irqreturn_t tabla_dce_handler(int irq, void *data)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006127{
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006128 int i, mask;
Joonwoo Parkfee17432012-04-16 16:33:55 -07006129 short dce, sta;
Joonwoo Park12334832012-07-23 19:27:52 -07006130 s32 mv, mv_s, stamv, stamv_s;
Joonwoo Parkfee17432012-04-16 16:33:55 -07006131 bool vddio;
Joonwoo Parkdd9d2962012-07-23 19:24:20 -07006132 u16 *btn_high;
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006133 int btn = -1, meas = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006134 struct tabla_priv *priv = data;
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006135 const struct tabla_mbhc_btn_detect_cfg *d =
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006136 TABLA_MBHC_CAL_BTN_DET_PTR(priv->mbhc_cfg.calibration);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006137 short btnmeas[d->n_btn_meas + 1];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006138 struct snd_soc_codec *codec = priv->codec;
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05306139 struct wcd9xxx *core = dev_get_drvdata(priv->codec->dev->parent);
Joonwoo Park03324832012-03-19 19:36:16 -07006140 int n_btn_meas = d->n_btn_meas;
6141 u8 mbhc_status = snd_soc_read(codec, TABLA_A_CDC_MBHC_B1_STATUS) & 0x3E;
Bradley Rubincb1e2732011-06-23 16:49:20 -07006142
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006143 pr_debug("%s: enter\n", __func__);
Bradley Rubincb1e2732011-06-23 16:49:20 -07006144
Joonwoo Parkdd9d2962012-07-23 19:24:20 -07006145 btn_high = tabla_mbhc_cal_btn_det_mp(d, TABLA_BTN_DET_V_BTN_HIGH);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006146 TABLA_ACQUIRE_LOCK(priv->codec_resource_lock);
6147 if (priv->mbhc_state == MBHC_STATE_POTENTIAL_RECOVERY) {
6148 pr_debug("%s: mbhc is being recovered, skip button press\n",
6149 __func__);
6150 goto done;
6151 }
6152
6153 priv->mbhc_state = MBHC_STATE_POTENTIAL;
6154
6155 if (!priv->mbhc_polling_active) {
6156 pr_warn("%s: mbhc polling is not active, skip button press\n",
6157 __func__);
6158 goto done;
6159 }
Joonwoo Park03324832012-03-19 19:36:16 -07006160
6161 dce = tabla_codec_read_dce_result(codec);
6162 mv = tabla_codec_sta_dce_v(codec, 1, dce);
6163
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006164 /* If GPIO interrupt already kicked in, ignore button press */
6165 if (priv->in_gpio_handler) {
6166 pr_debug("%s: GPIO State Changed, ignore button press\n",
6167 __func__);
6168 btn = -1;
6169 goto done;
6170 }
6171
Joonwoo Parkfee17432012-04-16 16:33:55 -07006172 vddio = (priv->mbhc_data.micb_mv != VDDIO_MICBIAS_MV &&
6173 priv->mbhc_micbias_switched);
6174 mv_s = vddio ? tabla_scale_v_micb_vddio(priv, mv, false) : mv;
6175
Joonwoo Park03324832012-03-19 19:36:16 -07006176 if (mbhc_status != TABLA_MBHC_STATUS_REL_DETECTION) {
6177 if (priv->mbhc_last_resume &&
6178 !time_after(jiffies, priv->mbhc_last_resume + HZ)) {
6179 pr_debug("%s: Button is already released shortly after "
6180 "resume\n", __func__);
6181 n_btn_meas = 0;
Joonwoo Park03324832012-03-19 19:36:16 -07006182 }
6183 }
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07006184
Joonwoo Park12334832012-07-23 19:27:52 -07006185 /* save hw dce */
Joonwoo Parkfee17432012-04-16 16:33:55 -07006186 btnmeas[meas++] = tabla_determine_button(priv, mv_s);
Joonwoo Park12334832012-07-23 19:27:52 -07006187 pr_debug("%s: meas HW - DCE %x,%d,%d button %d\n", __func__,
6188 dce, mv, mv_s, btnmeas[0]);
6189 if (n_btn_meas == 0) {
6190 sta = tabla_codec_read_sta_result(codec);
6191 stamv_s = stamv = tabla_codec_sta_dce_v(codec, 0, sta);
6192 if (vddio)
6193 stamv_s = tabla_scale_v_micb_vddio(priv, stamv, false);
6194 btn = tabla_determine_button(priv, stamv_s);
6195 pr_debug("%s: meas HW - STA %x,%d,%d button %d\n", __func__,
6196 sta, stamv, stamv_s, btn);
6197 BUG_ON(meas != 1);
6198 if (btnmeas[0] != btn)
6199 btn = -1;
6200 }
6201
6202 /* determine pressed button */
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006203 for (; ((d->n_btn_meas) && (meas < (d->n_btn_meas + 1))); meas++) {
Joonwoo Parkfee17432012-04-16 16:33:55 -07006204 dce = tabla_codec_sta_dce(codec, 1, false);
6205 mv = tabla_codec_sta_dce_v(codec, 1, dce);
6206 mv_s = vddio ? tabla_scale_v_micb_vddio(priv, mv, false) : mv;
6207
6208 btnmeas[meas] = tabla_determine_button(priv, mv_s);
Joonwoo Park12334832012-07-23 19:27:52 -07006209 pr_debug("%s: meas %d - DCE %x,%d,%d button %d\n",
Joonwoo Parkfee17432012-04-16 16:33:55 -07006210 __func__, meas, dce, mv, mv_s, btnmeas[meas]);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006211 /* if large enough measurements are collected,
6212 * start to check if last all n_btn_con measurements were
6213 * in same button low/high range */
6214 if (meas + 1 >= d->n_btn_con) {
6215 for (i = 0; i < d->n_btn_con; i++)
6216 if ((btnmeas[meas] < 0) ||
6217 (btnmeas[meas] != btnmeas[meas - i]))
6218 break;
6219 if (i == d->n_btn_con) {
6220 /* button pressed */
6221 btn = btnmeas[meas];
6222 break;
Joonwoo Park03324832012-03-19 19:36:16 -07006223 } else if ((n_btn_meas - meas) < (d->n_btn_con - 1)) {
6224 /* if left measurements are less than n_btn_con,
6225 * it's impossible to find button number */
6226 break;
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006227 }
6228 }
Joonwoo Park8b1f0982011-12-08 17:12:45 -08006229 }
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07006230
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006231 if (btn >= 0) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006232 if (priv->in_gpio_handler) {
6233 pr_debug("%s: GPIO already triggered, ignore button "
6234 "press\n", __func__);
6235 goto done;
6236 }
Joonwoo Parkdd9d2962012-07-23 19:24:20 -07006237 /* narrow down release threshold */
6238 tabla_mbhc_set_rel_thres(codec, btn_high[btn]);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006239 mask = tabla_get_button_mask(btn);
6240 priv->buttons_pressed |= mask;
Joonwoo Park03324832012-03-19 19:36:16 -07006241 wcd9xxx_lock_sleep(core);
6242 if (schedule_delayed_work(&priv->mbhc_btn_dwork,
6243 msecs_to_jiffies(400)) == 0) {
6244 WARN(1, "Button pressed twice without release"
6245 "event\n");
6246 wcd9xxx_unlock_sleep(core);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006247 }
Joonwoo Park816b8e62012-01-23 16:03:21 -08006248 } else {
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006249 pr_debug("%s: bogus button press, too short press?\n",
6250 __func__);
Joonwoo Park816b8e62012-01-23 16:03:21 -08006251 }
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006252
Joonwoo Park03324832012-03-19 19:36:16 -07006253 done:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006254 pr_debug("%s: leave\n", __func__);
6255 TABLA_RELEASE_LOCK(priv->codec_resource_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006256 return IRQ_HANDLED;
6257}
6258
Joonwoo Park03324832012-03-19 19:36:16 -07006259static int tabla_is_fake_press(struct tabla_priv *priv)
6260{
6261 int i;
6262 int r = 0;
6263 struct snd_soc_codec *codec = priv->codec;
6264 const int dces = MBHC_NUM_DCE_PLUG_DETECT;
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006265 s16 mb_v, v_ins_hu, v_ins_h;
6266
6267 v_ins_hu = tabla_get_current_v_ins(priv, true);
6268 v_ins_h = tabla_get_current_v_ins(priv, false);
Joonwoo Park03324832012-03-19 19:36:16 -07006269
6270 for (i = 0; i < dces; i++) {
6271 usleep_range(10000, 10000);
6272 if (i == 0) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006273 mb_v = tabla_codec_sta_dce(codec, 0, true);
Joonwoo Park03324832012-03-19 19:36:16 -07006274 pr_debug("%s: STA[0]: %d,%d\n", __func__, mb_v,
6275 tabla_codec_sta_dce_v(codec, 0, mb_v));
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006276 if (mb_v < (s16)priv->mbhc_data.v_b1_hu ||
6277 mb_v > v_ins_hu) {
Joonwoo Park03324832012-03-19 19:36:16 -07006278 r = 1;
6279 break;
6280 }
6281 } else {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006282 mb_v = tabla_codec_sta_dce(codec, 1, true);
Joonwoo Park03324832012-03-19 19:36:16 -07006283 pr_debug("%s: DCE[%d]: %d,%d\n", __func__, i, mb_v,
6284 tabla_codec_sta_dce_v(codec, 1, mb_v));
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006285 if (mb_v < (s16)priv->mbhc_data.v_b1_h ||
6286 mb_v > v_ins_h) {
Joonwoo Park03324832012-03-19 19:36:16 -07006287 r = 1;
6288 break;
6289 }
6290 }
6291 }
6292
6293 return r;
6294}
6295
Bradley Rubincb1e2732011-06-23 16:49:20 -07006296static irqreturn_t tabla_release_handler(int irq, void *data)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006297{
Joonwoo Parke5d3aa92012-01-11 14:47:15 -08006298 int ret;
Joonwoo Park816b8e62012-01-23 16:03:21 -08006299 struct tabla_priv *priv = data;
6300 struct snd_soc_codec *codec = priv->codec;
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07006301
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006302 pr_debug("%s: enter\n", __func__);
Joonwoo Park03324832012-03-19 19:36:16 -07006303
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006304 TABLA_ACQUIRE_LOCK(priv->codec_resource_lock);
6305 priv->mbhc_state = MBHC_STATE_RELEASE;
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07006306
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006307 tabla_codec_drive_v_to_micbias(codec, 10000);
6308
Joonwoo Park03324832012-03-19 19:36:16 -07006309 if (priv->buttons_pressed & TABLA_JACK_BUTTON_MASK) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006310 ret = tabla_cancel_btn_work(priv);
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07006311 if (ret == 0) {
Joonwoo Park03324832012-03-19 19:36:16 -07006312 pr_debug("%s: Reporting long button release event\n",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006313 __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006314 if (priv->mbhc_cfg.button_jack)
Joonwoo Park8b1f0982011-12-08 17:12:45 -08006315 tabla_snd_soc_jack_report(priv,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006316 priv->mbhc_cfg.button_jack, 0,
6317 priv->buttons_pressed);
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07006318 } else {
Joonwoo Park03324832012-03-19 19:36:16 -07006319 if (tabla_is_fake_press(priv)) {
6320 pr_debug("%s: Fake button press interrupt\n",
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006321 __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006322 } else if (priv->mbhc_cfg.button_jack) {
6323 if (priv->in_gpio_handler) {
6324 pr_debug("%s: GPIO kicked in, ignore\n",
6325 __func__);
6326 } else {
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006327 pr_debug("%s: Reporting short button "
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006328 "press and release\n",
6329 __func__);
6330 tabla_snd_soc_jack_report(priv,
6331 priv->mbhc_cfg.button_jack,
6332 priv->buttons_pressed,
6333 priv->buttons_pressed);
6334 tabla_snd_soc_jack_report(priv,
6335 priv->mbhc_cfg.button_jack, 0,
6336 priv->buttons_pressed);
6337 }
Bhalchandra Gajare0a792b12011-09-06 16:36:58 -07006338 }
6339 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006340
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08006341 priv->buttons_pressed &= ~TABLA_JACK_BUTTON_MASK;
6342 }
6343
Joonwoo Parkdd9d2962012-07-23 19:24:20 -07006344 /* revert narrowed release threshold */
6345 tabla_mbhc_calc_rel_thres(codec, tabla_mbhc_highest_btn_mv(codec));
Joonwoo Park03324832012-03-19 19:36:16 -07006346 tabla_codec_calibrate_hs_polling(codec);
6347
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006348 if (priv->mbhc_cfg.gpio)
6349 msleep(TABLA_MBHC_GPIO_REL_DEBOUNCE_TIME_MS);
Joonwoo Park03324832012-03-19 19:36:16 -07006350
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006351 tabla_codec_start_hs_polling(codec);
6352
6353 pr_debug("%s: leave\n", __func__);
6354 TABLA_RELEASE_LOCK(priv->codec_resource_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006355 return IRQ_HANDLED;
6356}
6357
Bradley Rubincb1e2732011-06-23 16:49:20 -07006358static void tabla_codec_shutdown_hs_removal_detect(struct snd_soc_codec *codec)
6359{
6360 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Park0976d012011-12-22 11:48:18 -08006361 const struct tabla_mbhc_general_cfg *generic =
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006362 TABLA_MBHC_CAL_GENERAL_PTR(tabla->mbhc_cfg.calibration);
Bradley Rubincb1e2732011-06-23 16:49:20 -07006363
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07006364 if (!tabla->mclk_enabled && !tabla->mbhc_polling_active)
Bradley Rubincb1e2732011-06-23 16:49:20 -07006365 tabla_codec_enable_config_mode(codec, 1);
6366
6367 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0x2, 0x2);
6368 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x6, 0x0);
Bradley Rubincb1e2732011-06-23 16:49:20 -07006369
Joonwoo Park0976d012011-12-22 11:48:18 -08006370 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg, 0x80, 0x00);
6371
6372 usleep_range(generic->t_shutdown_plug_rem,
6373 generic->t_shutdown_plug_rem);
Bradley Rubincb1e2732011-06-23 16:49:20 -07006374
6375 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL, 0xA, 0x8);
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07006376 if (!tabla->mclk_enabled && !tabla->mbhc_polling_active)
Bradley Rubincb1e2732011-06-23 16:49:20 -07006377 tabla_codec_enable_config_mode(codec, 0);
6378
6379 snd_soc_write(codec, TABLA_A_MBHC_SCALING_MUX_1, 0x00);
6380}
6381
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006382static void tabla_codec_cleanup_hs_polling(struct snd_soc_codec *codec)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006383{
6384 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Bradley Rubincb1e2732011-06-23 16:49:20 -07006385
6386 tabla_codec_shutdown_hs_removal_detect(codec);
6387
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07006388 if (!tabla->mclk_enabled) {
Asish Bhattacharya486745a2012-01-20 06:41:53 +05306389 tabla_codec_disable_clock_block(codec);
6390 tabla_codec_enable_bandgap(codec, TABLA_BANDGAP_OFF);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006391 }
6392
6393 tabla->mbhc_polling_active = false;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006394 tabla->mbhc_state = MBHC_STATE_NONE;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07006395}
6396
Patrick Lai49efeac2011-11-03 11:01:12 -07006397static irqreturn_t tabla_hphl_ocp_irq(int irq, void *data)
6398{
6399 struct tabla_priv *tabla = data;
6400 struct snd_soc_codec *codec;
6401
6402 pr_info("%s: received HPHL OCP irq\n", __func__);
6403
6404 if (tabla) {
6405 codec = tabla->codec;
Patrick Laie381d1e2012-07-06 22:42:52 -07006406 if ((tabla->hphlocp_cnt < TABLA_OCP_ATTEMPT) &&
6407 (!tabla->hphrocp_cnt)) {
Patrick Laic7cae882011-11-18 11:52:49 -08006408 pr_info("%s: retry\n", __func__);
Patrick Laie381d1e2012-07-06 22:42:52 -07006409 tabla->hphlocp_cnt++;
Patrick Laic7cae882011-11-18 11:52:49 -08006410 snd_soc_update_bits(codec, TABLA_A_RX_HPH_OCP_CTL, 0x10,
6411 0x00);
6412 snd_soc_update_bits(codec, TABLA_A_RX_HPH_OCP_CTL, 0x10,
6413 0x10);
6414 } else {
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05306415 wcd9xxx_disable_irq(codec->control_data,
Patrick Laic7cae882011-11-18 11:52:49 -08006416 TABLA_IRQ_HPH_PA_OCPL_FAULT);
Patrick Laic7cae882011-11-18 11:52:49 -08006417 tabla->hph_status |= SND_JACK_OC_HPHL;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006418 if (tabla->mbhc_cfg.headset_jack)
Patrick Laic7cae882011-11-18 11:52:49 -08006419 tabla_snd_soc_jack_report(tabla,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006420 tabla->mbhc_cfg.headset_jack,
6421 tabla->hph_status,
6422 TABLA_JACK_MASK);
Patrick Lai49efeac2011-11-03 11:01:12 -07006423 }
6424 } else {
6425 pr_err("%s: Bad tabla private data\n", __func__);
6426 }
6427
6428 return IRQ_HANDLED;
6429}
6430
6431static irqreturn_t tabla_hphr_ocp_irq(int irq, void *data)
6432{
6433 struct tabla_priv *tabla = data;
6434 struct snd_soc_codec *codec;
6435
6436 pr_info("%s: received HPHR OCP irq\n", __func__);
6437
6438 if (tabla) {
6439 codec = tabla->codec;
Patrick Laie381d1e2012-07-06 22:42:52 -07006440 if ((tabla->hphrocp_cnt < TABLA_OCP_ATTEMPT) &&
6441 (!tabla->hphlocp_cnt)) {
Patrick Laic7cae882011-11-18 11:52:49 -08006442 pr_info("%s: retry\n", __func__);
Patrick Laie381d1e2012-07-06 22:42:52 -07006443 tabla->hphrocp_cnt++;
Patrick Laic7cae882011-11-18 11:52:49 -08006444 snd_soc_update_bits(codec, TABLA_A_RX_HPH_OCP_CTL, 0x10,
6445 0x00);
6446 snd_soc_update_bits(codec, TABLA_A_RX_HPH_OCP_CTL, 0x10,
6447 0x10);
6448 } else {
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05306449 wcd9xxx_disable_irq(codec->control_data,
Patrick Laic7cae882011-11-18 11:52:49 -08006450 TABLA_IRQ_HPH_PA_OCPR_FAULT);
Patrick Laic7cae882011-11-18 11:52:49 -08006451 tabla->hph_status |= SND_JACK_OC_HPHR;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006452 if (tabla->mbhc_cfg.headset_jack)
Patrick Laic7cae882011-11-18 11:52:49 -08006453 tabla_snd_soc_jack_report(tabla,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006454 tabla->mbhc_cfg.headset_jack,
6455 tabla->hph_status,
6456 TABLA_JACK_MASK);
Patrick Lai49efeac2011-11-03 11:01:12 -07006457 }
6458 } else {
6459 pr_err("%s: Bad tabla private data\n", __func__);
6460 }
6461
6462 return IRQ_HANDLED;
6463}
6464
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006465static bool tabla_is_inval_ins_range(struct snd_soc_codec *codec,
6466 s32 mic_volt, bool highhph, bool *highv)
Joonwoo Park03324832012-03-19 19:36:16 -07006467{
6468 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006469 bool invalid = false;
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006470 s16 v_hs_max;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006471
6472 /* Perform this check only when the high voltage headphone
6473 * needs to be considered as invalid
6474 */
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006475 v_hs_max = tabla_get_current_v_hs_max(tabla);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006476 *highv = mic_volt > v_hs_max;
6477 if (!highhph && *highv)
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006478 invalid = true;
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006479 else if (mic_volt < tabla->mbhc_data.v_inval_ins_high &&
6480 (mic_volt > tabla->mbhc_data.v_inval_ins_low))
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006481 invalid = true;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006482
6483 return invalid;
6484}
6485
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006486static bool tabla_is_inval_ins_delta(struct snd_soc_codec *codec,
6487 int mic_volt, int mic_volt_prev,
6488 int threshold)
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006489{
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006490 return abs(mic_volt - mic_volt_prev) > threshold;
Joonwoo Park03324832012-03-19 19:36:16 -07006491}
6492
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006493/* called under codec_resource_lock acquisition */
6494void tabla_find_plug_and_report(struct snd_soc_codec *codec,
6495 enum tabla_mbhc_plug_type plug_type)
Joonwoo Park03324832012-03-19 19:36:16 -07006496{
6497 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006498
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006499 pr_debug("%s: enter current_plug(%d) new_plug(%d)\n",
6500 __func__, tabla->current_plug, plug_type);
6501
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006502 if (plug_type == PLUG_TYPE_HEADPHONE &&
6503 tabla->current_plug == PLUG_TYPE_NONE) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006504 /* Nothing was reported previously
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006505 * report a headphone or unsupported
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006506 */
6507 tabla_codec_report_plug(codec, 1, SND_JACK_HEADPHONE);
6508 tabla_codec_cleanup_hs_polling(codec);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006509 } else if (plug_type == PLUG_TYPE_GND_MIC_SWAP) {
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006510 if (!tabla->mbhc_cfg.detect_extn_cable) {
6511 if (tabla->current_plug == PLUG_TYPE_HEADSET)
6512 tabla_codec_report_plug(codec, 0,
6513 SND_JACK_HEADSET);
6514 else if (tabla->current_plug == PLUG_TYPE_HEADPHONE)
6515 tabla_codec_report_plug(codec, 0,
6516 SND_JACK_HEADPHONE);
6517 }
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006518 tabla_codec_report_plug(codec, 1, SND_JACK_UNSUPPORTED);
6519 tabla_codec_cleanup_hs_polling(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006520 } else if (plug_type == PLUG_TYPE_HEADSET) {
6521 /* If Headphone was reported previously, this will
6522 * only report the mic line
6523 */
6524 tabla_codec_report_plug(codec, 1, SND_JACK_HEADSET);
6525 msleep(100);
6526 tabla_codec_start_hs_polling(codec);
6527 } else if (plug_type == PLUG_TYPE_HIGH_HPH) {
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006528 if (tabla->mbhc_cfg.detect_extn_cable) {
6529 /* High impedance device found. Report as LINEOUT*/
6530 tabla_codec_report_plug(codec, 1, SND_JACK_LINEOUT);
6531 tabla_codec_cleanup_hs_polling(codec);
6532 pr_debug("%s: setup mic trigger for further detection\n",
6533 __func__);
6534 tabla->lpi_enabled = true;
6535 /*
6536 * Do not enable HPHL trigger. If playback is active,
6537 * it might lead to continuous false HPHL triggers
6538 */
6539 tabla_codec_enable_hs_detect(codec, 1,
6540 MBHC_USE_MB_TRIGGER,
6541 false);
6542 } else {
6543 if (tabla->current_plug == PLUG_TYPE_NONE)
6544 tabla_codec_report_plug(codec, 1,
6545 SND_JACK_HEADPHONE);
6546 tabla_codec_cleanup_hs_polling(codec);
6547 pr_debug("setup mic trigger for further detection\n");
6548 tabla->lpi_enabled = true;
6549 tabla_codec_enable_hs_detect(codec, 1,
6550 MBHC_USE_MB_TRIGGER |
6551 MBHC_USE_HPHL_TRIGGER,
6552 false);
6553 }
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006554 } else {
6555 WARN(1, "Unexpected current plug_type %d, plug_type %d\n",
6556 tabla->current_plug, plug_type);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006557 }
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006558 pr_debug("%s: leave\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006559}
6560
6561/* should be called under interrupt context that hold suspend */
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07006562static void tabla_schedule_hs_detect_plug(struct tabla_priv *tabla,
6563 struct work_struct *correct_plug_work)
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006564{
6565 pr_debug("%s: scheduling tabla_hs_correct_gpio_plug\n", __func__);
6566 tabla->hs_detect_work_stop = false;
6567 wcd9xxx_lock_sleep(tabla->codec->control_data);
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07006568 schedule_work(correct_plug_work);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006569}
6570
6571/* called under codec_resource_lock acquisition */
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07006572static void tabla_cancel_hs_detect_plug(struct tabla_priv *tabla,
6573 struct work_struct *correct_plug_work)
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006574{
6575 pr_debug("%s: canceling hs_correct_plug_work\n", __func__);
6576 tabla->hs_detect_work_stop = true;
6577 wmb();
6578 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07006579 if (cancel_work_sync(correct_plug_work)) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006580 pr_debug("%s: hs_correct_plug_work is canceled\n", __func__);
6581 wcd9xxx_unlock_sleep(tabla->codec->control_data);
6582 }
6583 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
6584}
6585
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006586static bool tabla_hs_gpio_level_remove(struct tabla_priv *tabla)
6587{
6588 return (gpio_get_value_cansleep(tabla->mbhc_cfg.gpio) !=
6589 tabla->mbhc_cfg.gpio_level_insert);
6590}
6591
Joonwoo Park41956722012-04-18 13:13:07 -07006592/* called under codec_resource_lock acquisition */
6593static void tabla_codec_hphr_gnd_switch(struct snd_soc_codec *codec, bool on)
6594{
6595 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x01, on);
6596 if (on)
6597 usleep_range(5000, 5000);
6598}
6599
6600/* called under codec_resource_lock acquisition and mbhc override = 1 */
6601static enum tabla_mbhc_plug_type
6602tabla_codec_get_plug_type(struct snd_soc_codec *codec, bool highhph)
6603{
6604 int i;
6605 bool gndswitch, vddioswitch;
6606 int scaled;
6607 struct tabla_mbhc_plug_type_cfg *plug_type_ptr;
6608 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Park6a5a4f12012-06-15 15:56:25 -07006609 int num_det = MBHC_NUM_DCE_PLUG_DETECT + 1;
Joonwoo Park41956722012-04-18 13:13:07 -07006610 enum tabla_mbhc_plug_type plug_type[num_det];
6611 s16 mb_v[num_det];
6612 s32 mic_mv[num_det];
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006613 bool inval;
6614 bool highdelta;
6615 bool ahighv = false, highv;
Joonwoo Park6a5a4f12012-06-15 15:56:25 -07006616 bool gndmicswapped = false;
Joonwoo Park41956722012-04-18 13:13:07 -07006617
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006618 pr_debug("%s: enter\n", __func__);
6619
Joonwoo Park41956722012-04-18 13:13:07 -07006620 /* make sure override is on */
6621 WARN_ON(!(snd_soc_read(codec, TABLA_A_CDC_MBHC_B1_CTL) & 0x04));
6622
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006623 /* GND and MIC swap detection requires at least 2 rounds of DCE */
6624 BUG_ON(num_det < 2);
6625
6626 plug_type_ptr =
6627 TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla->mbhc_cfg.calibration);
6628
6629 plug_type[0] = PLUG_TYPE_INVALID;
6630
Joonwoo Park41956722012-04-18 13:13:07 -07006631 /* performs DCEs for N times
6632 * 1st: check if voltage is in invalid range
6633 * 2nd - N-2nd: check voltage range and delta
6634 * N-1st: check voltage range, delta with HPHR GND switch
Joonwoo Park6a5a4f12012-06-15 15:56:25 -07006635 * Nth: check voltage range with VDDIO switch */
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006636 for (i = 0; i < num_det; i++) {
Joonwoo Park6a5a4f12012-06-15 15:56:25 -07006637 gndswitch = (i == (num_det - 2));
6638 vddioswitch = (i == (num_det - 1)) || (i == (num_det - 2));
Joonwoo Park41956722012-04-18 13:13:07 -07006639 if (i == 0) {
6640 mb_v[i] = tabla_codec_setup_hs_polling(codec);
6641 mic_mv[i] = tabla_codec_sta_dce_v(codec, 1 , mb_v[i]);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006642 inval = tabla_is_inval_ins_range(codec, mic_mv[i],
6643 highhph, &highv);
6644 ahighv |= highv;
Joonwoo Park41956722012-04-18 13:13:07 -07006645 scaled = mic_mv[i];
Joonwoo Park41956722012-04-18 13:13:07 -07006646 } else {
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006647 if (vddioswitch)
6648 __tabla_codec_switch_micbias(tabla->codec, 1,
6649 false, false);
Joonwoo Park41956722012-04-18 13:13:07 -07006650 if (gndswitch)
6651 tabla_codec_hphr_gnd_switch(codec, true);
6652 mb_v[i] = __tabla_codec_sta_dce(codec, 1, true, true);
6653 mic_mv[i] = tabla_codec_sta_dce_v(codec, 1 , mb_v[i]);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006654 if (vddioswitch)
6655 scaled = tabla_scale_v_micb_vddio(tabla,
Joonwoo Park41956722012-04-18 13:13:07 -07006656 mic_mv[i],
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006657 false);
6658 else
6659 scaled = mic_mv[i];
6660 /* !gndswitch & vddioswitch means the previous DCE
6661 * was done with gndswitch, don't compare with DCE
6662 * with gndswitch */
6663 highdelta = tabla_is_inval_ins_delta(codec, scaled,
Joonwoo Park6a5a4f12012-06-15 15:56:25 -07006664 mic_mv[i - 1],
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006665 TABLA_MBHC_FAKE_INS_DELTA_SCALED_MV);
6666 inval = (tabla_is_inval_ins_range(codec, mic_mv[i],
6667 highhph, &highv) ||
6668 highdelta);
6669 ahighv |= highv;
Joonwoo Park41956722012-04-18 13:13:07 -07006670 if (gndswitch)
6671 tabla_codec_hphr_gnd_switch(codec, false);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006672 if (vddioswitch)
6673 __tabla_codec_switch_micbias(tabla->codec, 0,
6674 false, false);
6675 /* claim UNSUPPORTED plug insertion when
6676 * good headset is detected but HPHR GND switch makes
6677 * delta difference */
6678 if (i == (num_det - 2) && highdelta && !ahighv)
Joonwoo Park6a5a4f12012-06-15 15:56:25 -07006679 gndmicswapped = true;
6680 else if (i == (num_det - 1) && inval) {
6681 if (gndmicswapped)
6682 plug_type[0] = PLUG_TYPE_GND_MIC_SWAP;
6683 else
6684 plug_type[0] = PLUG_TYPE_INVALID;
6685 }
Joonwoo Park41956722012-04-18 13:13:07 -07006686 }
6687 pr_debug("%s: DCE #%d, %04x, V %d, scaled V %d, GND %d, "
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006688 "VDDIO %d, inval %d\n", __func__,
Joonwoo Park41956722012-04-18 13:13:07 -07006689 i + 1, mb_v[i] & 0xffff, mic_mv[i], scaled, gndswitch,
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006690 vddioswitch, inval);
6691 /* don't need to run further DCEs */
6692 if (ahighv && inval)
6693 break;
6694 mic_mv[i] = scaled;
Joonwoo Park41956722012-04-18 13:13:07 -07006695 }
6696
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006697 for (i = 0; (plug_type[0] != PLUG_TYPE_GND_MIC_SWAP && !inval) &&
6698 i < num_det; i++) {
Joonwoo Park41956722012-04-18 13:13:07 -07006699 /*
6700 * If we are here, means none of the all
6701 * measurements are fake, continue plug type detection.
6702 * If all three measurements do not produce same
6703 * plug type, restart insertion detection
6704 */
6705 if (mic_mv[i] < plug_type_ptr->v_no_mic) {
6706 plug_type[i] = PLUG_TYPE_HEADPHONE;
6707 pr_debug("%s: Detect attempt %d, detected Headphone\n",
6708 __func__, i);
6709 } else if (highhph && (mic_mv[i] > plug_type_ptr->v_hs_max)) {
6710 plug_type[i] = PLUG_TYPE_HIGH_HPH;
6711 pr_debug("%s: Detect attempt %d, detected High "
6712 "Headphone\n", __func__, i);
6713 } else {
6714 plug_type[i] = PLUG_TYPE_HEADSET;
6715 pr_debug("%s: Detect attempt %d, detected Headset\n",
6716 __func__, i);
6717 }
6718
6719 if (i > 0 && (plug_type[i - 1] != plug_type[i])) {
6720 pr_err("%s: Detect attempt %d and %d are not same",
6721 __func__, i - 1, i);
6722 plug_type[0] = PLUG_TYPE_INVALID;
6723 inval = true;
6724 break;
6725 }
6726 }
6727
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006728 pr_debug("%s: Detected plug type %d\n", __func__, plug_type[0]);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006729 pr_debug("%s: leave\n", __func__);
Joonwoo Park41956722012-04-18 13:13:07 -07006730 return plug_type[0];
6731}
6732
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006733static void tabla_hs_correct_gpio_plug(struct work_struct *work)
6734{
6735 struct tabla_priv *tabla;
6736 struct snd_soc_codec *codec;
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006737 int retry = 0, pt_gnd_mic_swap_cnt = 0;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006738 bool correction = false;
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006739 enum tabla_mbhc_plug_type plug_type = PLUG_TYPE_INVALID;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006740 unsigned long timeout;
6741
6742 tabla = container_of(work, struct tabla_priv, hs_correct_plug_work);
6743 codec = tabla->codec;
6744
6745 pr_debug("%s: enter\n", __func__);
6746 tabla->mbhc_cfg.mclk_cb_fn(codec, 1, false);
6747
6748 /* Keep override on during entire plug type correction work.
6749 *
6750 * This is okay under the assumption that any GPIO irqs which use
6751 * MBHC block cancel and sync this work so override is off again
6752 * prior to GPIO interrupt handler's MBHC block usage.
6753 * Also while this correction work is running, we can guarantee
6754 * DAPM doesn't use any MBHC block as this work only runs with
6755 * headphone detection.
6756 */
6757 tabla_turn_onoff_override(codec, true);
6758
6759 timeout = jiffies + msecs_to_jiffies(TABLA_HS_DETECT_PLUG_TIME_MS);
6760 while (!time_after(jiffies, timeout)) {
6761 ++retry;
6762 rmb();
6763 if (tabla->hs_detect_work_stop) {
6764 pr_debug("%s: stop requested\n", __func__);
6765 break;
6766 }
6767
6768 msleep(TABLA_HS_DETECT_PLUG_INERVAL_MS);
6769 if (tabla_hs_gpio_level_remove(tabla)) {
6770 pr_debug("%s: GPIO value is low\n", __func__);
6771 break;
6772 }
6773
6774 /* can race with removal interrupt */
6775 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Joonwoo Park41956722012-04-18 13:13:07 -07006776 plug_type = tabla_codec_get_plug_type(codec, true);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006777 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
6778
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006779 pr_debug("%s: attempt(%d) current_plug(%d) new_plug(%d)\n",
6780 __func__, retry, tabla->current_plug, plug_type);
Joonwoo Park41956722012-04-18 13:13:07 -07006781 if (plug_type == PLUG_TYPE_INVALID) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006782 pr_debug("Invalid plug in attempt # %d\n", retry);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006783 if (!tabla->mbhc_cfg.detect_extn_cable &&
6784 retry == NUM_ATTEMPTS_TO_REPORT &&
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006785 tabla->current_plug == PLUG_TYPE_NONE) {
6786 tabla_codec_report_plug(codec, 1,
6787 SND_JACK_HEADPHONE);
6788 }
Joonwoo Park41956722012-04-18 13:13:07 -07006789 } else if (plug_type == PLUG_TYPE_HEADPHONE) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006790 pr_debug("Good headphone detected, continue polling mic\n");
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006791 if (tabla->mbhc_cfg.detect_extn_cable) {
6792 if (tabla->current_plug != plug_type)
6793 tabla_codec_report_plug(codec, 1,
6794 SND_JACK_HEADPHONE);
6795 } else if (tabla->current_plug == PLUG_TYPE_NONE)
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006796 tabla_codec_report_plug(codec, 1,
6797 SND_JACK_HEADPHONE);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006798 } else {
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006799 if (plug_type == PLUG_TYPE_GND_MIC_SWAP) {
6800 pt_gnd_mic_swap_cnt++;
6801 if (pt_gnd_mic_swap_cnt <
6802 TABLA_MBHC_GND_MIC_SWAP_THRESHOLD)
6803 continue;
6804 else if (pt_gnd_mic_swap_cnt >
6805 TABLA_MBHC_GND_MIC_SWAP_THRESHOLD) {
6806 /* This is due to GND/MIC switch didn't
6807 * work, Report unsupported plug */
6808 } else if (tabla->mbhc_cfg.swap_gnd_mic) {
6809 /* if switch is toggled, check again,
6810 * otherwise report unsupported plug */
6811 if (tabla->mbhc_cfg.swap_gnd_mic(codec))
6812 continue;
6813 }
6814 } else
6815 pt_gnd_mic_swap_cnt = 0;
6816
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006817 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
6818 /* Turn off override */
6819 tabla_turn_onoff_override(codec, false);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006820 /* The valid plug also includes PLUG_TYPE_GND_MIC_SWAP
6821 */
Joonwoo Park41956722012-04-18 13:13:07 -07006822 tabla_find_plug_and_report(codec, plug_type);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006823 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
6824 pr_debug("Attempt %d found correct plug %d\n", retry,
Joonwoo Park41956722012-04-18 13:13:07 -07006825 plug_type);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006826 correction = true;
6827 break;
6828 }
6829 }
6830
6831 /* Turn off override */
6832 if (!correction)
6833 tabla_turn_onoff_override(codec, false);
6834
6835 tabla->mbhc_cfg.mclk_cb_fn(codec, 0, false);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006836
6837 if (tabla->mbhc_cfg.detect_extn_cable) {
6838 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
6839 if (tabla->current_plug == PLUG_TYPE_HEADPHONE ||
6840 tabla->current_plug == PLUG_TYPE_GND_MIC_SWAP ||
6841 tabla->current_plug == PLUG_TYPE_INVALID ||
6842 plug_type == PLUG_TYPE_INVALID) {
6843 /* Enable removal detection */
6844 tabla_codec_cleanup_hs_polling(codec);
6845 tabla_codec_enable_hs_detect(codec, 0, 0, false);
6846 }
6847 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
6848 }
6849 pr_debug("%s: leave current_plug(%d)\n",
6850 __func__, tabla->current_plug);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006851 /* unlock sleep */
6852 wcd9xxx_unlock_sleep(tabla->codec->control_data);
6853}
6854
6855/* called under codec_resource_lock acquisition */
6856static void tabla_codec_decide_gpio_plug(struct snd_soc_codec *codec)
6857{
Joonwoo Park41956722012-04-18 13:13:07 -07006858 enum tabla_mbhc_plug_type plug_type;
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006859 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006860
6861 pr_debug("%s: enter\n", __func__);
6862
6863 tabla_turn_onoff_override(codec, true);
Joonwoo Park41956722012-04-18 13:13:07 -07006864 plug_type = tabla_codec_get_plug_type(codec, true);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006865 tabla_turn_onoff_override(codec, false);
6866
6867 if (tabla_hs_gpio_level_remove(tabla)) {
6868 pr_debug("%s: GPIO value is low when determining plug\n",
6869 __func__);
6870 return;
6871 }
6872
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006873 if (plug_type == PLUG_TYPE_INVALID ||
6874 plug_type == PLUG_TYPE_GND_MIC_SWAP) {
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07006875 tabla_schedule_hs_detect_plug(tabla,
6876 &tabla->hs_correct_plug_work);
Joonwoo Park41956722012-04-18 13:13:07 -07006877 } else if (plug_type == PLUG_TYPE_HEADPHONE) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006878 tabla_codec_report_plug(codec, 1, SND_JACK_HEADPHONE);
6879
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07006880 tabla_schedule_hs_detect_plug(tabla,
6881 &tabla->hs_correct_plug_work);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006882 } else {
Joonwoo Park41956722012-04-18 13:13:07 -07006883 pr_debug("%s: Valid plug found, determine plug type %d\n",
6884 __func__, plug_type);
6885 tabla_find_plug_and_report(codec, plug_type);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006886 }
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006887 pr_debug("%s: leave\n", __func__);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006888}
6889
6890/* called under codec_resource_lock acquisition */
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006891static void tabla_codec_detect_plug_type(struct snd_soc_codec *codec)
6892{
Joonwoo Park41956722012-04-18 13:13:07 -07006893 enum tabla_mbhc_plug_type plug_type;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006894 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
6895 const struct tabla_mbhc_plug_detect_cfg *plug_det =
6896 TABLA_MBHC_CAL_PLUG_DET_PTR(tabla->mbhc_cfg.calibration);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006897 pr_debug("%s: enter\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006898 /* Turn on the override,
6899 * tabla_codec_setup_hs_polling requires override on */
6900 tabla_turn_onoff_override(codec, true);
Joonwoo Park03324832012-03-19 19:36:16 -07006901
6902 if (plug_det->t_ins_complete > 20)
6903 msleep(plug_det->t_ins_complete);
6904 else
6905 usleep_range(plug_det->t_ins_complete * 1000,
6906 plug_det->t_ins_complete * 1000);
6907
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006908 if (tabla->mbhc_cfg.gpio) {
6909 /* Turn off the override */
6910 tabla_turn_onoff_override(codec, false);
6911 if (tabla_hs_gpio_level_remove(tabla))
6912 pr_debug("%s: GPIO value is low when determining "
6913 "plug\n", __func__);
6914 else
6915 tabla_codec_decide_gpio_plug(codec);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006916 pr_debug("%s: leave\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006917 return;
6918 }
6919
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006920 plug_type = tabla_codec_get_plug_type(codec, false);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006921 tabla_turn_onoff_override(codec, false);
Joonwoo Park03324832012-03-19 19:36:16 -07006922
Joonwoo Park41956722012-04-18 13:13:07 -07006923 if (plug_type == PLUG_TYPE_INVALID) {
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006924 pr_debug("%s: Invalid plug type detected\n", __func__);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006925 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_B1_CTL, 0x02, 0x02);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07006926 tabla_codec_cleanup_hs_polling(codec);
6927 tabla_codec_enable_hs_detect(codec, 1,
6928 MBHC_USE_MB_TRIGGER |
6929 MBHC_USE_HPHL_TRIGGER, false);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07006930 } else if (plug_type == PLUG_TYPE_GND_MIC_SWAP) {
6931 pr_debug("%s: GND-MIC swapped plug type detected\n", __func__);
6932 tabla_codec_report_plug(codec, 1, SND_JACK_UNSUPPORTED);
6933 tabla_codec_cleanup_hs_polling(codec);
6934 tabla_codec_enable_hs_detect(codec, 0, 0, false);
Joonwoo Park41956722012-04-18 13:13:07 -07006935 } else if (plug_type == PLUG_TYPE_HEADPHONE) {
Joonwoo Park03324832012-03-19 19:36:16 -07006936 pr_debug("%s: Headphone Detected\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006937 tabla_codec_report_plug(codec, 1, SND_JACK_HEADPHONE);
6938 tabla_codec_cleanup_hs_polling(codec);
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07006939 tabla_schedule_hs_detect_plug(tabla,
6940 &tabla->hs_correct_plug_work_nogpio);
Joonwoo Park41956722012-04-18 13:13:07 -07006941 } else if (plug_type == PLUG_TYPE_HEADSET) {
Joonwoo Park03324832012-03-19 19:36:16 -07006942 pr_debug("%s: Headset detected\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006943 tabla_codec_report_plug(codec, 1, SND_JACK_HEADSET);
6944
Joonwoo Park03324832012-03-19 19:36:16 -07006945 /* avoid false button press detect */
6946 msleep(50);
Joonwoo Park03324832012-03-19 19:36:16 -07006947 tabla_codec_start_hs_polling(codec);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006948 } else if (tabla->mbhc_cfg.detect_extn_cable &&
6949 plug_type == PLUG_TYPE_HIGH_HPH) {
6950 pr_debug("%s: High impedance plug type detected\n", __func__);
6951 tabla_codec_report_plug(codec, 1, SND_JACK_LINEOUT);
6952 /* Enable insertion detection on the other end of cable */
6953 tabla_codec_cleanup_hs_polling(codec);
6954 tabla_codec_enable_hs_detect(codec, 1,
6955 MBHC_USE_MB_TRIGGER, false);
Joonwoo Park03324832012-03-19 19:36:16 -07006956 }
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006957 pr_debug("%s: leave\n", __func__);
Joonwoo Park03324832012-03-19 19:36:16 -07006958}
6959
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006960/* called only from interrupt which is under codec_resource_lock acquisition */
6961static void tabla_hs_insert_irq_gpio(struct tabla_priv *priv, bool is_removal)
Bradley Rubincb1e2732011-06-23 16:49:20 -07006962{
Bradley Rubincb1e2732011-06-23 16:49:20 -07006963 struct snd_soc_codec *codec = priv->codec;
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006964 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006965
6966 if (!is_removal) {
6967 pr_debug("%s: MIC trigger insertion interrupt\n", __func__);
6968
6969 rmb();
6970 if (priv->lpi_enabled)
6971 msleep(100);
6972
6973 rmb();
6974 if (!priv->lpi_enabled) {
6975 pr_debug("%s: lpi is disabled\n", __func__);
6976 } else if (gpio_get_value_cansleep(priv->mbhc_cfg.gpio) ==
6977 priv->mbhc_cfg.gpio_level_insert) {
6978 pr_debug("%s: Valid insertion, "
6979 "detect plug type\n", __func__);
6980 tabla_codec_decide_gpio_plug(codec);
6981 } else {
6982 pr_debug("%s: Invalid insertion, "
6983 "stop plug detection\n", __func__);
6984 }
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07006985 } else if (tabla->mbhc_cfg.detect_extn_cable) {
6986 pr_debug("%s: Removal\n", __func__);
6987 if (!tabla_hs_gpio_level_remove(tabla)) {
6988 /*
6989 * gpio says, something is still inserted, could be
6990 * extension cable i.e. headset is removed from
6991 * extension cable
6992 */
6993 /* cancel detect plug */
6994 tabla_cancel_hs_detect_plug(tabla,
6995 &tabla->hs_correct_plug_work);
6996 tabla_codec_decide_gpio_plug(codec);
6997 }
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07006998 } else {
6999 pr_err("%s: GPIO used, invalid MBHC Removal\n", __func__);
7000 }
7001}
7002
7003/* called only from interrupt which is under codec_resource_lock acquisition */
7004static void tabla_hs_insert_irq_nogpio(struct tabla_priv *priv, bool is_removal,
7005 bool is_mb_trigger)
7006{
Joonwoo Park03324832012-03-19 19:36:16 -07007007 int ret;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007008 struct snd_soc_codec *codec = priv->codec;
7009 struct wcd9xxx *core = dev_get_drvdata(priv->codec->dev->parent);
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07007010 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
7011
7012 /* Cancel possibly running hs_detect_work */
7013 tabla_cancel_hs_detect_plug(tabla,
7014 &tabla->hs_correct_plug_work_nogpio);
Bhalchandra Gajare7fc72332011-10-13 19:01:55 -07007015
7016 if (is_removal) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007017
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07007018 /*
7019 * If headphone is removed while playback is in progress,
7020 * it is possible that micbias will be switched to VDDIO.
7021 */
Joonwoo Park03324832012-03-19 19:36:16 -07007022 tabla_codec_switch_micbias(codec, 0);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07007023 if (priv->current_plug == PLUG_TYPE_HEADPHONE)
7024 tabla_codec_report_plug(codec, 0, SND_JACK_HEADPHONE);
7025 else if (priv->current_plug == PLUG_TYPE_GND_MIC_SWAP)
7026 tabla_codec_report_plug(codec, 0, SND_JACK_UNSUPPORTED);
7027 else
7028 WARN(1, "%s: Unexpected current plug type %d\n",
7029 __func__, priv->current_plug);
Bradley Rubincb1e2732011-06-23 16:49:20 -07007030 tabla_codec_shutdown_hs_removal_detect(codec);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007031 tabla_codec_enable_hs_detect(codec, 1,
7032 MBHC_USE_MB_TRIGGER |
7033 MBHC_USE_HPHL_TRIGGER,
Joonwoo Park03324832012-03-19 19:36:16 -07007034 true);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007035 } else if (is_mb_trigger && !is_removal) {
Joonwoo Park03324832012-03-19 19:36:16 -07007036 pr_debug("%s: Waiting for Headphone left trigger\n",
7037 __func__);
7038 wcd9xxx_lock_sleep(core);
7039 if (schedule_delayed_work(&priv->mbhc_insert_dwork,
7040 usecs_to_jiffies(1000000)) == 0) {
7041 pr_err("%s: mbhc_insert_dwork is already scheduled\n",
7042 __func__);
7043 wcd9xxx_unlock_sleep(core);
Joonwoo Parkf4267c22012-01-10 13:25:24 -08007044 }
Joonwoo Park03324832012-03-19 19:36:16 -07007045 tabla_codec_enable_hs_detect(codec, 1, MBHC_USE_HPHL_TRIGGER,
7046 false);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007047 } else {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007048 ret = cancel_delayed_work(&priv->mbhc_insert_dwork);
7049 if (ret != 0) {
7050 pr_debug("%s: Complete plug insertion, Detecting plug "
7051 "type\n", __func__);
7052 tabla_codec_detect_plug_type(codec);
7053 wcd9xxx_unlock_sleep(core);
7054 } else {
7055 wcd9xxx_enable_irq(codec->control_data,
7056 TABLA_IRQ_MBHC_INSERTION);
7057 pr_err("%s: Error detecting plug insertion\n",
7058 __func__);
7059 }
Joonwoo Park03324832012-03-19 19:36:16 -07007060 }
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007061}
Bhalchandra Gajare9494fa262011-11-10 19:25:59 -08007062
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007063/* called only from interrupt which is under codec_resource_lock acquisition */
7064static void tabla_hs_insert_irq_extn(struct tabla_priv *priv,
7065 bool is_mb_trigger)
7066{
7067 struct snd_soc_codec *codec = priv->codec;
7068 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
7069
7070 /* Cancel possibly running hs_detect_work */
7071 tabla_cancel_hs_detect_plug(tabla,
7072 &tabla->hs_correct_plug_work);
7073
7074 if (is_mb_trigger) {
7075 pr_debug("%s: Waiting for Headphone left trigger\n",
7076 __func__);
7077 tabla_codec_enable_hs_detect(codec, 1, MBHC_USE_HPHL_TRIGGER,
7078 false);
7079 } else {
7080 pr_debug("%s: HPHL trigger received, detecting plug type\n",
7081 __func__);
7082 tabla_codec_detect_plug_type(codec);
7083 }
7084}
7085
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007086static irqreturn_t tabla_hs_insert_irq(int irq, void *data)
7087{
7088 bool is_mb_trigger, is_removal;
7089 struct tabla_priv *priv = data;
7090 struct snd_soc_codec *codec = priv->codec;
Bradley Rubincb1e2732011-06-23 16:49:20 -07007091
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007092 pr_debug("%s: enter\n", __func__);
7093 TABLA_ACQUIRE_LOCK(priv->codec_resource_lock);
7094 wcd9xxx_disable_irq(codec->control_data, TABLA_IRQ_MBHC_INSERTION);
7095
7096 is_mb_trigger = !!(snd_soc_read(codec, priv->mbhc_bias_regs.mbhc_reg) &
7097 0x10);
7098 is_removal = !!(snd_soc_read(codec, TABLA_A_CDC_MBHC_INT_CTL) & 0x02);
7099 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_INT_CTL, 0x03, 0x00);
7100
7101 /* Turn off both HPH and MIC line schmitt triggers */
7102 snd_soc_update_bits(codec, priv->mbhc_bias_regs.mbhc_reg, 0x90, 0x00);
7103 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x13, 0x00);
7104 snd_soc_update_bits(codec, priv->mbhc_bias_regs.ctl_reg, 0x01, 0x00);
7105
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007106 if (priv->mbhc_cfg.detect_extn_cable &&
7107 priv->current_plug == PLUG_TYPE_HIGH_HPH)
7108 tabla_hs_insert_irq_extn(priv, is_mb_trigger);
7109 else if (priv->mbhc_cfg.gpio)
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007110 tabla_hs_insert_irq_gpio(priv, is_removal);
7111 else
7112 tabla_hs_insert_irq_nogpio(priv, is_removal, is_mb_trigger);
7113
7114 TABLA_RELEASE_LOCK(priv->codec_resource_lock);
Bradley Rubincb1e2732011-06-23 16:49:20 -07007115 return IRQ_HANDLED;
7116}
7117
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007118static bool is_valid_mic_voltage(struct snd_soc_codec *codec, s32 mic_mv)
7119{
7120 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007121 const struct tabla_mbhc_plug_type_cfg *plug_type =
7122 TABLA_MBHC_CAL_PLUG_TYPE_PTR(tabla->mbhc_cfg.calibration);
7123 const s16 v_hs_max = tabla_get_current_v_hs_max(tabla);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007124
7125 return (!(mic_mv > 10 && mic_mv < 80) && (mic_mv > plug_type->v_no_mic)
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007126 && (mic_mv < v_hs_max)) ? true : false;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007127}
7128
7129/* called under codec_resource_lock acquisition
7130 * returns true if mic voltage range is back to normal insertion
7131 * returns false either if timedout or removed */
7132static bool tabla_hs_remove_settle(struct snd_soc_codec *codec)
7133{
7134 int i;
7135 bool timedout, settled = false;
7136 s32 mic_mv[MBHC_NUM_DCE_PLUG_DETECT];
7137 short mb_v[MBHC_NUM_DCE_PLUG_DETECT];
7138 unsigned long retry = 0, timeout;
7139 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007140 const s16 v_hs_max = tabla_get_current_v_hs_max(tabla);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007141
7142 timeout = jiffies + msecs_to_jiffies(TABLA_HS_DETECT_PLUG_TIME_MS);
7143 while (!(timedout = time_after(jiffies, timeout))) {
7144 retry++;
7145 if (tabla->mbhc_cfg.gpio && tabla_hs_gpio_level_remove(tabla)) {
7146 pr_debug("%s: GPIO indicates removal\n", __func__);
7147 break;
7148 }
7149
7150 if (tabla->mbhc_cfg.gpio) {
7151 if (retry > 1)
7152 msleep(250);
7153 else
7154 msleep(50);
7155 }
7156
7157 if (tabla->mbhc_cfg.gpio && tabla_hs_gpio_level_remove(tabla)) {
7158 pr_debug("%s: GPIO indicates removal\n", __func__);
7159 break;
7160 }
7161
7162 for (i = 0; i < MBHC_NUM_DCE_PLUG_DETECT; i++) {
7163 mb_v[i] = tabla_codec_sta_dce(codec, 1, true);
7164 mic_mv[i] = tabla_codec_sta_dce_v(codec, 1 , mb_v[i]);
7165 pr_debug("%s : DCE run %lu, mic_mv = %d(%x)\n",
7166 __func__, retry, mic_mv[i], mb_v[i]);
7167 }
7168
7169 if (tabla->mbhc_cfg.gpio && tabla_hs_gpio_level_remove(tabla)) {
7170 pr_debug("%s: GPIO indicates removal\n", __func__);
7171 break;
7172 }
7173
7174 if (tabla->current_plug == PLUG_TYPE_NONE) {
7175 pr_debug("%s : headset/headphone is removed\n",
7176 __func__);
7177 break;
7178 }
7179
7180 for (i = 0; i < MBHC_NUM_DCE_PLUG_DETECT; i++)
7181 if (!is_valid_mic_voltage(codec, mic_mv[i]))
7182 break;
7183
7184 if (i == MBHC_NUM_DCE_PLUG_DETECT) {
7185 pr_debug("%s: MIC voltage settled\n", __func__);
7186 settled = true;
7187 msleep(200);
7188 break;
7189 }
7190
7191 /* only for non-GPIO remove irq */
7192 if (!tabla->mbhc_cfg.gpio) {
7193 for (i = 0; i < MBHC_NUM_DCE_PLUG_DETECT; i++)
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007194 if (mic_mv[i] < v_hs_max)
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007195 break;
7196 if (i == MBHC_NUM_DCE_PLUG_DETECT) {
7197 pr_debug("%s: Headset is removed\n", __func__);
7198 break;
7199 }
7200 }
7201 }
7202
7203 if (timedout)
7204 pr_debug("%s: Microphone did not settle in %d seconds\n",
7205 __func__, TABLA_HS_DETECT_PLUG_TIME_MS);
7206 return settled;
7207}
7208
7209/* called only from interrupt which is under codec_resource_lock acquisition */
7210static void tabla_hs_remove_irq_gpio(struct tabla_priv *priv)
7211{
7212 struct snd_soc_codec *codec = priv->codec;
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007213 pr_debug("%s: enter\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007214 if (tabla_hs_remove_settle(codec))
7215 tabla_codec_start_hs_polling(codec);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007216 pr_debug("%s: leave\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007217}
7218
7219/* called only from interrupt which is under codec_resource_lock acquisition */
7220static void tabla_hs_remove_irq_nogpio(struct tabla_priv *priv)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007221{
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08007222 short bias_value;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007223 bool removed = true;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007224 struct snd_soc_codec *codec = priv->codec;
Joonwoo Park0976d012011-12-22 11:48:18 -08007225 const struct tabla_mbhc_general_cfg *generic =
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007226 TABLA_MBHC_CAL_GENERAL_PTR(priv->mbhc_cfg.calibration);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08007227 int min_us = TABLA_FAKE_REMOVAL_MIN_PERIOD_MS * 1000;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007228
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007229 pr_debug("%s: enter\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007230 if (priv->current_plug != PLUG_TYPE_HEADSET) {
7231 pr_debug("%s(): Headset is not inserted, ignore removal\n",
7232 __func__);
7233 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL,
7234 0x08, 0x08);
7235 return;
7236 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007237
Joonwoo Park0976d012011-12-22 11:48:18 -08007238 usleep_range(generic->t_shutdown_plug_rem,
7239 generic->t_shutdown_plug_rem);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007240
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08007241 do {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007242 bias_value = tabla_codec_sta_dce(codec, 1, true);
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08007243 pr_debug("%s: DCE %d,%d, %d us left\n", __func__, bias_value,
7244 tabla_codec_sta_dce_v(codec, 1, bias_value), min_us);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007245 if (bias_value < tabla_get_current_v_ins(priv, false)) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007246 pr_debug("%s: checking false removal\n", __func__);
7247 msleep(500);
7248 removed = !tabla_hs_remove_settle(codec);
7249 pr_debug("%s: headset %sactually removed\n", __func__,
7250 removed ? "" : "not ");
Joonwoo Park6b9b03f2012-01-23 18:48:54 -08007251 break;
7252 }
7253 min_us -= priv->mbhc_data.t_dce;
7254 } while (min_us > 0);
Bradley Rubin89ffd0a2011-07-21 16:04:06 -07007255
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007256 if (removed) {
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007257 if (priv->mbhc_cfg.detect_extn_cable) {
7258 if (!tabla_hs_gpio_level_remove(priv)) {
7259 /*
7260 * extension cable is still plugged in
7261 * report it as LINEOUT device
7262 */
7263 tabla_codec_report_plug(codec, 1,
7264 SND_JACK_LINEOUT);
7265 tabla_codec_cleanup_hs_polling(codec);
7266 tabla_codec_enable_hs_detect(codec, 1,
7267 MBHC_USE_MB_TRIGGER,
7268 false);
7269 }
7270 } else {
7271 /* Cancel possibly running hs_detect_work */
7272 tabla_cancel_hs_detect_plug(priv,
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07007273 &priv->hs_correct_plug_work_nogpio);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007274 /*
7275 * If this removal is not false, first check the micbias
7276 * switch status and switch it to LDOH if it is already
7277 * switched to VDDIO.
7278 */
7279 tabla_codec_switch_micbias(codec, 0);
Joonwoo Park03324832012-03-19 19:36:16 -07007280
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007281 tabla_codec_report_plug(codec, 0, SND_JACK_HEADSET);
7282 tabla_codec_cleanup_hs_polling(codec);
7283 tabla_codec_enable_hs_detect(codec, 1,
7284 MBHC_USE_MB_TRIGGER |
7285 MBHC_USE_HPHL_TRIGGER,
7286 true);
7287 }
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007288 } else {
7289 tabla_codec_start_hs_polling(codec);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007290 }
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007291 pr_debug("%s: leave\n", __func__);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007292}
Joonwoo Park8b1f0982011-12-08 17:12:45 -08007293
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007294static irqreturn_t tabla_hs_remove_irq(int irq, void *data)
7295{
7296 struct tabla_priv *priv = data;
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007297 bool vddio;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007298 pr_debug("%s: enter, removal interrupt\n", __func__);
7299
7300 TABLA_ACQUIRE_LOCK(priv->codec_resource_lock);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007301 vddio = (priv->mbhc_data.micb_mv != VDDIO_MICBIAS_MV &&
7302 priv->mbhc_micbias_switched);
7303 if (vddio)
7304 __tabla_codec_switch_micbias(priv->codec, 0, false, true);
7305
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007306 if ((priv->mbhc_cfg.detect_extn_cable &&
7307 !tabla_hs_gpio_level_remove(priv)) ||
7308 !priv->mbhc_cfg.gpio) {
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007309 tabla_hs_remove_irq_nogpio(priv);
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007310 } else
7311 tabla_hs_remove_irq_gpio(priv);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007312
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007313 /* if driver turned off vddio switch and headset is not removed,
7314 * turn on the vddio switch back, if headset is removed then vddio
7315 * switch is off by time now and shouldn't be turn on again from here */
7316 if (vddio && priv->current_plug == PLUG_TYPE_HEADSET)
7317 __tabla_codec_switch_micbias(priv->codec, 1, true, true);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007318 TABLA_RELEASE_LOCK(priv->codec_resource_lock);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07007319
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007320 return IRQ_HANDLED;
7321}
7322
Joonwoo Park03324832012-03-19 19:36:16 -07007323void mbhc_insert_work(struct work_struct *work)
7324{
7325 struct delayed_work *dwork;
7326 struct tabla_priv *tabla;
7327 struct snd_soc_codec *codec;
7328 struct wcd9xxx *tabla_core;
7329
7330 dwork = to_delayed_work(work);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007331 tabla = container_of(dwork, struct tabla_priv, mbhc_insert_dwork);
Joonwoo Park03324832012-03-19 19:36:16 -07007332 codec = tabla->codec;
7333 tabla_core = dev_get_drvdata(codec->dev->parent);
7334
7335 pr_debug("%s:\n", __func__);
7336
7337 /* Turn off both HPH and MIC line schmitt triggers */
7338 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.mbhc_reg, 0x90, 0x00);
7339 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x13, 0x00);
7340 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.ctl_reg, 0x01, 0x00);
7341 wcd9xxx_disable_irq_sync(codec->control_data, TABLA_IRQ_MBHC_INSERTION);
7342 tabla_codec_detect_plug_type(codec);
7343 wcd9xxx_unlock_sleep(tabla_core);
7344}
7345
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007346static void tabla_hs_gpio_handler(struct snd_soc_codec *codec)
7347{
7348 bool insert;
7349 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
7350 bool is_removed = false;
7351
7352 pr_debug("%s: enter\n", __func__);
7353
7354 tabla->in_gpio_handler = true;
7355 /* Wait here for debounce time */
7356 usleep_range(TABLA_GPIO_IRQ_DEBOUNCE_TIME_US,
7357 TABLA_GPIO_IRQ_DEBOUNCE_TIME_US);
7358
7359 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
7360
7361 /* cancel pending button press */
7362 if (tabla_cancel_btn_work(tabla))
7363 pr_debug("%s: button press is canceled\n", __func__);
7364
7365 insert = (gpio_get_value_cansleep(tabla->mbhc_cfg.gpio) ==
7366 tabla->mbhc_cfg.gpio_level_insert);
7367 if ((tabla->current_plug == PLUG_TYPE_NONE) && insert) {
7368 tabla->lpi_enabled = false;
7369 wmb();
7370
7371 /* cancel detect plug */
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07007372 tabla_cancel_hs_detect_plug(tabla,
7373 &tabla->hs_correct_plug_work);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007374
7375 /* Disable Mic Bias pull down and HPH Switch to GND */
7376 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.ctl_reg, 0x01,
7377 0x00);
7378 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x01, 0x00);
7379 tabla_codec_detect_plug_type(codec);
7380 } else if ((tabla->current_plug != PLUG_TYPE_NONE) && !insert) {
7381 tabla->lpi_enabled = false;
7382 wmb();
7383
7384 /* cancel detect plug */
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07007385 tabla_cancel_hs_detect_plug(tabla,
7386 &tabla->hs_correct_plug_work);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007387
7388 if (tabla->current_plug == PLUG_TYPE_HEADPHONE) {
7389 tabla_codec_report_plug(codec, 0, SND_JACK_HEADPHONE);
7390 is_removed = true;
Joonwoo Park2cc13f02012-05-09 12:44:25 -07007391 } else if (tabla->current_plug == PLUG_TYPE_GND_MIC_SWAP) {
7392 tabla_codec_report_plug(codec, 0, SND_JACK_UNSUPPORTED);
7393 is_removed = true;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007394 } else if (tabla->current_plug == PLUG_TYPE_HEADSET) {
7395 tabla_codec_pause_hs_polling(codec);
7396 tabla_codec_cleanup_hs_polling(codec);
7397 tabla_codec_report_plug(codec, 0, SND_JACK_HEADSET);
7398 is_removed = true;
Ravi Kumar Alamanda113eb8a2012-08-15 18:39:47 -07007399 } else if (tabla->current_plug == PLUG_TYPE_HIGH_HPH) {
7400 tabla_codec_report_plug(codec, 0, SND_JACK_LINEOUT);
7401 is_removed = true;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007402 }
7403
7404 if (is_removed) {
7405 /* Enable Mic Bias pull down and HPH Switch to GND */
7406 snd_soc_update_bits(codec,
7407 tabla->mbhc_bias_regs.ctl_reg, 0x01,
7408 0x01);
7409 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x01,
7410 0x01);
7411 /* Make sure mic trigger is turned off */
7412 snd_soc_update_bits(codec,
7413 tabla->mbhc_bias_regs.ctl_reg,
7414 0x01, 0x01);
7415 snd_soc_update_bits(codec,
7416 tabla->mbhc_bias_regs.mbhc_reg,
7417 0x90, 0x00);
7418 /* Reset MBHC State Machine */
7419 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL,
7420 0x08, 0x08);
7421 snd_soc_update_bits(codec, TABLA_A_CDC_MBHC_CLK_CTL,
7422 0x08, 0x00);
7423 /* Turn off override */
7424 tabla_turn_onoff_override(codec, false);
7425 }
7426 }
7427
7428 tabla->in_gpio_handler = false;
7429 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
7430 pr_debug("%s: leave\n", __func__);
7431}
7432
7433static irqreturn_t tabla_mechanical_plug_detect_irq(int irq, void *data)
7434{
7435 int r = IRQ_HANDLED;
7436 struct snd_soc_codec *codec = data;
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -07007437 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007438
7439 if (unlikely(wcd9xxx_lock_sleep(codec->control_data) == false)) {
7440 pr_warn("%s: failed to hold suspend\n", __func__);
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -07007441 /*
7442 * Give up this IRQ for now and resend this IRQ so IRQ can be
7443 * handled after system resume
7444 */
7445 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
7446 tabla->gpio_irq_resend = true;
7447 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
7448 wake_lock_timeout(&tabla->irq_resend_wlock, HZ);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007449 r = IRQ_NONE;
7450 } else {
7451 tabla_hs_gpio_handler(codec);
7452 wcd9xxx_unlock_sleep(codec->control_data);
7453 }
7454
7455 return r;
7456}
7457
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07007458static void tabla_hs_correct_plug_nogpio(struct work_struct *work)
7459{
7460 struct tabla_priv *tabla;
7461 struct snd_soc_codec *codec;
7462 unsigned long timeout;
7463 int retry = 0;
7464 enum tabla_mbhc_plug_type plug_type;
7465 bool is_headset = false;
7466
7467 pr_debug("%s(): Poll Microphone voltage for %d seconds\n",
7468 __func__, TABLA_HS_DETECT_PLUG_TIME_MS / 1000);
7469
7470 tabla = container_of(work, struct tabla_priv,
7471 hs_correct_plug_work_nogpio);
7472 codec = tabla->codec;
7473
7474 /* Make sure the MBHC mux is connected to MIC Path */
7475 snd_soc_write(codec, TABLA_A_MBHC_SCALING_MUX_1, 0x84);
7476
7477 /* setup for microphone polling */
7478 tabla_turn_onoff_override(codec, true);
7479 tabla->mbhc_cfg.mclk_cb_fn(codec, 1, false);
7480
7481 timeout = jiffies + msecs_to_jiffies(TABLA_HS_DETECT_PLUG_TIME_MS);
7482 while (!time_after(jiffies, timeout)) {
7483 ++retry;
7484
7485 msleep(TABLA_HS_DETECT_PLUG_INERVAL_MS);
7486 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
7487 plug_type = tabla_codec_get_plug_type(codec, false);
7488 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
7489
7490 if (plug_type == PLUG_TYPE_HIGH_HPH
7491 || plug_type == PLUG_TYPE_INVALID) {
7492
7493 /* this means the plug is removed
7494 * End microphone polling and setup
7495 * for low power removal detection.
7496 */
7497 pr_debug("%s(): Plug may be removed, setup removal\n",
7498 __func__);
7499 break;
7500 } else if (plug_type == PLUG_TYPE_HEADSET) {
7501 /* Plug is corrected from headphone to headset,
7502 * report headset and end the polling
7503 */
7504 is_headset = true;
7505 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
7506 tabla_turn_onoff_override(codec, false);
7507 tabla_codec_report_plug(codec, 1, SND_JACK_HEADSET);
7508 tabla_codec_start_hs_polling(codec);
7509 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
7510 pr_debug("%s(): corrected from headphone to headset\n",
7511 __func__);
7512 break;
7513 }
7514 }
7515
7516 /* Undo setup for microphone polling depending
7517 * result from polling
7518 */
7519 tabla->mbhc_cfg.mclk_cb_fn(codec, 0, false);
7520 if (!is_headset) {
Bhalchandra Gajare18d10ee2012-08-23 13:44:07 -07007521 pr_debug("%s: Inserted headphone is not a headset\n",
7522 __func__);
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07007523 tabla_turn_onoff_override(codec, false);
7524 tabla_codec_cleanup_hs_polling(codec);
7525 tabla_codec_enable_hs_detect(codec, 0, 0, false);
7526 }
7527 wcd9xxx_unlock_sleep(codec->control_data);
7528}
7529
Joonwoo Park1305bab2012-05-21 15:08:42 -07007530static int tabla_mbhc_init_and_calibrate(struct tabla_priv *tabla)
7531{
7532 int ret = 0;
7533 struct snd_soc_codec *codec = tabla->codec;
7534
7535 tabla->mbhc_cfg.mclk_cb_fn(codec, 1, false);
7536 tabla_mbhc_init(codec);
7537 tabla_mbhc_cal(codec);
7538 tabla_mbhc_calc_thres(codec);
7539 tabla->mbhc_cfg.mclk_cb_fn(codec, 0, false);
7540 tabla_codec_calibrate_hs_polling(codec);
7541 if (!tabla->mbhc_cfg.gpio) {
Bhalchandra Gajareac0bddf2012-04-06 12:33:54 -07007542 INIT_WORK(&tabla->hs_correct_plug_work_nogpio,
7543 tabla_hs_correct_plug_nogpio);
Joonwoo Park1305bab2012-05-21 15:08:42 -07007544 ret = tabla_codec_enable_hs_detect(codec, 1,
7545 MBHC_USE_MB_TRIGGER |
7546 MBHC_USE_HPHL_TRIGGER,
7547 false);
7548
7549 if (IS_ERR_VALUE(ret))
7550 pr_err("%s: Failed to setup MBHC detection\n",
7551 __func__);
7552 } else {
7553 /* Enable Mic Bias pull down and HPH Switch to GND */
7554 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.ctl_reg,
7555 0x01, 0x01);
7556 snd_soc_update_bits(codec, TABLA_A_MBHC_HPH, 0x01, 0x01);
7557 INIT_WORK(&tabla->hs_correct_plug_work,
7558 tabla_hs_correct_gpio_plug);
7559 }
7560
7561 if (!IS_ERR_VALUE(ret)) {
7562 snd_soc_update_bits(codec, TABLA_A_RX_HPH_OCP_CTL, 0x10, 0x10);
7563 wcd9xxx_enable_irq(codec->control_data,
7564 TABLA_IRQ_HPH_PA_OCPL_FAULT);
7565 wcd9xxx_enable_irq(codec->control_data,
7566 TABLA_IRQ_HPH_PA_OCPR_FAULT);
7567
7568 if (tabla->mbhc_cfg.gpio) {
7569 ret = request_threaded_irq(tabla->mbhc_cfg.gpio_irq,
7570 NULL,
7571 tabla_mechanical_plug_detect_irq,
7572 (IRQF_TRIGGER_RISING |
7573 IRQF_TRIGGER_FALLING),
7574 "tabla-gpio", codec);
7575 if (!IS_ERR_VALUE(ret)) {
7576 ret = enable_irq_wake(tabla->mbhc_cfg.gpio_irq);
7577 /* Bootup time detection */
7578 tabla_hs_gpio_handler(codec);
7579 }
7580 }
7581 }
7582
7583 return ret;
7584}
7585
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007586static void mbhc_fw_read(struct work_struct *work)
7587{
7588 struct delayed_work *dwork;
7589 struct tabla_priv *tabla;
7590 struct snd_soc_codec *codec;
7591 const struct firmware *fw;
Joonwoo Park1305bab2012-05-21 15:08:42 -07007592 int ret = -1, retry = 0;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007593
7594 dwork = to_delayed_work(work);
Joonwoo Park1305bab2012-05-21 15:08:42 -07007595 tabla = container_of(dwork, struct tabla_priv, mbhc_firmware_dwork);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007596 codec = tabla->codec;
7597
7598 while (retry < MBHC_FW_READ_ATTEMPTS) {
7599 retry++;
7600 pr_info("%s:Attempt %d to request MBHC firmware\n",
7601 __func__, retry);
7602 ret = request_firmware(&fw, "wcd9310/wcd9310_mbhc.bin",
7603 codec->dev);
7604
7605 if (ret != 0) {
7606 usleep_range(MBHC_FW_READ_TIMEOUT,
Joonwoo Park1305bab2012-05-21 15:08:42 -07007607 MBHC_FW_READ_TIMEOUT);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007608 } else {
7609 pr_info("%s: MBHC Firmware read succesful\n", __func__);
7610 break;
7611 }
7612 }
7613
7614 if (ret != 0) {
7615 pr_err("%s: Cannot load MBHC firmware use default cal\n",
7616 __func__);
7617 } else if (tabla_mbhc_fw_validate(fw) == false) {
7618 pr_err("%s: Invalid MBHC cal data size use default cal\n",
7619 __func__);
7620 release_firmware(fw);
7621 } else {
7622 tabla->mbhc_cfg.calibration = (void *)fw->data;
7623 tabla->mbhc_fw = fw;
7624 }
7625
Joonwoo Park1305bab2012-05-21 15:08:42 -07007626 (void) tabla_mbhc_init_and_calibrate(tabla);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007627}
7628
Joonwoo Park03324832012-03-19 19:36:16 -07007629int tabla_hs_detect(struct snd_soc_codec *codec,
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007630 const struct tabla_mbhc_config *cfg)
Joonwoo Park03324832012-03-19 19:36:16 -07007631{
7632 struct tabla_priv *tabla;
7633 int rc = 0;
7634
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007635 if (!codec || !cfg->calibration) {
Joonwoo Park03324832012-03-19 19:36:16 -07007636 pr_err("Error: no codec or calibration\n");
7637 return -EINVAL;
7638 }
7639
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007640 if (cfg->mclk_rate != TABLA_MCLK_RATE_12288KHZ) {
7641 if (cfg->mclk_rate == TABLA_MCLK_RATE_9600KHZ)
Joonwoo Park03324832012-03-19 19:36:16 -07007642 pr_err("Error: clock rate %dHz is not yet supported\n",
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007643 cfg->mclk_rate);
Joonwoo Park03324832012-03-19 19:36:16 -07007644 else
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007645 pr_err("Error: unsupported clock rate %d\n",
7646 cfg->mclk_rate);
Joonwoo Park03324832012-03-19 19:36:16 -07007647 return -EINVAL;
7648 }
7649
7650 tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007651 tabla->mbhc_cfg = *cfg;
7652 tabla->in_gpio_handler = false;
7653 tabla->current_plug = PLUG_TYPE_NONE;
7654 tabla->lpi_enabled = false;
Joonwoo Park03324832012-03-19 19:36:16 -07007655 tabla_get_mbhc_micbias_regs(codec, &tabla->mbhc_bias_regs);
7656
7657 /* Put CFILT in fast mode by default */
7658 snd_soc_update_bits(codec, tabla->mbhc_bias_regs.cfilt_ctl,
7659 0x40, TABLA_CFILT_FAST_MODE);
7660 INIT_DELAYED_WORK(&tabla->mbhc_firmware_dwork, mbhc_fw_read);
7661 INIT_DELAYED_WORK(&tabla->mbhc_btn_dwork, btn_lpress_fn);
7662 INIT_WORK(&tabla->hphlocp_work, hphlocp_off_report);
7663 INIT_WORK(&tabla->hphrocp_work, hphrocp_off_report);
7664 INIT_DELAYED_WORK(&tabla->mbhc_insert_dwork, mbhc_insert_work);
7665
Joonwoo Park1305bab2012-05-21 15:08:42 -07007666 if (!tabla->mbhc_cfg.read_fw_bin)
7667 rc = tabla_mbhc_init_and_calibrate(tabla);
7668 else
Joonwoo Park03324832012-03-19 19:36:16 -07007669 schedule_delayed_work(&tabla->mbhc_firmware_dwork,
7670 usecs_to_jiffies(MBHC_FW_READ_TIMEOUT));
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07007671
Joonwoo Park03324832012-03-19 19:36:16 -07007672 return rc;
7673}
7674EXPORT_SYMBOL_GPL(tabla_hs_detect);
7675
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007676static irqreturn_t tabla_slimbus_irq(int irq, void *data)
7677{
7678 struct tabla_priv *priv = data;
7679 struct snd_soc_codec *codec = priv->codec;
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07007680 struct tabla_priv *tabla_p = snd_soc_codec_get_drvdata(codec);
7681 int i, j, port_id, k, ch_mask_temp;
Swaminathan Sathappan4bd38942012-07-17 11:31:31 -07007682 unsigned long slimbus_value;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007683 u8 val;
7684
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05307685 for (i = 0; i < WCD9XXX_SLIM_NUM_PORT_REG; i++) {
7686 slimbus_value = wcd9xxx_interface_reg_read(codec->control_data,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007687 TABLA_SLIM_PGD_PORT_INT_STATUS0 + i);
7688 for_each_set_bit(j, &slimbus_value, BITS_PER_BYTE) {
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05307689 val = wcd9xxx_interface_reg_read(codec->control_data,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007690 TABLA_SLIM_PGD_PORT_INT_SOURCE0 + i*8 + j);
7691 if (val & 0x1)
7692 pr_err_ratelimited("overflow error on port %x,"
7693 " value %x\n", i*8 + j, val);
7694 if (val & 0x2)
7695 pr_err_ratelimited("underflow error on port %x,"
7696 " value %x\n", i*8 + j, val);
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07007697 if (val & 0x4) {
7698 pr_debug("%s: port %x disconnect value %x\n",
7699 __func__, i*8 + j, val);
7700 port_id = i*8 + j;
7701 for (k = 0; k < ARRAY_SIZE(tabla_dai); k++) {
7702 ch_mask_temp = 1 << port_id;
7703 if (ch_mask_temp &
7704 tabla_p->dai[k].ch_mask) {
7705 tabla_p->dai[k].ch_mask &=
7706 ~ch_mask_temp;
7707 if (!tabla_p->dai[k].ch_mask)
7708 wake_up(
7709 &tabla_p->dai[k].dai_wait);
7710 }
7711 }
7712 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007713 }
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05307714 wcd9xxx_interface_reg_write(codec->control_data,
Swaminathan Sathappan4bd38942012-07-17 11:31:31 -07007715 TABLA_SLIM_PGD_PORT_INT_CLR0 + i, slimbus_value);
7716 val = 0x0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07007717 }
7718
7719 return IRQ_HANDLED;
7720}
7721
Patrick Lai3043fba2011-08-01 14:15:57 -07007722static int tabla_handle_pdata(struct tabla_priv *tabla)
7723{
7724 struct snd_soc_codec *codec = tabla->codec;
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05307725 struct wcd9xxx_pdata *pdata = tabla->pdata;
Patrick Lai3043fba2011-08-01 14:15:57 -07007726 int k1, k2, k3, rc = 0;
Santosh Mardi22920282011-10-26 02:38:40 +05307727 u8 leg_mode = pdata->amic_settings.legacy_mode;
7728 u8 txfe_bypass = pdata->amic_settings.txfe_enable;
7729 u8 txfe_buff = pdata->amic_settings.txfe_buff;
7730 u8 flag = pdata->amic_settings.use_pdata;
7731 u8 i = 0, j = 0;
7732 u8 val_txfe = 0, value = 0;
Patrick Lai3043fba2011-08-01 14:15:57 -07007733
7734 if (!pdata) {
7735 rc = -ENODEV;
7736 goto done;
7737 }
7738
7739 /* Make sure settings are correct */
7740 if ((pdata->micbias.ldoh_v > TABLA_LDOH_2P85_V) ||
7741 (pdata->micbias.bias1_cfilt_sel > TABLA_CFILT3_SEL) ||
7742 (pdata->micbias.bias2_cfilt_sel > TABLA_CFILT3_SEL) ||
7743 (pdata->micbias.bias3_cfilt_sel > TABLA_CFILT3_SEL) ||
7744 (pdata->micbias.bias4_cfilt_sel > TABLA_CFILT3_SEL)) {
7745 rc = -EINVAL;
7746 goto done;
7747 }
7748
7749 /* figure out k value */
7750 k1 = tabla_find_k_value(pdata->micbias.ldoh_v,
7751 pdata->micbias.cfilt1_mv);
7752 k2 = tabla_find_k_value(pdata->micbias.ldoh_v,
7753 pdata->micbias.cfilt2_mv);
7754 k3 = tabla_find_k_value(pdata->micbias.ldoh_v,
7755 pdata->micbias.cfilt3_mv);
7756
7757 if (IS_ERR_VALUE(k1) || IS_ERR_VALUE(k2) || IS_ERR_VALUE(k3)) {
7758 rc = -EINVAL;
7759 goto done;
7760 }
7761
7762 /* Set voltage level and always use LDO */
7763 snd_soc_update_bits(codec, TABLA_A_LDO_H_MODE_1, 0x0C,
7764 (pdata->micbias.ldoh_v << 2));
7765
7766 snd_soc_update_bits(codec, TABLA_A_MICB_CFILT_1_VAL, 0xFC,
7767 (k1 << 2));
7768 snd_soc_update_bits(codec, TABLA_A_MICB_CFILT_2_VAL, 0xFC,
7769 (k2 << 2));
7770 snd_soc_update_bits(codec, TABLA_A_MICB_CFILT_3_VAL, 0xFC,
7771 (k3 << 2));
7772
7773 snd_soc_update_bits(codec, TABLA_A_MICB_1_CTL, 0x60,
7774 (pdata->micbias.bias1_cfilt_sel << 5));
7775 snd_soc_update_bits(codec, TABLA_A_MICB_2_CTL, 0x60,
7776 (pdata->micbias.bias2_cfilt_sel << 5));
7777 snd_soc_update_bits(codec, TABLA_A_MICB_3_CTL, 0x60,
7778 (pdata->micbias.bias3_cfilt_sel << 5));
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08007779 snd_soc_update_bits(codec, tabla->reg_addr.micb_4_ctl, 0x60,
7780 (pdata->micbias.bias4_cfilt_sel << 5));
Patrick Lai3043fba2011-08-01 14:15:57 -07007781
Santosh Mardi22920282011-10-26 02:38:40 +05307782 for (i = 0; i < 6; j++, i += 2) {
7783 if (flag & (0x01 << i)) {
7784 value = (leg_mode & (0x01 << i)) ? 0x10 : 0x00;
7785 val_txfe = (txfe_bypass & (0x01 << i)) ? 0x20 : 0x00;
7786 val_txfe = val_txfe |
7787 ((txfe_buff & (0x01 << i)) ? 0x10 : 0x00);
7788 snd_soc_update_bits(codec, TABLA_A_TX_1_2_EN + j * 10,
7789 0x10, value);
7790 snd_soc_update_bits(codec,
7791 TABLA_A_TX_1_2_TEST_EN + j * 10,
7792 0x30, val_txfe);
7793 }
7794 if (flag & (0x01 << (i + 1))) {
7795 value = (leg_mode & (0x01 << (i + 1))) ? 0x01 : 0x00;
7796 val_txfe = (txfe_bypass &
7797 (0x01 << (i + 1))) ? 0x02 : 0x00;
7798 val_txfe |= (txfe_buff &
7799 (0x01 << (i + 1))) ? 0x01 : 0x00;
7800 snd_soc_update_bits(codec, TABLA_A_TX_1_2_EN + j * 10,
7801 0x01, value);
7802 snd_soc_update_bits(codec,
7803 TABLA_A_TX_1_2_TEST_EN + j * 10,
7804 0x03, val_txfe);
7805 }
7806 }
7807 if (flag & 0x40) {
7808 value = (leg_mode & 0x40) ? 0x10 : 0x00;
7809 value = value | ((txfe_bypass & 0x40) ? 0x02 : 0x00);
7810 value = value | ((txfe_buff & 0x40) ? 0x01 : 0x00);
7811 snd_soc_update_bits(codec, TABLA_A_TX_7_MBHC_EN,
7812 0x13, value);
7813 }
Patrick Lai49efeac2011-11-03 11:01:12 -07007814
7815 if (pdata->ocp.use_pdata) {
7816 /* not defined in CODEC specification */
7817 if (pdata->ocp.hph_ocp_limit == 1 ||
7818 pdata->ocp.hph_ocp_limit == 5) {
7819 rc = -EINVAL;
7820 goto done;
7821 }
7822 snd_soc_update_bits(codec, TABLA_A_RX_COM_OCP_CTL,
7823 0x0F, pdata->ocp.num_attempts);
7824 snd_soc_write(codec, TABLA_A_RX_COM_OCP_COUNT,
7825 ((pdata->ocp.run_time << 4) | pdata->ocp.wait_time));
7826 snd_soc_update_bits(codec, TABLA_A_RX_HPH_OCP_CTL,
7827 0xE0, (pdata->ocp.hph_ocp_limit << 5));
7828 }
Joonwoo Park03324832012-03-19 19:36:16 -07007829
7830 for (i = 0; i < ARRAY_SIZE(pdata->regulator); i++) {
7831 if (!strncmp(pdata->regulator[i].name, "CDC_VDDA_RX", 11)) {
7832 if (pdata->regulator[i].min_uV == 1800000 &&
7833 pdata->regulator[i].max_uV == 1800000) {
7834 snd_soc_write(codec, TABLA_A_BIAS_REF_CTL,
7835 0x1C);
7836 } else if (pdata->regulator[i].min_uV == 2200000 &&
7837 pdata->regulator[i].max_uV == 2200000) {
7838 snd_soc_write(codec, TABLA_A_BIAS_REF_CTL,
7839 0x1E);
7840 } else {
7841 pr_err("%s: unsupported CDC_VDDA_RX voltage "
7842 "min %d, max %d\n", __func__,
7843 pdata->regulator[i].min_uV,
7844 pdata->regulator[i].max_uV);
7845 rc = -EINVAL;
7846 }
7847 break;
7848 }
7849 }
Patrick Lai3043fba2011-08-01 14:15:57 -07007850done:
7851 return rc;
7852}
7853
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007854static const struct tabla_reg_mask_val tabla_1_1_reg_defaults[] = {
7855
7856 /* Tabla 1.1 MICBIAS changes */
7857 TABLA_REG_VAL(TABLA_A_MICB_1_INT_RBIAS, 0x24),
7858 TABLA_REG_VAL(TABLA_A_MICB_2_INT_RBIAS, 0x24),
7859 TABLA_REG_VAL(TABLA_A_MICB_3_INT_RBIAS, 0x24),
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007860
7861 /* Tabla 1.1 HPH changes */
7862 TABLA_REG_VAL(TABLA_A_RX_HPH_BIAS_PA, 0x57),
7863 TABLA_REG_VAL(TABLA_A_RX_HPH_BIAS_LDO, 0x56),
7864
7865 /* Tabla 1.1 EAR PA changes */
7866 TABLA_REG_VAL(TABLA_A_RX_EAR_BIAS_PA, 0xA6),
7867 TABLA_REG_VAL(TABLA_A_RX_EAR_GAIN, 0x02),
7868 TABLA_REG_VAL(TABLA_A_RX_EAR_VCM, 0x03),
7869
7870 /* Tabla 1.1 Lineout_5 Changes */
7871 TABLA_REG_VAL(TABLA_A_RX_LINE_5_GAIN, 0x10),
7872
7873 /* Tabla 1.1 RX Changes */
7874 TABLA_REG_VAL(TABLA_A_CDC_RX1_B5_CTL, 0x78),
7875 TABLA_REG_VAL(TABLA_A_CDC_RX2_B5_CTL, 0x78),
7876 TABLA_REG_VAL(TABLA_A_CDC_RX3_B5_CTL, 0x78),
7877 TABLA_REG_VAL(TABLA_A_CDC_RX4_B5_CTL, 0x78),
7878 TABLA_REG_VAL(TABLA_A_CDC_RX5_B5_CTL, 0x78),
7879 TABLA_REG_VAL(TABLA_A_CDC_RX6_B5_CTL, 0x78),
7880 TABLA_REG_VAL(TABLA_A_CDC_RX7_B5_CTL, 0x78),
7881
7882 /* Tabla 1.1 RX1 and RX2 Changes */
7883 TABLA_REG_VAL(TABLA_A_CDC_RX1_B6_CTL, 0xA0),
7884 TABLA_REG_VAL(TABLA_A_CDC_RX2_B6_CTL, 0xA0),
7885
7886 /* Tabla 1.1 RX3 to RX7 Changes */
7887 TABLA_REG_VAL(TABLA_A_CDC_RX3_B6_CTL, 0x80),
7888 TABLA_REG_VAL(TABLA_A_CDC_RX4_B6_CTL, 0x80),
7889 TABLA_REG_VAL(TABLA_A_CDC_RX5_B6_CTL, 0x80),
7890 TABLA_REG_VAL(TABLA_A_CDC_RX6_B6_CTL, 0x80),
7891 TABLA_REG_VAL(TABLA_A_CDC_RX7_B6_CTL, 0x80),
7892
7893 /* Tabla 1.1 CLASSG Changes */
7894 TABLA_REG_VAL(TABLA_A_CDC_CLSG_FREQ_THRESH_B3_CTL, 0x1B),
7895};
7896
7897static const struct tabla_reg_mask_val tabla_2_0_reg_defaults[] = {
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007898 /* Tabla 2.0 MICBIAS changes */
7899 TABLA_REG_VAL(TABLA_A_MICB_2_MBHC, 0x02),
7900};
7901
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08007902static const struct tabla_reg_mask_val tabla_1_x_only_reg_2_0_defaults[] = {
7903 TABLA_REG_VAL(TABLA_1_A_MICB_4_INT_RBIAS, 0x24),
7904};
7905
7906static const struct tabla_reg_mask_val tabla_2_only_reg_2_0_defaults[] = {
7907 TABLA_REG_VAL(TABLA_2_A_MICB_4_INT_RBIAS, 0x24),
7908};
7909
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007910static void tabla_update_reg_defaults(struct snd_soc_codec *codec)
7911{
7912 u32 i;
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05307913 struct wcd9xxx *tabla_core = dev_get_drvdata(codec->dev->parent);
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007914
7915 for (i = 0; i < ARRAY_SIZE(tabla_1_1_reg_defaults); i++)
7916 snd_soc_write(codec, tabla_1_1_reg_defaults[i].reg,
7917 tabla_1_1_reg_defaults[i].val);
7918
7919 for (i = 0; i < ARRAY_SIZE(tabla_2_0_reg_defaults); i++)
7920 snd_soc_write(codec, tabla_2_0_reg_defaults[i].reg,
7921 tabla_2_0_reg_defaults[i].val);
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08007922
7923 if (TABLA_IS_1_X(tabla_core->version)) {
7924 for (i = 0; i < ARRAY_SIZE(tabla_1_x_only_reg_2_0_defaults);
7925 i++)
7926 snd_soc_write(codec,
7927 tabla_1_x_only_reg_2_0_defaults[i].reg,
7928 tabla_1_x_only_reg_2_0_defaults[i].val);
7929 } else {
7930 for (i = 0; i < ARRAY_SIZE(tabla_2_only_reg_2_0_defaults); i++)
7931 snd_soc_write(codec,
7932 tabla_2_only_reg_2_0_defaults[i].reg,
7933 tabla_2_only_reg_2_0_defaults[i].val);
7934 }
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007935}
7936
7937static const struct tabla_reg_mask_val tabla_codec_reg_init_val[] = {
Patrick Laic7cae882011-11-18 11:52:49 -08007938 /* Initialize current threshold to 350MA
7939 * number of wait and run cycles to 4096
7940 */
Patrick Lai49efeac2011-11-03 11:01:12 -07007941 {TABLA_A_RX_HPH_OCP_CTL, 0xE0, 0x60},
Patrick Laic7cae882011-11-18 11:52:49 -08007942 {TABLA_A_RX_COM_OCP_COUNT, 0xFF, 0xFF},
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007943
Santosh Mardi32171012011-10-28 23:32:06 +05307944 {TABLA_A_QFUSE_CTL, 0xFF, 0x03},
7945
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007946 /* Initialize gain registers to use register gain */
7947 {TABLA_A_RX_HPH_L_GAIN, 0x10, 0x10},
7948 {TABLA_A_RX_HPH_R_GAIN, 0x10, 0x10},
7949 {TABLA_A_RX_LINE_1_GAIN, 0x10, 0x10},
7950 {TABLA_A_RX_LINE_2_GAIN, 0x10, 0x10},
7951 {TABLA_A_RX_LINE_3_GAIN, 0x10, 0x10},
7952 {TABLA_A_RX_LINE_4_GAIN, 0x10, 0x10},
7953
Kuirong Wangccb29c62012-06-15 11:09:07 -07007954 /* Set the MICBIAS default output as pull down*/
7955 {TABLA_A_MICB_1_CTL, 0x01, 0x01},
7956 {TABLA_A_MICB_2_CTL, 0x01, 0x01},
7957 {TABLA_A_MICB_3_CTL, 0x01, 0x01},
7958
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007959 /* Initialize mic biases to differential mode */
7960 {TABLA_A_MICB_1_INT_RBIAS, 0x24, 0x24},
7961 {TABLA_A_MICB_2_INT_RBIAS, 0x24, 0x24},
7962 {TABLA_A_MICB_3_INT_RBIAS, 0x24, 0x24},
Kiran Kandi1f6fd722011-08-11 10:36:11 -07007963
7964 {TABLA_A_CDC_CONN_CLSG_CTL, 0x3C, 0x14},
7965
7966 /* Use 16 bit sample size for TX1 to TX6 */
7967 {TABLA_A_CDC_CONN_TX_SB_B1_CTL, 0x30, 0x20},
7968 {TABLA_A_CDC_CONN_TX_SB_B2_CTL, 0x30, 0x20},
7969 {TABLA_A_CDC_CONN_TX_SB_B3_CTL, 0x30, 0x20},
7970 {TABLA_A_CDC_CONN_TX_SB_B4_CTL, 0x30, 0x20},
7971 {TABLA_A_CDC_CONN_TX_SB_B5_CTL, 0x30, 0x20},
7972 {TABLA_A_CDC_CONN_TX_SB_B6_CTL, 0x30, 0x20},
7973
7974 /* Use 16 bit sample size for TX7 to TX10 */
7975 {TABLA_A_CDC_CONN_TX_SB_B7_CTL, 0x60, 0x40},
7976 {TABLA_A_CDC_CONN_TX_SB_B8_CTL, 0x60, 0x40},
7977 {TABLA_A_CDC_CONN_TX_SB_B9_CTL, 0x60, 0x40},
7978 {TABLA_A_CDC_CONN_TX_SB_B10_CTL, 0x60, 0x40},
7979
7980 /* Use 16 bit sample size for RX */
7981 {TABLA_A_CDC_CONN_RX_SB_B1_CTL, 0xFF, 0xAA},
7982 {TABLA_A_CDC_CONN_RX_SB_B2_CTL, 0xFF, 0xAA},
7983
7984 /*enable HPF filter for TX paths */
7985 {TABLA_A_CDC_TX1_MUX_CTL, 0x8, 0x0},
7986 {TABLA_A_CDC_TX2_MUX_CTL, 0x8, 0x0},
7987 {TABLA_A_CDC_TX3_MUX_CTL, 0x8, 0x0},
7988 {TABLA_A_CDC_TX4_MUX_CTL, 0x8, 0x0},
7989 {TABLA_A_CDC_TX5_MUX_CTL, 0x8, 0x0},
7990 {TABLA_A_CDC_TX6_MUX_CTL, 0x8, 0x0},
7991 {TABLA_A_CDC_TX7_MUX_CTL, 0x8, 0x0},
7992 {TABLA_A_CDC_TX8_MUX_CTL, 0x8, 0x0},
7993 {TABLA_A_CDC_TX9_MUX_CTL, 0x8, 0x0},
7994 {TABLA_A_CDC_TX10_MUX_CTL, 0x8, 0x0},
Kiran Kandi0ba468f2012-05-08 11:45:05 -07007995
7996 /* config Decimator for DMIC CLK_MODE_1(3.072Mhz@12.88Mhz mclk) */
7997 {TABLA_A_CDC_TX1_DMIC_CTL, 0x1, 0x1},
7998 {TABLA_A_CDC_TX2_DMIC_CTL, 0x1, 0x1},
7999 {TABLA_A_CDC_TX3_DMIC_CTL, 0x1, 0x1},
8000 {TABLA_A_CDC_TX4_DMIC_CTL, 0x1, 0x1},
8001 {TABLA_A_CDC_TX5_DMIC_CTL, 0x1, 0x1},
8002 {TABLA_A_CDC_TX6_DMIC_CTL, 0x1, 0x1},
8003 {TABLA_A_CDC_TX7_DMIC_CTL, 0x1, 0x1},
8004 {TABLA_A_CDC_TX8_DMIC_CTL, 0x1, 0x1},
8005 {TABLA_A_CDC_TX9_DMIC_CTL, 0x1, 0x1},
8006 {TABLA_A_CDC_TX10_DMIC_CTL, 0x1, 0x1},
8007
8008 /* config DMIC clk to CLK_MODE_1 (3.072Mhz@12.88Mhz mclk) */
8009 {TABLA_A_CDC_CLK_DMIC_CTL, 0x2A, 0x2A},
8010
Kiran Kandi1f6fd722011-08-11 10:36:11 -07008011};
8012
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008013static const struct tabla_reg_mask_val tabla_1_x_codec_reg_init_val[] = {
Kuirong Wangccb29c62012-06-15 11:09:07 -07008014 /* Set the MICBIAS default output as pull down*/
8015 {TABLA_1_A_MICB_4_CTL, 0x01, 0x01},
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008016 /* Initialize mic biases to differential mode */
8017 {TABLA_1_A_MICB_4_INT_RBIAS, 0x24, 0x24},
8018};
8019
8020static const struct tabla_reg_mask_val tabla_2_higher_codec_reg_init_val[] = {
Kuirong Wangccb29c62012-06-15 11:09:07 -07008021
8022 /* Set the MICBIAS default output as pull down*/
8023 {TABLA_2_A_MICB_4_CTL, 0x01, 0x01},
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008024 /* Initialize mic biases to differential mode */
8025 {TABLA_2_A_MICB_4_INT_RBIAS, 0x24, 0x24},
8026};
8027
Kiran Kandi1f6fd722011-08-11 10:36:11 -07008028static void tabla_codec_init_reg(struct snd_soc_codec *codec)
8029{
8030 u32 i;
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308031 struct wcd9xxx *tabla_core = dev_get_drvdata(codec->dev->parent);
Kiran Kandi1f6fd722011-08-11 10:36:11 -07008032
8033 for (i = 0; i < ARRAY_SIZE(tabla_codec_reg_init_val); i++)
8034 snd_soc_update_bits(codec, tabla_codec_reg_init_val[i].reg,
8035 tabla_codec_reg_init_val[i].mask,
8036 tabla_codec_reg_init_val[i].val);
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008037 if (TABLA_IS_1_X(tabla_core->version)) {
8038 for (i = 0; i < ARRAY_SIZE(tabla_1_x_codec_reg_init_val); i++)
8039 snd_soc_update_bits(codec,
8040 tabla_1_x_codec_reg_init_val[i].reg,
8041 tabla_1_x_codec_reg_init_val[i].mask,
8042 tabla_1_x_codec_reg_init_val[i].val);
8043 } else {
8044 for (i = 0; i < ARRAY_SIZE(tabla_2_higher_codec_reg_init_val);
8045 i++)
8046 snd_soc_update_bits(codec,
8047 tabla_2_higher_codec_reg_init_val[i].reg,
8048 tabla_2_higher_codec_reg_init_val[i].mask,
8049 tabla_2_higher_codec_reg_init_val[i].val);
8050 }
8051}
8052
8053static void tabla_update_reg_address(struct tabla_priv *priv)
8054{
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308055 struct wcd9xxx *tabla_core = dev_get_drvdata(priv->codec->dev->parent);
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008056 struct tabla_reg_address *reg_addr = &priv->reg_addr;
8057
8058 if (TABLA_IS_1_X(tabla_core->version)) {
Joonwoo Parkcb7c8922012-02-16 23:12:59 -08008059 reg_addr->micb_4_mbhc = TABLA_1_A_MICB_4_MBHC;
8060 reg_addr->micb_4_int_rbias = TABLA_1_A_MICB_4_INT_RBIAS;
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008061 reg_addr->micb_4_ctl = TABLA_1_A_MICB_4_CTL;
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008062 } else if (TABLA_IS_2_0(tabla_core->version)) {
Joonwoo Parkcb7c8922012-02-16 23:12:59 -08008063 reg_addr->micb_4_mbhc = TABLA_2_A_MICB_4_MBHC;
8064 reg_addr->micb_4_int_rbias = TABLA_2_A_MICB_4_INT_RBIAS;
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008065 reg_addr->micb_4_ctl = TABLA_2_A_MICB_4_CTL;
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008066 }
Kiran Kandi1f6fd722011-08-11 10:36:11 -07008067}
8068
Joonwoo Park179b9ec2012-03-26 10:56:20 -07008069#ifdef CONFIG_DEBUG_FS
8070static int codec_debug_open(struct inode *inode, struct file *file)
8071{
8072 file->private_data = inode->i_private;
8073 return 0;
8074}
8075
8076static ssize_t codec_debug_write(struct file *filp,
8077 const char __user *ubuf, size_t cnt, loff_t *ppos)
8078{
8079 char lbuf[32];
8080 char *buf;
8081 int rc;
8082 struct tabla_priv *tabla = filp->private_data;
8083
8084 if (cnt > sizeof(lbuf) - 1)
8085 return -EINVAL;
8086
8087 rc = copy_from_user(lbuf, ubuf, cnt);
8088 if (rc)
8089 return -EFAULT;
8090
8091 lbuf[cnt] = '\0';
8092 buf = (char *)lbuf;
Joonwoo Park5bbcb0c2012-08-07 17:25:52 -07008093 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
8094 tabla->no_mic_headset_override =
8095 (*strsep(&buf, " ") == '0') ? false : true;
8096 if (tabla->no_mic_headset_override && tabla->mbhc_polling_active) {
8097 tabla_codec_pause_hs_polling(tabla->codec);
8098 tabla_codec_start_hs_polling(tabla->codec);
8099 }
8100 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
8101 return cnt;
Joonwoo Park179b9ec2012-03-26 10:56:20 -07008102}
8103
8104static ssize_t codec_mbhc_debug_read(struct file *file, char __user *buf,
8105 size_t count, loff_t *pos)
8106{
8107 const int size = 768;
8108 char buffer[size];
8109 int n = 0;
8110 struct tabla_priv *tabla = file->private_data;
8111 struct snd_soc_codec *codec = tabla->codec;
8112 const struct mbhc_internal_cal_data *p = &tabla->mbhc_data;
Joonwoo Parkcf473b42012-03-29 19:48:16 -07008113 const s16 v_ins_hu_cur = tabla_get_current_v_ins(tabla, true);
8114 const s16 v_ins_h_cur = tabla_get_current_v_ins(tabla, false);
Joonwoo Park179b9ec2012-03-26 10:56:20 -07008115
8116 n = scnprintf(buffer, size - n, "dce_z = %x(%dmv)\n", p->dce_z,
8117 tabla_codec_sta_dce_v(codec, 1, p->dce_z));
8118 n += scnprintf(buffer + n, size - n, "dce_mb = %x(%dmv)\n",
8119 p->dce_mb, tabla_codec_sta_dce_v(codec, 1, p->dce_mb));
8120 n += scnprintf(buffer + n, size - n, "sta_z = %x(%dmv)\n",
8121 p->sta_z, tabla_codec_sta_dce_v(codec, 0, p->sta_z));
8122 n += scnprintf(buffer + n, size - n, "sta_mb = %x(%dmv)\n",
8123 p->sta_mb, tabla_codec_sta_dce_v(codec, 0, p->sta_mb));
8124 n += scnprintf(buffer + n, size - n, "t_dce = %x\n", p->t_dce);
8125 n += scnprintf(buffer + n, size - n, "t_sta = %x\n", p->t_sta);
8126 n += scnprintf(buffer + n, size - n, "micb_mv = %dmv\n",
8127 p->micb_mv);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07008128 n += scnprintf(buffer + n, size - n, "v_ins_hu = %x(%dmv)%s\n",
Joonwoo Park179b9ec2012-03-26 10:56:20 -07008129 p->v_ins_hu,
Joonwoo Parkcf473b42012-03-29 19:48:16 -07008130 tabla_codec_sta_dce_v(codec, 0, p->v_ins_hu),
8131 p->v_ins_hu == v_ins_hu_cur ? "*" : "");
8132 n += scnprintf(buffer + n, size - n, "v_ins_h = %x(%dmv)%s\n",
8133 p->v_ins_h, tabla_codec_sta_dce_v(codec, 1, p->v_ins_h),
8134 p->v_ins_h == v_ins_h_cur ? "*" : "");
8135 n += scnprintf(buffer + n, size - n, "adj_v_ins_hu = %x(%dmv)%s\n",
8136 p->adj_v_ins_hu,
8137 tabla_codec_sta_dce_v(codec, 0, p->adj_v_ins_hu),
8138 p->adj_v_ins_hu == v_ins_hu_cur ? "*" : "");
8139 n += scnprintf(buffer + n, size - n, "adj_v_ins_h = %x(%dmv)%s\n",
8140 p->adj_v_ins_h,
8141 tabla_codec_sta_dce_v(codec, 1, p->adj_v_ins_h),
8142 p->adj_v_ins_h == v_ins_h_cur ? "*" : "");
Joonwoo Park179b9ec2012-03-26 10:56:20 -07008143 n += scnprintf(buffer + n, size - n, "v_b1_hu = %x(%dmv)\n",
8144 p->v_b1_hu, tabla_codec_sta_dce_v(codec, 0, p->v_b1_hu));
8145 n += scnprintf(buffer + n, size - n, "v_b1_h = %x(%dmv)\n",
8146 p->v_b1_h, tabla_codec_sta_dce_v(codec, 1, p->v_b1_h));
8147 n += scnprintf(buffer + n, size - n, "v_b1_huc = %x(%dmv)\n",
8148 p->v_b1_huc,
8149 tabla_codec_sta_dce_v(codec, 1, p->v_b1_huc));
8150 n += scnprintf(buffer + n, size - n, "v_brh = %x(%dmv)\n",
8151 p->v_brh, tabla_codec_sta_dce_v(codec, 1, p->v_brh));
8152 n += scnprintf(buffer + n, size - n, "v_brl = %x(%dmv)\n", p->v_brl,
8153 tabla_codec_sta_dce_v(codec, 0, p->v_brl));
8154 n += scnprintf(buffer + n, size - n, "v_no_mic = %x(%dmv)\n",
8155 p->v_no_mic,
8156 tabla_codec_sta_dce_v(codec, 0, p->v_no_mic));
8157 n += scnprintf(buffer + n, size - n, "npoll = %d\n", p->npoll);
8158 n += scnprintf(buffer + n, size - n, "nbounce_wait = %d\n",
8159 p->nbounce_wait);
Joonwoo Parkcf473b42012-03-29 19:48:16 -07008160 n += scnprintf(buffer + n, size - n, "v_inval_ins_low = %d\n",
8161 p->v_inval_ins_low);
8162 n += scnprintf(buffer + n, size - n, "v_inval_ins_high = %d\n",
8163 p->v_inval_ins_high);
Joonwoo Park2cc13f02012-05-09 12:44:25 -07008164 if (tabla->mbhc_cfg.gpio)
8165 n += scnprintf(buffer + n, size - n, "GPIO insert = %d\n",
8166 tabla_hs_gpio_level_remove(tabla));
Joonwoo Park179b9ec2012-03-26 10:56:20 -07008167 buffer[n] = 0;
8168
8169 return simple_read_from_buffer(buf, count, pos, buffer, n);
8170}
8171
8172static const struct file_operations codec_debug_ops = {
8173 .open = codec_debug_open,
8174 .write = codec_debug_write,
8175};
8176
8177static const struct file_operations codec_mbhc_debug_ops = {
8178 .open = codec_debug_open,
8179 .read = codec_mbhc_debug_read,
8180};
8181#endif
8182
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008183static int tabla_codec_probe(struct snd_soc_codec *codec)
8184{
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308185 struct wcd9xxx *control;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008186 struct tabla_priv *tabla;
8187 struct snd_soc_dapm_context *dapm = &codec->dapm;
8188 int ret = 0;
8189 int i;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08008190 int ch_cnt;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008191
8192 codec->control_data = dev_get_drvdata(codec->dev->parent);
8193 control = codec->control_data;
8194
8195 tabla = kzalloc(sizeof(struct tabla_priv), GFP_KERNEL);
8196 if (!tabla) {
8197 dev_err(codec->dev, "Failed to allocate private data\n");
8198 return -ENOMEM;
8199 }
Kiran Kandid8cf5212012-03-02 15:34:53 -08008200 for (i = 0 ; i < NUM_DECIMATORS; i++) {
8201 tx_hpf_work[i].tabla = tabla;
8202 tx_hpf_work[i].decimator = i + 1;
8203 INIT_DELAYED_WORK(&tx_hpf_work[i].dwork,
8204 tx_hpf_corner_freq_callback);
8205 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008206
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07008207 /* Make sure mbhc micbias register addresses are zeroed out */
8208 memset(&tabla->mbhc_bias_regs, 0,
8209 sizeof(struct mbhc_micbias_regs));
Bhalchandra Gajared9ebb6c2011-10-03 19:54:41 -07008210 tabla->mbhc_micbias_switched = false;
Bhalchandra Gajare02d90cd2011-09-30 16:14:00 -07008211
Joonwoo Park0976d012011-12-22 11:48:18 -08008212 /* Make sure mbhc intenal calibration data is zeroed out */
8213 memset(&tabla->mbhc_data, 0,
8214 sizeof(struct mbhc_internal_cal_data));
Joonwoo Park433149a2012-01-11 09:53:54 -08008215 tabla->mbhc_data.t_sta_dce = DEFAULT_DCE_STA_WAIT;
Joonwoo Park0976d012011-12-22 11:48:18 -08008216 tabla->mbhc_data.t_dce = DEFAULT_DCE_WAIT;
8217 tabla->mbhc_data.t_sta = DEFAULT_STA_WAIT;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008218 snd_soc_codec_set_drvdata(codec, tabla);
8219
Kiran Kandi6fae8bf2011-08-15 10:36:42 -07008220 tabla->mclk_enabled = false;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008221 tabla->bandgap_type = TABLA_BANDGAP_OFF;
8222 tabla->clock_active = false;
8223 tabla->config_mode_active = false;
8224 tabla->mbhc_polling_active = false;
Joonwoo Parkf4267c22012-01-10 13:25:24 -08008225 tabla->mbhc_fake_ins_start = 0;
Bradley Rubincb3950a2011-08-18 13:07:26 -07008226 tabla->no_mic_headset_override = false;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07008227 tabla->hs_polling_irq_prepared = false;
8228 mutex_init(&tabla->codec_resource_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008229 tabla->codec = codec;
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07008230 tabla->mbhc_state = MBHC_STATE_NONE;
Joonwoo Park03324832012-03-19 19:36:16 -07008231 tabla->mbhc_last_resume = 0;
Kuirong Wang0f8ade32012-02-27 16:29:45 -08008232 for (i = 0; i < COMPANDER_MAX; i++) {
8233 tabla->comp_enabled[i] = 0;
8234 tabla->comp_fs[i] = COMPANDER_FS_48KHZ;
8235 }
Patrick Lai3043fba2011-08-01 14:15:57 -07008236 tabla->pdata = dev_get_platdata(codec->dev->parent);
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308237 tabla->intf_type = wcd9xxx_get_intf_type();
Bhalchandra Gajareb0f15132012-02-07 15:00:21 -08008238 tabla->aux_pga_cnt = 0;
8239 tabla->aux_l_gain = 0x1F;
8240 tabla->aux_r_gain = 0x1F;
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008241 tabla_update_reg_address(tabla);
Santosh Mardi22920282011-10-26 02:38:40 +05308242 tabla_update_reg_defaults(codec);
8243 tabla_codec_init_reg(codec);
Santosh Mardi22920282011-10-26 02:38:40 +05308244 ret = tabla_handle_pdata(tabla);
Patrick Lai3043fba2011-08-01 14:15:57 -07008245 if (IS_ERR_VALUE(ret)) {
8246 pr_err("%s: bad pdata\n", __func__);
8247 goto err_pdata;
8248 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008249
Steve Mucklef132c6c2012-06-06 18:30:57 -07008250// snd_soc_add_codec_controls(codec, tabla_snd_controls,
8251// ARRAY_SIZE(tabla_snd_controls));
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008252 if (TABLA_IS_1_X(control->version))
Steve Mucklef132c6c2012-06-06 18:30:57 -07008253 snd_soc_add_codec_controls(codec, tabla_1_x_snd_controls,
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008254 ARRAY_SIZE(tabla_1_x_snd_controls));
8255 else
Steve Mucklef132c6c2012-06-06 18:30:57 -07008256 snd_soc_add_codec_controls(codec, tabla_2_higher_snd_controls,
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008257 ARRAY_SIZE(tabla_2_higher_snd_controls));
8258
Steve Mucklef132c6c2012-06-06 18:30:57 -07008259// snd_soc_dapm_new_controls(dapm, tabla_dapm_widgets,
8260// ARRAY_SIZE(tabla_dapm_widgets));
Kiran Kandi93923902012-06-20 17:00:25 -07008261
8262 snd_soc_dapm_new_controls(dapm, tabla_dapm_aif_in_widgets,
8263 ARRAY_SIZE(tabla_dapm_aif_in_widgets));
8264
8265 snd_soc_dapm_new_controls(dapm, tabla_dapm_aif_out_widgets,
8266 ARRAY_SIZE(tabla_dapm_aif_out_widgets));
8267
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008268 if (TABLA_IS_1_X(control->version))
8269 snd_soc_dapm_new_controls(dapm, tabla_1_x_dapm_widgets,
8270 ARRAY_SIZE(tabla_1_x_dapm_widgets));
8271 else
8272 snd_soc_dapm_new_controls(dapm, tabla_2_higher_dapm_widgets,
8273 ARRAY_SIZE(tabla_2_higher_dapm_widgets));
8274
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308275 if (tabla->intf_type == WCD9XXX_INTERFACE_TYPE_I2C) {
Santosh Mardie15e2302011-11-15 10:39:23 +05308276 snd_soc_dapm_new_controls(dapm, tabla_dapm_i2s_widgets,
8277 ARRAY_SIZE(tabla_dapm_i2s_widgets));
8278 snd_soc_dapm_add_routes(dapm, audio_i2s_map,
8279 ARRAY_SIZE(audio_i2s_map));
8280 }
Steve Mucklef132c6c2012-06-06 18:30:57 -07008281// snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map));
Kiran Kandi8b3a8302011-09-27 16:13:28 -07008282
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008283 if (TABLA_IS_1_X(control->version)) {
Kiran Kandi7a9fd902011-11-14 13:51:45 -08008284 snd_soc_dapm_add_routes(dapm, tabla_1_x_lineout_2_to_4_map,
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008285 ARRAY_SIZE(tabla_1_x_lineout_2_to_4_map));
8286 } else if (TABLA_IS_2_0(control->version)) {
Kiran Kandi7a9fd902011-11-14 13:51:45 -08008287 snd_soc_dapm_add_routes(dapm, tabla_2_x_lineout_2_to_4_map,
Joonwoo Park6c1ebb62012-01-16 19:08:43 -08008288 ARRAY_SIZE(tabla_2_x_lineout_2_to_4_map));
Kiran Kandi7a9fd902011-11-14 13:51:45 -08008289 } else {
8290 pr_err("%s : ERROR. Unsupported Tabla version 0x%2x\n",
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308291 __func__, control->version);
Kiran Kandi7a9fd902011-11-14 13:51:45 -08008292 goto err_pdata;
8293 }
8294
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008295 snd_soc_dapm_sync(dapm);
8296
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308297 ret = wcd9xxx_request_irq(codec->control_data, TABLA_IRQ_MBHC_INSERTION,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008298 tabla_hs_insert_irq, "Headset insert detect", tabla);
8299 if (ret) {
8300 pr_err("%s: Failed to request irq %d\n", __func__,
8301 TABLA_IRQ_MBHC_INSERTION);
8302 goto err_insert_irq;
8303 }
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308304 wcd9xxx_disable_irq(codec->control_data, TABLA_IRQ_MBHC_INSERTION);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008305
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308306 ret = wcd9xxx_request_irq(codec->control_data, TABLA_IRQ_MBHC_REMOVAL,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008307 tabla_hs_remove_irq, "Headset remove detect", tabla);
8308 if (ret) {
8309 pr_err("%s: Failed to request irq %d\n", __func__,
8310 TABLA_IRQ_MBHC_REMOVAL);
8311 goto err_remove_irq;
8312 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008313
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308314 ret = wcd9xxx_request_irq(codec->control_data, TABLA_IRQ_MBHC_POTENTIAL,
Bradley Rubincb1e2732011-06-23 16:49:20 -07008315 tabla_dce_handler, "DC Estimation detect", tabla);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008316 if (ret) {
8317 pr_err("%s: Failed to request irq %d\n", __func__,
8318 TABLA_IRQ_MBHC_POTENTIAL);
8319 goto err_potential_irq;
8320 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008321
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308322 ret = wcd9xxx_request_irq(codec->control_data, TABLA_IRQ_MBHC_RELEASE,
Bradley Rubincb1e2732011-06-23 16:49:20 -07008323 tabla_release_handler, "Button Release detect", tabla);
8324 if (ret) {
8325 pr_err("%s: Failed to request irq %d\n", __func__,
8326 TABLA_IRQ_MBHC_RELEASE);
8327 goto err_release_irq;
8328 }
8329
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308330 ret = wcd9xxx_request_irq(codec->control_data, TABLA_IRQ_SLIMBUS,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008331 tabla_slimbus_irq, "SLIMBUS Slave", tabla);
8332 if (ret) {
8333 pr_err("%s: Failed to request irq %d\n", __func__,
8334 TABLA_IRQ_SLIMBUS);
8335 goto err_slimbus_irq;
8336 }
8337
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308338 for (i = 0; i < WCD9XXX_SLIM_NUM_PORT_REG; i++)
8339 wcd9xxx_interface_reg_write(codec->control_data,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008340 TABLA_SLIM_PGD_PORT_INT_EN0 + i, 0xFF);
8341
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308342 ret = wcd9xxx_request_irq(codec->control_data,
Patrick Lai49efeac2011-11-03 11:01:12 -07008343 TABLA_IRQ_HPH_PA_OCPL_FAULT, tabla_hphl_ocp_irq,
8344 "HPH_L OCP detect", tabla);
8345 if (ret) {
8346 pr_err("%s: Failed to request irq %d\n", __func__,
8347 TABLA_IRQ_HPH_PA_OCPL_FAULT);
8348 goto err_hphl_ocp_irq;
8349 }
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308350 wcd9xxx_disable_irq(codec->control_data, TABLA_IRQ_HPH_PA_OCPL_FAULT);
Patrick Lai49efeac2011-11-03 11:01:12 -07008351
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308352 ret = wcd9xxx_request_irq(codec->control_data,
Patrick Lai49efeac2011-11-03 11:01:12 -07008353 TABLA_IRQ_HPH_PA_OCPR_FAULT, tabla_hphr_ocp_irq,
8354 "HPH_R OCP detect", tabla);
8355 if (ret) {
8356 pr_err("%s: Failed to request irq %d\n", __func__,
8357 TABLA_IRQ_HPH_PA_OCPR_FAULT);
8358 goto err_hphr_ocp_irq;
8359 }
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308360 wcd9xxx_disable_irq(codec->control_data, TABLA_IRQ_HPH_PA_OCPR_FAULT);
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -07008361
8362 /*
8363 * Register suspend lock and notifier to resend edge triggered
8364 * gpio IRQs
8365 */
8366 wake_lock_init(&tabla->irq_resend_wlock, WAKE_LOCK_SUSPEND,
8367 "tabla_gpio_irq_resend");
8368 tabla->gpio_irq_resend = false;
8369
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08008370 for (i = 0; i < ARRAY_SIZE(tabla_dai); i++) {
8371 switch (tabla_dai[i].id) {
8372 case AIF1_PB:
8373 ch_cnt = tabla_dai[i].playback.channels_max;
8374 break;
8375 case AIF1_CAP:
8376 ch_cnt = tabla_dai[i].capture.channels_max;
8377 break;
Neema Shettyd3a89262012-02-16 10:23:50 -08008378 case AIF2_PB:
8379 ch_cnt = tabla_dai[i].playback.channels_max;
8380 break;
Kiran Kandi1e6371d2012-03-29 11:48:57 -07008381 case AIF2_CAP:
8382 ch_cnt = tabla_dai[i].capture.channels_max;
8383 break;
Kiran Kandia9fffe92012-05-20 23:42:30 -07008384 case AIF3_PB:
8385 ch_cnt = tabla_dai[i].playback.channels_max;
8386 break;
Neema Shetty3fb1b802012-04-27 13:53:24 -07008387 case AIF3_CAP:
8388 ch_cnt = tabla_dai[i].capture.channels_max;
8389 break;
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08008390 default:
8391 continue;
8392 }
8393 tabla->dai[i].ch_num = kzalloc((sizeof(unsigned int)*
8394 ch_cnt), GFP_KERNEL);
Swaminathan Sathappan2aa4c042012-06-26 13:08:45 -07008395 init_waitqueue_head(&tabla->dai[i].dai_wait);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08008396 }
Patrick Lai49efeac2011-11-03 11:01:12 -07008397
Bradley Rubincb3950a2011-08-18 13:07:26 -07008398#ifdef CONFIG_DEBUG_FS
Joonwoo Park179b9ec2012-03-26 10:56:20 -07008399 if (ret == 0) {
8400 tabla->debugfs_poke =
8401 debugfs_create_file("TRRS", S_IFREG | S_IRUGO, NULL, tabla,
8402 &codec_debug_ops);
8403 tabla->debugfs_mbhc =
8404 debugfs_create_file("tabla_mbhc", S_IFREG | S_IRUGO,
8405 NULL, tabla, &codec_mbhc_debug_ops);
8406 }
Bradley Rubincb3950a2011-08-18 13:07:26 -07008407#endif
Steve Mucklef132c6c2012-06-06 18:30:57 -07008408 codec->ignore_pmdown_time = 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008409 return ret;
8410
Patrick Lai49efeac2011-11-03 11:01:12 -07008411err_hphr_ocp_irq:
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308412 wcd9xxx_free_irq(codec->control_data,
8413 TABLA_IRQ_HPH_PA_OCPL_FAULT, tabla);
Patrick Lai49efeac2011-11-03 11:01:12 -07008414err_hphl_ocp_irq:
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308415 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_SLIMBUS, tabla);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008416err_slimbus_irq:
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308417 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_MBHC_RELEASE, tabla);
Bradley Rubincb1e2732011-06-23 16:49:20 -07008418err_release_irq:
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308419 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_MBHC_POTENTIAL, tabla);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008420err_potential_irq:
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308421 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_MBHC_REMOVAL, tabla);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008422err_remove_irq:
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308423 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_MBHC_INSERTION, tabla);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008424err_insert_irq:
Patrick Lai3043fba2011-08-01 14:15:57 -07008425err_pdata:
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07008426 mutex_destroy(&tabla->codec_resource_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008427 kfree(tabla);
8428 return ret;
8429}
8430static int tabla_codec_remove(struct snd_soc_codec *codec)
8431{
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08008432 int i;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008433 struct tabla_priv *tabla = snd_soc_codec_get_drvdata(codec);
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -07008434
8435 wake_lock_destroy(&tabla->irq_resend_wlock);
8436
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308437 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_SLIMBUS, tabla);
8438 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_MBHC_RELEASE, tabla);
8439 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_MBHC_POTENTIAL, tabla);
8440 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_MBHC_REMOVAL, tabla);
8441 wcd9xxx_free_irq(codec->control_data, TABLA_IRQ_MBHC_INSERTION, tabla);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07008442 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008443 tabla_codec_disable_clock_block(codec);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07008444 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008445 tabla_codec_enable_bandgap(codec, TABLA_BANDGAP_OFF);
Patrick Lai64b43262011-12-06 17:29:15 -08008446 if (tabla->mbhc_fw)
8447 release_firmware(tabla->mbhc_fw);
Bharath Ramachandramurthy9c79f132011-11-28 11:18:57 -08008448 for (i = 0; i < ARRAY_SIZE(tabla_dai); i++)
8449 kfree(tabla->dai[i].ch_num);
Joonwoo Parkd7cf2e92012-03-19 19:38:23 -07008450 mutex_destroy(&tabla->codec_resource_lock);
Joonwoo Park179b9ec2012-03-26 10:56:20 -07008451#ifdef CONFIG_DEBUG_FS
8452 debugfs_remove(tabla->debugfs_poke);
8453 debugfs_remove(tabla->debugfs_mbhc);
8454#endif
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008455 kfree(tabla);
8456 return 0;
8457}
8458static struct snd_soc_codec_driver soc_codec_dev_tabla = {
8459 .probe = tabla_codec_probe,
8460 .remove = tabla_codec_remove,
8461 .read = tabla_read,
8462 .write = tabla_write,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008463 .readable_register = tabla_readable,
8464 .volatile_register = tabla_volatile,
8465
8466 .reg_cache_size = TABLA_CACHE_SIZE,
8467 .reg_cache_default = tabla_reg_defaults,
8468 .reg_word_size = 1,
Steve Mucklef132c6c2012-06-06 18:30:57 -07008469 .controls = tabla_snd_controls,
8470 .num_controls = ARRAY_SIZE(tabla_snd_controls),
8471 .dapm_widgets = tabla_dapm_widgets,
8472 .num_dapm_widgets = ARRAY_SIZE(tabla_dapm_widgets),
8473 .dapm_routes = audio_map,
8474 .num_dapm_routes = ARRAY_SIZE(audio_map),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008475};
Bradley Rubincb3950a2011-08-18 13:07:26 -07008476
Joonwoo Park8b1f0982011-12-08 17:12:45 -08008477#ifdef CONFIG_PM
8478static int tabla_suspend(struct device *dev)
8479{
Joonwoo Park816b8e62012-01-23 16:03:21 -08008480 dev_dbg(dev, "%s: system suspend\n", __func__);
8481 return 0;
Joonwoo Park8b1f0982011-12-08 17:12:45 -08008482}
8483
8484static int tabla_resume(struct device *dev)
8485{
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -07008486 int irq;
Joonwoo Park03324832012-03-19 19:36:16 -07008487 struct platform_device *pdev = to_platform_device(dev);
8488 struct tabla_priv *tabla = platform_get_drvdata(pdev);
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -07008489
Joonwoo Parkbf4e4842012-09-20 11:14:15 -07008490 dev_dbg(dev, "%s: system resume tabla %p\n", __func__, tabla);
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -07008491 if (tabla) {
8492 TABLA_ACQUIRE_LOCK(tabla->codec_resource_lock);
Joonwoo Parkbf4e4842012-09-20 11:14:15 -07008493 tabla->mbhc_last_resume = jiffies;
Joonwoo Parkb16c0bd2012-10-04 16:57:52 -07008494 if (tabla->gpio_irq_resend) {
8495 WARN_ON(!tabla->mbhc_cfg.gpio_irq);
8496 tabla->gpio_irq_resend = false;
8497
8498 irq = tabla->mbhc_cfg.gpio_irq;
8499 pr_debug("%s: Resending GPIO IRQ %d\n", __func__, irq);
8500 irq_set_pending(irq);
8501 check_irq_resend(irq_to_desc(irq), irq);
8502
8503 /* release suspend lock */
8504 wake_unlock(&tabla->irq_resend_wlock);
8505 }
8506 TABLA_RELEASE_LOCK(tabla->codec_resource_lock);
8507 }
8508
Joonwoo Park816b8e62012-01-23 16:03:21 -08008509 return 0;
Joonwoo Park8b1f0982011-12-08 17:12:45 -08008510}
8511
8512static const struct dev_pm_ops tabla_pm_ops = {
8513 .suspend = tabla_suspend,
8514 .resume = tabla_resume,
8515};
8516#endif
8517
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008518static int __devinit tabla_probe(struct platform_device *pdev)
8519{
Santosh Mardie15e2302011-11-15 10:39:23 +05308520 int ret = 0;
Steve Mucklef132c6c2012-06-06 18:30:57 -07008521 pr_err("tabla_probe\n");
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308522 if (wcd9xxx_get_intf_type() == WCD9XXX_INTERFACE_TYPE_SLIMBUS)
Santosh Mardie15e2302011-11-15 10:39:23 +05308523 ret = snd_soc_register_codec(&pdev->dev, &soc_codec_dev_tabla,
8524 tabla_dai, ARRAY_SIZE(tabla_dai));
Asish Bhattacharyab1aeae22012-02-15 08:29:28 +05308525 else if (wcd9xxx_get_intf_type() == WCD9XXX_INTERFACE_TYPE_I2C)
Santosh Mardie15e2302011-11-15 10:39:23 +05308526 ret = snd_soc_register_codec(&pdev->dev, &soc_codec_dev_tabla,
8527 tabla_i2s_dai, ARRAY_SIZE(tabla_i2s_dai));
8528 return ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008529}
8530static int __devexit tabla_remove(struct platform_device *pdev)
8531{
8532 snd_soc_unregister_codec(&pdev->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008533 return 0;
8534}
8535static struct platform_driver tabla_codec_driver = {
8536 .probe = tabla_probe,
8537 .remove = tabla_remove,
8538 .driver = {
8539 .name = "tabla_codec",
8540 .owner = THIS_MODULE,
Joonwoo Park8b1f0982011-12-08 17:12:45 -08008541#ifdef CONFIG_PM
8542 .pm = &tabla_pm_ops,
8543#endif
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008544 },
8545};
8546
Kuirong Wangcd4b6da2012-01-16 22:54:45 -08008547static struct platform_driver tabla1x_codec_driver = {
8548 .probe = tabla_probe,
8549 .remove = tabla_remove,
8550 .driver = {
8551 .name = "tabla1x_codec",
8552 .owner = THIS_MODULE,
8553#ifdef CONFIG_PM
8554 .pm = &tabla_pm_ops,
8555#endif
8556 },
8557};
8558
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008559static int __init tabla_codec_init(void)
8560{
Kuirong Wangcd4b6da2012-01-16 22:54:45 -08008561 int rtn = platform_driver_register(&tabla_codec_driver);
8562 if (rtn == 0) {
8563 rtn = platform_driver_register(&tabla1x_codec_driver);
8564 if (rtn != 0)
8565 platform_driver_unregister(&tabla_codec_driver);
8566 }
8567 return rtn;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008568}
8569
8570static void __exit tabla_codec_exit(void)
8571{
Kuirong Wangcd4b6da2012-01-16 22:54:45 -08008572 platform_driver_unregister(&tabla1x_codec_driver);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07008573 platform_driver_unregister(&tabla_codec_driver);
8574}
8575
8576module_init(tabla_codec_init);
8577module_exit(tabla_codec_exit);
8578
8579MODULE_DESCRIPTION("Tabla codec driver");
8580MODULE_VERSION("1.0");
8581MODULE_LICENSE("GPL v2");