ASoC: Add support for CS42L42 codec
authorJames Schulman <james.schulman@cirrus.com>
Mon, 7 Nov 2016 20:38:37 +0000 (14:38 -0600)
committerMark Brown <broonie@kernel.org>
Wed, 9 Nov 2016 13:50:14 +0000 (13:50 +0000)
Add support for Cirrus Logic CS42L42 codec. SoundWire support
is not enabled. Features support for I2C control and I2S audio.

Signed-off-by: James Schulman <james.schulman@cirrus.com>
Signed-off-by: Mark Brown <broonie@kernel.org>
include/dt-bindings/sound/cs42l42.h [new file with mode: 0644]
sound/soc/codecs/Kconfig
sound/soc/codecs/Makefile
sound/soc/codecs/cs42l42.c [new file with mode: 0644]
sound/soc/codecs/cs42l42.h [new file with mode: 0644]

diff --git a/include/dt-bindings/sound/cs42l42.h b/include/dt-bindings/sound/cs42l42.h
new file mode 100644 (file)
index 0000000..399a123
--- /dev/null
@@ -0,0 +1,73 @@
+/*
+ * cs42l42.h -- CS42L42 ALSA SoC audio driver DT bindings header
+ *
+ * Copyright 2016 Cirrus Logic, Inc.
+ *
+ * Author: James Schulman <james.schulman@cirrus.com>
+ * Author: Brian Austin <brian.austin@cirrus.com>
+ * Author: Michael White <michael.white@cirrus.com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ */
+
+#ifndef __DT_CS42L42_H
+#define __DT_CS42L42_H
+
+/* HPOUT Load Capacity */
+#define CS42L42_HPOUT_LOAD_1NF         0
+#define CS42L42_HPOUT_LOAD_10NF                1
+
+/* HPOUT Clamp to GND Overide */
+#define CS42L42_HPOUT_CLAMP_EN         0
+#define CS42L42_HPOUT_CLAMP_DIS                1
+
+/* Tip Sense Inversion */
+#define CS42L42_TS_INV_DIS                     0
+#define CS42L42_TS_INV_EN                      1
+
+/* Tip Sense Debounce */
+#define CS42L42_TS_DBNCE_0                     0
+#define CS42L42_TS_DBNCE_125                   1
+#define CS42L42_TS_DBNCE_250                   2
+#define CS42L42_TS_DBNCE_500                   3
+#define CS42L42_TS_DBNCE_750                   4
+#define CS42L42_TS_DBNCE_1000                  5
+#define CS42L42_TS_DBNCE_1250                  6
+#define CS42L42_TS_DBNCE_1500                  7
+
+/* Button Press Software Debounce Times */
+#define CS42L42_BTN_DET_INIT_DBNCE_MIN         0
+#define CS42L42_BTN_DET_INIT_DBNCE_DEFAULT     100
+#define CS42L42_BTN_DET_INIT_DBNCE_MAX         200
+
+#define CS42L42_BTN_DET_EVENT_DBNCE_MIN                0
+#define CS42L42_BTN_DET_EVENT_DBNCE_DEFAULT    10
+#define CS42L42_BTN_DET_EVENT_DBNCE_MAX                20
+
+/* Button Detect Level Sensitivities */
+#define CS42L42_NUM_BIASES             4
+
+#define CS42L42_HS_DET_LEVEL_15                0x0F
+#define CS42L42_HS_DET_LEVEL_8         0x08
+#define CS42L42_HS_DET_LEVEL_4         0x04
+#define CS42L42_HS_DET_LEVEL_1         0x01
+
+#define CS42L42_HS_DET_LEVEL_MIN       0
+#define CS42L42_HS_DET_LEVEL_MAX       0x3F
+
+/* HS Bias Ramp Rate */
+
+#define CS42L42_HSBIAS_RAMP_FAST_RISE_SLOW_FALL                0
+#define CS42L42_HSBIAS_RAMP_FAST                       1
+#define CS42L42_HSBIAS_RAMP_SLOW                       2
+#define CS42L42_HSBIAS_RAMP_SLOWEST                    3
+
+#define CS42L42_HSBIAS_RAMP_TIME0                      10
+#define CS42L42_HSBIAS_RAMP_TIME1                      40
+#define CS42L42_HSBIAS_RAMP_TIME2                      90
+#define CS42L42_HSBIAS_RAMP_TIME3                      170
+
+#endif /* __DT_CS42L42_H */
index bf2f26d..2aa709a 100644 (file)
@@ -49,6 +49,7 @@ config SND_SOC_ALL_CODECS
        select SND_SOC_CS35L32 if I2C
        select SND_SOC_CS35L33 if I2C
        select SND_SOC_CS35L34 if I2C
+       select SND_SOC_CS42L42 if I2C
        select SND_SOC_CS42L51_I2C if I2C
        select SND_SOC_CS42L52 if I2C && INPUT
        select SND_SOC_CS42L56 if I2C && INPUT
@@ -404,6 +405,10 @@ config SND_SOC_CS35L34
        tristate "Cirrus Logic CS35L34 CODEC"
        depends on I2C
 
+config SND_SOC_CS42L42
+       tristate "Cirrus Logic CS42L42 CODEC"
+       depends on I2C
+
 config SND_SOC_CS42L51
        tristate
 
index 3f0cd90..4bcafc3 100644 (file)
@@ -39,6 +39,7 @@ snd-soc-cq93vc-objs := cq93vc.o
 snd-soc-cs35l32-objs := cs35l32.o
 snd-soc-cs35l33-objs := cs35l33.o
 snd-soc-cs35l34-objs := cs35l34.o
+snd-soc-cs42l42-objs := cs42l42.o
 snd-soc-cs42l51-objs := cs42l51.o
 snd-soc-cs42l51-i2c-objs := cs42l51-i2c.o
 snd-soc-cs42l52-objs := cs42l52.o
@@ -265,6 +266,7 @@ obj-$(CONFIG_SND_SOC_CQ0093VC) += snd-soc-cq93vc.o
 obj-$(CONFIG_SND_SOC_CS35L32)  += snd-soc-cs35l32.o
 obj-$(CONFIG_SND_SOC_CS35L33)  += snd-soc-cs35l33.o
 obj-$(CONFIG_SND_SOC_CS35L34)  += snd-soc-cs35l34.o
+obj-$(CONFIG_SND_SOC_CS42L42)  += snd-soc-cs42l42.o
 obj-$(CONFIG_SND_SOC_CS42L51)  += snd-soc-cs42l51.o
 obj-$(CONFIG_SND_SOC_CS42L51_I2C)      += snd-soc-cs42l51-i2c.o
 obj-$(CONFIG_SND_SOC_CS42L52)  += snd-soc-cs42l52.o
diff --git a/sound/soc/codecs/cs42l42.c b/sound/soc/codecs/cs42l42.c
new file mode 100644 (file)
index 0000000..f3c10f5
--- /dev/null
@@ -0,0 +1,1987 @@
+/*
+ * cs42l42.c -- CS42L42 ALSA SoC audio driver
+ *
+ * Copyright 2016 Cirrus Logic, Inc.
+ *
+ * Author: James Schulman <james.schulman@cirrus.com>
+ * Author: Brian Austin <brian.austin@cirrus.com>
+ * Author: Michael White <michael.white@cirrus.com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ */
+
+#include <linux/module.h>
+#include <linux/moduleparam.h>
+#include <linux/version.h>
+#include <linux/kernel.h>
+#include <linux/init.h>
+#include <linux/delay.h>
+#include <linux/i2c.h>
+#include <linux/gpio.h>
+#include <linux/regmap.h>
+#include <linux/slab.h>
+#include <linux/platform_device.h>
+#include <linux/regulator/consumer.h>
+#include <linux/gpio/consumer.h>
+#include <linux/of.h>
+#include <linux/of_gpio.h>
+#include <linux/of_device.h>
+#include <sound/core.h>
+#include <sound/pcm.h>
+#include <sound/pcm_params.h>
+#include <sound/soc.h>
+#include <sound/soc-dapm.h>
+#include <sound/initval.h>
+#include <sound/tlv.h>
+#include <dt-bindings/sound/cs42l42.h>
+
+#include "cs42l42.h"
+
+static const struct reg_default cs42l42_reg_defaults[] = {
+       { CS42L42_FRZ_CTL,                      0x00 },
+       { CS42L42_SRC_CTL,                      0x10 },
+       { CS42L42_MCLK_STATUS,                  0x02 },
+       { CS42L42_MCLK_CTL,                     0x02 },
+       { CS42L42_SFTRAMP_RATE,                 0xA4 },
+       { CS42L42_I2C_DEBOUNCE,                 0x88 },
+       { CS42L42_I2C_STRETCH,                  0x03 },
+       { CS42L42_I2C_TIMEOUT,                  0xB7 },
+       { CS42L42_PWR_CTL1,                     0xFF },
+       { CS42L42_PWR_CTL2,                     0x84 },
+       { CS42L42_PWR_CTL3,                     0x20 },
+       { CS42L42_RSENSE_CTL1,                  0x40 },
+       { CS42L42_RSENSE_CTL2,                  0x00 },
+       { CS42L42_OSC_SWITCH,                   0x00 },
+       { CS42L42_OSC_SWITCH_STATUS,            0x05 },
+       { CS42L42_RSENSE_CTL3,                  0x1B },
+       { CS42L42_TSENSE_CTL,                   0x1B },
+       { CS42L42_TSRS_INT_DISABLE,             0x00 },
+       { CS42L42_TRSENSE_STATUS,               0x00 },
+       { CS42L42_HSDET_CTL1,                   0x77 },
+       { CS42L42_HSDET_CTL2,                   0x00 },
+       { CS42L42_HS_SWITCH_CTL,                0xF3 },
+       { CS42L42_HS_DET_STATUS,                0x00 },
+       { CS42L42_HS_CLAMP_DISABLE,             0x00 },
+       { CS42L42_MCLK_SRC_SEL,                 0x00 },
+       { CS42L42_SPDIF_CLK_CFG,                0x00 },
+       { CS42L42_FSYNC_PW_LOWER,               0x00 },
+       { CS42L42_FSYNC_PW_UPPER,               0x00 },
+       { CS42L42_FSYNC_P_LOWER,                0xF9 },
+       { CS42L42_FSYNC_P_UPPER,                0x00 },
+       { CS42L42_ASP_CLK_CFG,                  0x00 },
+       { CS42L42_ASP_FRM_CFG,                  0x10 },
+       { CS42L42_FS_RATE_EN,                   0x00 },
+       { CS42L42_IN_ASRC_CLK,                  0x00 },
+       { CS42L42_OUT_ASRC_CLK,                 0x00 },
+       { CS42L42_PLL_DIV_CFG1,                 0x00 },
+       { CS42L42_ADC_OVFL_STATUS,              0x00 },
+       { CS42L42_MIXER_STATUS,                 0x00 },
+       { CS42L42_SRC_STATUS,                   0x00 },
+       { CS42L42_ASP_RX_STATUS,                0x00 },
+       { CS42L42_ASP_TX_STATUS,                0x00 },
+       { CS42L42_CODEC_STATUS,                 0x00 },
+       { CS42L42_DET_INT_STATUS1,              0x00 },
+       { CS42L42_DET_INT_STATUS2,              0x00 },
+       { CS42L42_SRCPL_INT_STATUS,             0x00 },
+       { CS42L42_VPMON_STATUS,                 0x00 },
+       { CS42L42_PLL_LOCK_STATUS,              0x00 },
+       { CS42L42_TSRS_PLUG_STATUS,             0x00 },
+       { CS42L42_ADC_OVFL_INT_MASK,            0x01 },
+       { CS42L42_MIXER_INT_MASK,               0x0F },
+       { CS42L42_SRC_INT_MASK,                 0x0F },
+       { CS42L42_ASP_RX_INT_MASK,              0x1F },
+       { CS42L42_ASP_TX_INT_MASK,              0x0F },
+       { CS42L42_CODEC_INT_MASK,               0x03 },
+       { CS42L42_SRCPL_INT_MASK,               0xFF },
+       { CS42L42_VPMON_INT_MASK,               0x01 },
+       { CS42L42_PLL_LOCK_INT_MASK,            0x01 },
+       { CS42L42_TSRS_PLUG_INT_MASK,           0x0F },
+       { CS42L42_PLL_CTL1,                     0x00 },
+       { CS42L42_PLL_DIV_FRAC0,                0x00 },
+       { CS42L42_PLL_DIV_FRAC1,                0x00 },
+       { CS42L42_PLL_DIV_FRAC2,                0x00 },
+       { CS42L42_PLL_DIV_INT,                  0x40 },
+       { CS42L42_PLL_CTL3,                     0x10 },
+       { CS42L42_PLL_CAL_RATIO,                0x80 },
+       { CS42L42_PLL_CTL4,                     0x03 },
+       { CS42L42_LOAD_DET_RCSTAT,              0x00 },
+       { CS42L42_LOAD_DET_DONE,                0x00 },
+       { CS42L42_LOAD_DET_EN,                  0x00 },
+       { CS42L42_HSBIAS_SC_AUTOCTL,            0x03 },
+       { CS42L42_WAKE_CTL,                     0xC0 },
+       { CS42L42_ADC_DISABLE_MUTE,             0x00 },
+       { CS42L42_TIPSENSE_CTL,                 0x02 },
+       { CS42L42_MISC_DET_CTL,                 0x03 },
+       { CS42L42_MIC_DET_CTL1,                 0x1F },
+       { CS42L42_MIC_DET_CTL2,                 0x2F },
+       { CS42L42_DET_STATUS1,                  0x00 },
+       { CS42L42_DET_STATUS2,                  0x00 },
+       { CS42L42_DET_INT1_MASK,                0xE0 },
+       { CS42L42_DET_INT2_MASK,                0xFF },
+       { CS42L42_HS_BIAS_CTL,                  0xC2 },
+       { CS42L42_ADC_CTL,                      0x00 },
+       { CS42L42_ADC_VOLUME,                   0x00 },
+       { CS42L42_ADC_WNF_HPF_CTL,              0x71 },
+       { CS42L42_DAC_CTL1,                     0x00 },
+       { CS42L42_DAC_CTL2,                     0x02 },
+       { CS42L42_HP_CTL,                       0x0D },
+       { CS42L42_CLASSH_CTL,                   0x07 },
+       { CS42L42_MIXER_CHA_VOL,                0x3F },
+       { CS42L42_MIXER_ADC_VOL,                0x3F },
+       { CS42L42_MIXER_CHB_VOL,                0x3F },
+       { CS42L42_EQ_COEF_IN0,                  0x22 },
+       { CS42L42_EQ_COEF_IN1,                  0x00 },
+       { CS42L42_EQ_COEF_IN2,                  0x00 },
+       { CS42L42_EQ_COEF_IN3,                  0x00 },
+       { CS42L42_EQ_COEF_RW,                   0x00 },
+       { CS42L42_EQ_COEF_OUT0,                 0x00 },
+       { CS42L42_EQ_COEF_OUT1,                 0x00 },
+       { CS42L42_EQ_COEF_OUT2,                 0x00 },
+       { CS42L42_EQ_COEF_OUT3,                 0x00 },
+       { CS42L42_EQ_INIT_STAT,                 0x00 },
+       { CS42L42_EQ_START_FILT,                0x00 },
+       { CS42L42_EQ_MUTE_CTL,                  0x00 },
+       { CS42L42_SP_RX_CH_SEL,                 0x04 },
+       { CS42L42_SP_RX_ISOC_CTL,               0x04 },
+       { CS42L42_SP_RX_FS,                     0x8C },
+       { CS42l42_SPDIF_CH_SEL,                 0x0E },
+       { CS42L42_SP_TX_ISOC_CTL,               0x04 },
+       { CS42L42_SP_TX_FS,                     0xCC },
+       { CS42L42_SPDIF_SW_CTL1,                0x3F },
+       { CS42L42_SRC_SDIN_FS,                  0x40 },
+       { CS42L42_SRC_SDOUT_FS,                 0x40 },
+       { CS42L42_SPDIF_CTL1,                   0x01 },
+       { CS42L42_SPDIF_CTL2,                   0x00 },
+       { CS42L42_SPDIF_CTL3,                   0x00 },
+       { CS42L42_SPDIF_CTL4,                   0x42 },
+       { CS42L42_ASP_TX_SZ_EN,                 0x00 },
+       { CS42L42_ASP_TX_CH_EN,                 0x00 },
+       { CS42L42_ASP_TX_CH_AP_RES,             0x0F },
+       { CS42L42_ASP_TX_CH1_BIT_MSB,           0x00 },
+       { CS42L42_ASP_TX_CH1_BIT_LSB,           0x00 },
+       { CS42L42_ASP_TX_HIZ_DLY_CFG,           0x00 },
+       { CS42L42_ASP_TX_CH2_BIT_MSB,           0x00 },
+       { CS42L42_ASP_TX_CH2_BIT_LSB,           0x00 },
+       { CS42L42_ASP_RX_DAI0_EN,               0x00 },
+       { CS42L42_ASP_RX_DAI0_CH1_AP_RES,       0x03 },
+       { CS42L42_ASP_RX_DAI0_CH1_BIT_MSB,      0x00 },
+       { CS42L42_ASP_RX_DAI0_CH1_BIT_LSB,      0x00 },
+       { CS42L42_ASP_RX_DAI0_CH2_AP_RES,       0x03 },
+       { CS42L42_ASP_RX_DAI0_CH2_BIT_MSB,      0x00 },
+       { CS42L42_ASP_RX_DAI0_CH2_BIT_LSB,      0x00 },
+       { CS42L42_ASP_RX_DAI0_CH3_AP_RES,       0x03 },
+       { CS42L42_ASP_RX_DAI0_CH3_BIT_MSB,      0x00 },
+       { CS42L42_ASP_RX_DAI0_CH3_BIT_LSB,      0x00 },
+       { CS42L42_ASP_RX_DAI0_CH4_AP_RES,       0x03 },
+       { CS42L42_ASP_RX_DAI0_CH4_BIT_MSB,      0x00 },
+       { CS42L42_ASP_RX_DAI0_CH4_BIT_LSB,      0x00 },
+       { CS42L42_ASP_RX_DAI1_CH1_AP_RES,       0x03 },
+       { CS42L42_ASP_RX_DAI1_CH1_BIT_MSB,      0x00 },
+       { CS42L42_ASP_RX_DAI1_CH1_BIT_LSB,      0x00 },
+       { CS42L42_ASP_RX_DAI1_CH2_AP_RES,       0x03 },
+       { CS42L42_ASP_RX_DAI1_CH2_BIT_MSB,      0x00 },
+       { CS42L42_ASP_RX_DAI1_CH2_BIT_LSB,      0x00 },
+       { CS42L42_SUB_REVID,                    0x03 },
+};
+
+static bool cs42l42_readable_register(struct device *dev, unsigned int reg)
+{
+       switch (reg) {
+       case CS42L42_PAGE_REGISTER:
+       case CS42L42_DEVID_AB:
+       case CS42L42_DEVID_CD:
+       case CS42L42_DEVID_E:
+       case CS42L42_FABID:
+       case CS42L42_REVID:
+       case CS42L42_FRZ_CTL:
+       case CS42L42_SRC_CTL:
+       case CS42L42_MCLK_STATUS:
+       case CS42L42_MCLK_CTL:
+       case CS42L42_SFTRAMP_RATE:
+       case CS42L42_I2C_DEBOUNCE:
+       case CS42L42_I2C_STRETCH:
+       case CS42L42_I2C_TIMEOUT:
+       case CS42L42_PWR_CTL1:
+       case CS42L42_PWR_CTL2:
+       case CS42L42_PWR_CTL3:
+       case CS42L42_RSENSE_CTL1:
+       case CS42L42_RSENSE_CTL2:
+       case CS42L42_OSC_SWITCH:
+       case CS42L42_OSC_SWITCH_STATUS:
+       case CS42L42_RSENSE_CTL3:
+       case CS42L42_TSENSE_CTL:
+       case CS42L42_TSRS_INT_DISABLE:
+       case CS42L42_TRSENSE_STATUS:
+       case CS42L42_HSDET_CTL1:
+       case CS42L42_HSDET_CTL2:
+       case CS42L42_HS_SWITCH_CTL:
+       case CS42L42_HS_DET_STATUS:
+       case CS42L42_HS_CLAMP_DISABLE:
+       case CS42L42_MCLK_SRC_SEL:
+       case CS42L42_SPDIF_CLK_CFG:
+       case CS42L42_FSYNC_PW_LOWER:
+       case CS42L42_FSYNC_PW_UPPER:
+       case CS42L42_FSYNC_P_LOWER:
+       case CS42L42_FSYNC_P_UPPER:
+       case CS42L42_ASP_CLK_CFG:
+       case CS42L42_ASP_FRM_CFG:
+       case CS42L42_FS_RATE_EN:
+       case CS42L42_IN_ASRC_CLK:
+       case CS42L42_OUT_ASRC_CLK:
+       case CS42L42_PLL_DIV_CFG1:
+       case CS42L42_ADC_OVFL_STATUS:
+       case CS42L42_MIXER_STATUS:
+       case CS42L42_SRC_STATUS:
+       case CS42L42_ASP_RX_STATUS:
+       case CS42L42_ASP_TX_STATUS:
+       case CS42L42_CODEC_STATUS:
+       case CS42L42_DET_INT_STATUS1:
+       case CS42L42_DET_INT_STATUS2:
+       case CS42L42_SRCPL_INT_STATUS:
+       case CS42L42_VPMON_STATUS:
+       case CS42L42_PLL_LOCK_STATUS:
+       case CS42L42_TSRS_PLUG_STATUS:
+       case CS42L42_ADC_OVFL_INT_MASK:
+       case CS42L42_MIXER_INT_MASK:
+       case CS42L42_SRC_INT_MASK:
+       case CS42L42_ASP_RX_INT_MASK:
+       case CS42L42_ASP_TX_INT_MASK:
+       case CS42L42_CODEC_INT_MASK:
+       case CS42L42_SRCPL_INT_MASK:
+       case CS42L42_VPMON_INT_MASK:
+       case CS42L42_PLL_LOCK_INT_MASK:
+       case CS42L42_TSRS_PLUG_INT_MASK:
+       case CS42L42_PLL_CTL1:
+       case CS42L42_PLL_DIV_FRAC0:
+       case CS42L42_PLL_DIV_FRAC1:
+       case CS42L42_PLL_DIV_FRAC2:
+       case CS42L42_PLL_DIV_INT:
+       case CS42L42_PLL_CTL3:
+       case CS42L42_PLL_CAL_RATIO:
+       case CS42L42_PLL_CTL4:
+       case CS42L42_LOAD_DET_RCSTAT:
+       case CS42L42_LOAD_DET_DONE:
+       case CS42L42_LOAD_DET_EN:
+       case CS42L42_HSBIAS_SC_AUTOCTL:
+       case CS42L42_WAKE_CTL:
+       case CS42L42_ADC_DISABLE_MUTE:
+       case CS42L42_TIPSENSE_CTL:
+       case CS42L42_MISC_DET_CTL:
+       case CS42L42_MIC_DET_CTL1:
+       case CS42L42_MIC_DET_CTL2:
+       case CS42L42_DET_STATUS1:
+       case CS42L42_DET_STATUS2:
+       case CS42L42_DET_INT1_MASK:
+       case CS42L42_DET_INT2_MASK:
+       case CS42L42_HS_BIAS_CTL:
+       case CS42L42_ADC_CTL:
+       case CS42L42_ADC_VOLUME:
+       case CS42L42_ADC_WNF_HPF_CTL:
+       case CS42L42_DAC_CTL1:
+       case CS42L42_DAC_CTL2:
+       case CS42L42_HP_CTL:
+       case CS42L42_CLASSH_CTL:
+       case CS42L42_MIXER_CHA_VOL:
+       case CS42L42_MIXER_ADC_VOL:
+       case CS42L42_MIXER_CHB_VOL:
+       case CS42L42_EQ_COEF_IN0:
+       case CS42L42_EQ_COEF_IN1:
+       case CS42L42_EQ_COEF_IN2:
+       case CS42L42_EQ_COEF_IN3:
+       case CS42L42_EQ_COEF_RW:
+       case CS42L42_EQ_COEF_OUT0:
+       case CS42L42_EQ_COEF_OUT1:
+       case CS42L42_EQ_COEF_OUT2:
+       case CS42L42_EQ_COEF_OUT3:
+       case CS42L42_EQ_INIT_STAT:
+       case CS42L42_EQ_START_FILT:
+       case CS42L42_EQ_MUTE_CTL:
+       case CS42L42_SP_RX_CH_SEL:
+       case CS42L42_SP_RX_ISOC_CTL:
+       case CS42L42_SP_RX_FS:
+       case CS42l42_SPDIF_CH_SEL:
+       case CS42L42_SP_TX_ISOC_CTL:
+       case CS42L42_SP_TX_FS:
+       case CS42L42_SPDIF_SW_CTL1:
+       case CS42L42_SRC_SDIN_FS:
+       case CS42L42_SRC_SDOUT_FS:
+       case CS42L42_SPDIF_CTL1:
+       case CS42L42_SPDIF_CTL2:
+       case CS42L42_SPDIF_CTL3:
+       case CS42L42_SPDIF_CTL4:
+       case CS42L42_ASP_TX_SZ_EN:
+       case CS42L42_ASP_TX_CH_EN:
+       case CS42L42_ASP_TX_CH_AP_RES:
+       case CS42L42_ASP_TX_CH1_BIT_MSB:
+       case CS42L42_ASP_TX_CH1_BIT_LSB:
+       case CS42L42_ASP_TX_HIZ_DLY_CFG:
+       case CS42L42_ASP_TX_CH2_BIT_MSB:
+       case CS42L42_ASP_TX_CH2_BIT_LSB:
+       case CS42L42_ASP_RX_DAI0_EN:
+       case CS42L42_ASP_RX_DAI0_CH1_AP_RES:
+       case CS42L42_ASP_RX_DAI0_CH1_BIT_MSB:
+       case CS42L42_ASP_RX_DAI0_CH1_BIT_LSB:
+       case CS42L42_ASP_RX_DAI0_CH2_AP_RES:
+       case CS42L42_ASP_RX_DAI0_CH2_BIT_MSB:
+       case CS42L42_ASP_RX_DAI0_CH2_BIT_LSB:
+       case CS42L42_ASP_RX_DAI0_CH3_AP_RES:
+       case CS42L42_ASP_RX_DAI0_CH3_BIT_MSB:
+       case CS42L42_ASP_RX_DAI0_CH3_BIT_LSB:
+       case CS42L42_ASP_RX_DAI0_CH4_AP_RES:
+       case CS42L42_ASP_RX_DAI0_CH4_BIT_MSB:
+       case CS42L42_ASP_RX_DAI0_CH4_BIT_LSB:
+       case CS42L42_ASP_RX_DAI1_CH1_AP_RES:
+       case CS42L42_ASP_RX_DAI1_CH1_BIT_MSB:
+       case CS42L42_ASP_RX_DAI1_CH1_BIT_LSB:
+       case CS42L42_ASP_RX_DAI1_CH2_AP_RES:
+       case CS42L42_ASP_RX_DAI1_CH2_BIT_MSB:
+       case CS42L42_ASP_RX_DAI1_CH2_BIT_LSB:
+       case CS42L42_SUB_REVID:
+               return true;
+       default:
+               return false;
+       }
+}
+
+static bool cs42l42_volatile_register(struct device *dev, unsigned int reg)
+{
+       switch (reg) {
+       case CS42L42_DEVID_AB:
+       case CS42L42_DEVID_CD:
+       case CS42L42_DEVID_E:
+       case CS42L42_MCLK_STATUS:
+       case CS42L42_TRSENSE_STATUS:
+       case CS42L42_HS_DET_STATUS:
+       case CS42L42_ADC_OVFL_STATUS:
+       case CS42L42_MIXER_STATUS:
+       case CS42L42_SRC_STATUS:
+       case CS42L42_ASP_RX_STATUS:
+       case CS42L42_ASP_TX_STATUS:
+       case CS42L42_CODEC_STATUS:
+       case CS42L42_DET_INT_STATUS1:
+       case CS42L42_DET_INT_STATUS2:
+       case CS42L42_SRCPL_INT_STATUS:
+       case CS42L42_VPMON_STATUS:
+       case CS42L42_PLL_LOCK_STATUS:
+       case CS42L42_TSRS_PLUG_STATUS:
+       case CS42L42_LOAD_DET_RCSTAT:
+       case CS42L42_LOAD_DET_DONE:
+       case CS42L42_DET_STATUS1:
+       case CS42L42_DET_STATUS2:
+               return true;
+       default:
+               return false;
+       }
+}
+
+static const struct regmap_range_cfg cs42l42_page_range = {
+       .name = "Pages",
+       .range_min = 0,
+       .range_max = CS42L42_MAX_REGISTER,
+       .selector_reg = CS42L42_PAGE_REGISTER,
+       .selector_mask = 0xff,
+       .selector_shift = 0,
+       .window_start = 0,
+       .window_len = 256,
+};
+
+static const struct regmap_config cs42l42_regmap = {
+       .reg_bits = 8,
+       .val_bits = 8,
+
+       .readable_reg = cs42l42_readable_register,
+       .volatile_reg = cs42l42_volatile_register,
+
+       .ranges = &cs42l42_page_range,
+       .num_ranges = 1,
+
+       .max_register = CS42L42_MAX_REGISTER,
+       .reg_defaults = cs42l42_reg_defaults,
+       .num_reg_defaults = ARRAY_SIZE(cs42l42_reg_defaults),
+       .cache_type = REGCACHE_RBTREE,
+};
+
+static DECLARE_TLV_DB_SCALE(adc_tlv, -9600, 100, false);
+static DECLARE_TLV_DB_SCALE(mixer_tlv, -6200, 100, false);
+
+static const char * const cs42l42_hpf_freq_text[] = {
+       "1.86Hz", "120Hz", "235Hz", "466Hz"
+};
+
+static SOC_ENUM_SINGLE_DECL(cs42l42_hpf_freq_enum, CS42L42_ADC_WNF_HPF_CTL,
+                           CS42L42_ADC_HPF_CF_SHIFT,
+                           cs42l42_hpf_freq_text);
+
+static const char * const cs42l42_wnf3_freq_text[] = {
+       "160Hz", "180Hz", "200Hz", "220Hz",
+       "240Hz", "260Hz", "280Hz", "300Hz"
+};
+
+static SOC_ENUM_SINGLE_DECL(cs42l42_wnf3_freq_enum, CS42L42_ADC_WNF_HPF_CTL,
+                           CS42L42_ADC_WNF_CF_SHIFT,
+                           cs42l42_wnf3_freq_text);
+
+static const char * const cs42l42_wnf05_freq_text[] = {
+       "280Hz", "315Hz", "350Hz", "385Hz",
+       "420Hz", "455Hz", "490Hz", "525Hz"
+};
+
+static SOC_ENUM_SINGLE_DECL(cs42l42_wnf05_freq_enum, CS42L42_ADC_WNF_HPF_CTL,
+                           CS42L42_ADC_WNF_CF_SHIFT,
+                           cs42l42_wnf05_freq_text);
+
+static const struct snd_kcontrol_new cs42l42_snd_controls[] = {
+       /* ADC Volume and Filter Controls */
+       SOC_SINGLE("ADC Notch Switch", CS42L42_ADC_CTL,
+                               CS42L42_ADC_NOTCH_DIS_SHIFT, true, false),
+       SOC_SINGLE("ADC Weak Force Switch", CS42L42_ADC_CTL,
+                               CS42L42_ADC_FORCE_WEAK_VCM_SHIFT, true, false),
+       SOC_SINGLE("ADC Invert Switch", CS42L42_ADC_CTL,
+                               CS42L42_ADC_INV_SHIFT, true, false),
+       SOC_SINGLE("ADC Boost Switch", CS42L42_ADC_CTL,
+                               CS42L42_ADC_DIG_BOOST_SHIFT, true, false),
+       SOC_SINGLE_SX_TLV("ADC Volume", CS42L42_ADC_VOLUME,
+                               CS42L42_ADC_VOL_SHIFT, 0xA0, 0x6C, adc_tlv),
+       SOC_SINGLE("ADC WNF Switch", CS42L42_ADC_WNF_HPF_CTL,
+                               CS42L42_ADC_WNF_EN_SHIFT, true, false),
+       SOC_SINGLE("ADC HPF Switch", CS42L42_ADC_WNF_HPF_CTL,
+                               CS42L42_ADC_HPF_EN_SHIFT, true, false),
+       SOC_ENUM("HPF Corner Freq", cs42l42_hpf_freq_enum),
+       SOC_ENUM("WNF 3dB Freq", cs42l42_wnf3_freq_enum),
+       SOC_ENUM("WNF 05dB Freq", cs42l42_wnf05_freq_enum),
+
+       /* DAC Volume and Filter Controls */
+       SOC_SINGLE("DACA Invert Switch", CS42L42_DAC_CTL1,
+                               CS42L42_DACA_INV_SHIFT, true, false),
+       SOC_SINGLE("DACB Invert Switch", CS42L42_DAC_CTL1,
+                               CS42L42_DACB_INV_SHIFT, true, false),
+       SOC_SINGLE("DAC HPF Switch", CS42L42_DAC_CTL2,
+                               CS42L42_DAC_HPF_EN_SHIFT, true, false),
+       SOC_DOUBLE_R_TLV("Mixer Volume", CS42L42_MIXER_CHA_VOL,
+                        CS42L42_MIXER_CHB_VOL, CS42L42_MIXER_CH_VOL_SHIFT,
+                               0x3e, 1, mixer_tlv)
+};
+
+static int cs42l42_hpdrv_evt(struct snd_soc_dapm_widget *w,
+                               struct snd_kcontrol *kcontrol, int event)
+{
+       struct snd_soc_codec *codec = snd_soc_dapm_to_codec(w->dapm);
+
+       if (event & SND_SOC_DAPM_POST_PMU) {
+               /* Enable the channels */
+               snd_soc_update_bits(codec, CS42L42_ASP_RX_DAI0_EN,
+                               CS42L42_ASP_RX0_CH_EN_MASK,
+                               (CS42L42_ASP_RX0_CH1_EN |
+                               CS42L42_ASP_RX0_CH2_EN) <<
+                               CS42L42_ASP_RX0_CH_EN_SHIFT);
+
+               /* Power up */
+               snd_soc_update_bits(codec, CS42L42_PWR_CTL1,
+                       CS42L42_ASP_DAI_PDN_MASK | CS42L42_MIXER_PDN_MASK |
+                               CS42L42_HP_PDN_MASK, 0);
+       } else if (event & SND_SOC_DAPM_PRE_PMD) {
+               /* Disable the channels */
+               snd_soc_update_bits(codec, CS42L42_ASP_RX_DAI0_EN,
+                               CS42L42_ASP_RX0_CH_EN_MASK, 0);
+
+               /* Power down */
+               snd_soc_update_bits(codec, CS42L42_PWR_CTL1,
+                       CS42L42_ASP_DAI_PDN_MASK | CS42L42_MIXER_PDN_MASK |
+                               CS42L42_HP_PDN_MASK,
+                       CS42L42_ASP_DAI_PDN_MASK | CS42L42_MIXER_PDN_MASK |
+                               CS42L42_HP_PDN_MASK);
+       } else {
+               dev_err(codec->dev, "Invalid event 0x%x\n", event);
+       }
+       return 0;
+}
+
+static const struct snd_soc_dapm_widget cs42l42_dapm_widgets[] = {
+       SND_SOC_DAPM_OUTPUT("HP"),
+       SND_SOC_DAPM_AIF_IN("SDIN", NULL, 0, CS42L42_ASP_CLK_CFG,
+                                       CS42L42_ASP_SCLK_EN_SHIFT, false),
+       SND_SOC_DAPM_OUT_DRV_E("HPDRV", SND_SOC_NOPM, 0,
+                                       0, NULL, 0, cs42l42_hpdrv_evt,
+                                       SND_SOC_DAPM_POST_PMU |
+                                       SND_SOC_DAPM_PRE_PMD)
+};
+
+static const struct snd_soc_dapm_route cs42l42_audio_map[] = {
+       {"SDIN", NULL, "Playback"},
+       {"HPDRV", NULL, "SDIN"},
+       {"HP", NULL, "HPDRV"}
+};
+
+static int cs42l42_set_bias_level(struct snd_soc_codec *codec,
+                                       enum snd_soc_bias_level level)
+{
+       struct cs42l42_private *cs42l42 = snd_soc_codec_get_drvdata(codec);
+       int ret;
+
+       switch (level) {
+       case SND_SOC_BIAS_ON:
+               break;
+       case SND_SOC_BIAS_PREPARE:
+               break;
+       case SND_SOC_BIAS_STANDBY:
+               if (snd_soc_codec_get_bias_level(codec) == SND_SOC_BIAS_OFF) {
+                       regcache_cache_only(cs42l42->regmap, false);
+                       regcache_sync(cs42l42->regmap);
+                       ret = regulator_bulk_enable(
+                                               ARRAY_SIZE(cs42l42->supplies),
+                                               cs42l42->supplies);
+                       if (ret != 0) {
+                               dev_err(codec->dev,
+                                       "Failed to enable regulators: %d\n",
+                                       ret);
+                               return ret;
+                       }
+               }
+               break;
+       case SND_SOC_BIAS_OFF:
+
+               regcache_cache_only(cs42l42->regmap, true);
+               regulator_bulk_disable(ARRAY_SIZE(cs42l42->supplies),
+                                                   cs42l42->supplies);
+               break;
+       }
+
+       return 0;
+}
+
+static int cs42l42_codec_probe(struct snd_soc_codec *codec)
+{
+       struct cs42l42_private *cs42l42 =
+               (struct cs42l42_private *)snd_soc_codec_get_drvdata(codec);
+
+       cs42l42->codec = codec;
+
+       return 0;
+}
+
+static const struct snd_soc_codec_driver soc_codec_dev_cs42l42 = {
+       .probe = cs42l42_codec_probe,
+       .set_bias_level = cs42l42_set_bias_level,
+       .ignore_pmdown_time = true,
+
+       .component_driver = {
+               .dapm_widgets = cs42l42_dapm_widgets,
+               .num_dapm_widgets = ARRAY_SIZE(cs42l42_dapm_widgets),
+               .dapm_routes = cs42l42_audio_map,
+               .num_dapm_routes = ARRAY_SIZE(cs42l42_audio_map),
+
+               .controls = cs42l42_snd_controls,
+               .num_controls = ARRAY_SIZE(cs42l42_snd_controls),
+       },
+};
+
+struct cs42l42_pll_params {
+       u32 sclk;
+       u8 mclk_div;
+       u8 mclk_src_sel;
+       u8 sclk_prediv;
+       u8 pll_div_int;
+       u32 pll_div_frac;
+       u8 pll_mode;
+       u8 pll_divout;
+       u32 mclk_int;
+       u8 pll_cal_ratio;
+};
+
+/*
+ * Common PLL Settings for given SCLK
+ * Table 4-5 from the Datasheet
+ */
+static const struct cs42l42_pll_params pll_ratio_table[] = {
+       { 1536000, 0, 1, 0x00, 0x7D, 0x000000, 0x03, 0x10, 12000000, 125 },
+       { 2822400, 0, 1, 0x00, 0x40, 0x000000, 0x03, 0x10, 11289600, 128 },
+       { 3000000, 0, 1, 0x00, 0x40, 0x000000, 0x03, 0x10, 12000000, 128 },
+       { 3072000, 0, 1, 0x00, 0x3E, 0x800000, 0x03, 0x10, 12000000, 125 },
+       { 4000000, 0, 1, 0x00, 0x30, 0x800000, 0x03, 0x10, 12000000, 96 },
+       { 4096000, 0, 1, 0x00, 0x2E, 0xE00000, 0x03, 0x10, 12000000, 94 },
+       { 5644800, 0, 1, 0x01, 0x40, 0x000000, 0x03, 0x10, 11289600, 128 },
+       { 6000000, 0, 1, 0x01, 0x40, 0x000000, 0x03, 0x10, 12000000, 128 },
+       { 6144000, 0, 1, 0x01, 0x3E, 0x800000, 0x03, 0x10, 12000000, 125 },
+       { 11289600, 0, 0, 0, 0, 0, 0, 0, 11289600, 0 },
+       { 12000000, 0, 0, 0, 0, 0, 0, 0, 12000000, 0 },
+       { 12288000, 0, 0, 0, 0, 0, 0, 0, 12288000, 0 },
+       { 22579200, 1, 0, 0, 0, 0, 0, 0, 22579200, 0 },
+       { 24000000, 1, 0, 0, 0, 0, 0, 0, 24000000, 0 },
+       { 24576000, 1, 0, 0, 0, 0, 0, 0, 24576000, 0 }
+};
+
+static int cs42l42_pll_config(struct snd_soc_codec *codec)
+{
+       struct cs42l42_private *cs42l42 = snd_soc_codec_get_drvdata(codec);
+       int i;
+       u32 fsync;
+
+       for (i = 0; i < ARRAY_SIZE(pll_ratio_table); i++) {
+               if (pll_ratio_table[i].sclk == cs42l42->sclk) {
+                       /* Configure the internal sample rate */
+                       snd_soc_update_bits(codec, CS42L42_MCLK_CTL,
+                                       CS42L42_INTERNAL_FS_MASK,
+                                       ((pll_ratio_table[i].mclk_int !=
+                                       12000000) &&
+                                       (pll_ratio_table[i].mclk_int !=
+                                       24000000)) <<
+                                       CS42L42_INTERNAL_FS_SHIFT);
+                       /* Set the MCLK src (PLL or SCLK) and the divide
+                        * ratio
+                        */
+                       snd_soc_update_bits(codec, CS42L42_MCLK_SRC_SEL,
+                                       CS42L42_MCLK_SRC_SEL_MASK |
+                                       CS42L42_MCLKDIV_MASK,
+                                       (pll_ratio_table[i].mclk_src_sel
+                                       << CS42L42_MCLK_SRC_SEL_SHIFT) |
+                                       (pll_ratio_table[i].mclk_div <<
+                                       CS42L42_MCLKDIV_SHIFT));
+                       /* Set up the LRCLK */
+                       fsync = cs42l42->sclk / cs42l42->srate;
+                       if (((fsync * cs42l42->srate) != cs42l42->sclk)
+                               || ((fsync % 2) != 0)) {
+                               dev_err(codec->dev,
+                                       "Unsupported sclk %d/sample rate %d\n",
+                                       cs42l42->sclk,
+                                       cs42l42->srate);
+                               return -EINVAL;
+                       }
+                       /* Set the LRCLK period */
+                       snd_soc_update_bits(codec,
+                                       CS42L42_FSYNC_P_LOWER,
+                                       CS42L42_FSYNC_PERIOD_MASK,
+                                       CS42L42_FRAC0_VAL(fsync - 1) <<
+                                       CS42L42_FSYNC_PERIOD_SHIFT);
+                       snd_soc_update_bits(codec,
+                                       CS42L42_FSYNC_P_UPPER,
+                                       CS42L42_FSYNC_PERIOD_MASK,
+                                       CS42L42_FRAC1_VAL(fsync - 1) <<
+                                       CS42L42_FSYNC_PERIOD_SHIFT);
+                       /* Set the LRCLK to 50% duty cycle */
+                       fsync = fsync / 2;
+                       snd_soc_update_bits(codec,
+                                       CS42L42_FSYNC_PW_LOWER,
+                                       CS42L42_FSYNC_PULSE_WIDTH_MASK,
+                                       CS42L42_FRAC0_VAL(fsync - 1) <<
+                                       CS42L42_FSYNC_PULSE_WIDTH_SHIFT);
+                       snd_soc_update_bits(codec,
+                                       CS42L42_FSYNC_PW_UPPER,
+                                       CS42L42_FSYNC_PULSE_WIDTH_MASK,
+                                       CS42L42_FRAC1_VAL(fsync - 1) <<
+                                       CS42L42_FSYNC_PULSE_WIDTH_SHIFT);
+                       snd_soc_update_bits(codec,
+                                       CS42L42_ASP_FRM_CFG,
+                                       CS42L42_ASP_5050_MASK,
+                                       CS42L42_ASP_5050_MASK);
+                       /* Set the frame delay to 1.0 SCLK clocks */
+                       snd_soc_update_bits(codec, CS42L42_ASP_FRM_CFG,
+                                       CS42L42_ASP_FSD_MASK,
+                                       CS42L42_ASP_FSD_1_0 <<
+                                       CS42L42_ASP_FSD_SHIFT);
+                       /* Set the sample rates (96k or lower) */
+                       snd_soc_update_bits(codec, CS42L42_FS_RATE_EN,
+                                       CS42L42_FS_EN_MASK,
+                                       (CS42L42_FS_EN_IASRC_96K |
+                                       CS42L42_FS_EN_OASRC_96K) <<
+                                       CS42L42_FS_EN_SHIFT);
+                       /* Set the input/output internal MCLK clock ~12 MHz */
+                       snd_soc_update_bits(codec, CS42L42_IN_ASRC_CLK,
+                                       CS42L42_CLK_IASRC_SEL_MASK,
+                                       CS42L42_CLK_IASRC_SEL_12 <<
+                                       CS42L42_CLK_IASRC_SEL_SHIFT);
+                       snd_soc_update_bits(codec,
+                                       CS42L42_OUT_ASRC_CLK,
+                                       CS42L42_CLK_OASRC_SEL_MASK,
+                                       CS42L42_CLK_OASRC_SEL_12 <<
+                                       CS42L42_CLK_OASRC_SEL_SHIFT);
+                       /* channel 1 on low LRCLK, 32 bit */
+                       snd_soc_update_bits(codec,
+                                       CS42L42_ASP_RX_DAI0_CH1_AP_RES,
+                                       CS42L42_ASP_RX_CH_AP_MASK |
+                                       CS42L42_ASP_RX_CH_RES_MASK,
+                                       (CS42L42_ASP_RX_CH_AP_LOW <<
+                                       CS42L42_ASP_RX_CH_AP_SHIFT) |
+                                       (CS42L42_ASP_RX_CH_RES_32 <<
+                                       CS42L42_ASP_RX_CH_RES_SHIFT));
+                       /* Channel 2 on high LRCLK, 32 bit */
+                       snd_soc_update_bits(codec,
+                                       CS42L42_ASP_RX_DAI0_CH2_AP_RES,
+                                       CS42L42_ASP_RX_CH_AP_MASK |
+                                       CS42L42_ASP_RX_CH_RES_MASK,
+                                       (CS42L42_ASP_RX_CH_AP_HI <<
+                                       CS42L42_ASP_RX_CH_AP_SHIFT) |
+                                       (CS42L42_ASP_RX_CH_RES_32 <<
+                                       CS42L42_ASP_RX_CH_RES_SHIFT));
+                       if (pll_ratio_table[i].mclk_src_sel == 0) {
+                               /* Pass the clock straight through */
+                               snd_soc_update_bits(codec,
+                                       CS42L42_PLL_CTL1,
+                                       CS42L42_PLL_START_MASK, 0);
+                       } else {
+                               /* Configure PLL per table 4-5 */
+                               snd_soc_update_bits(codec,
+                                       CS42L42_PLL_DIV_CFG1,
+                                       CS42L42_SCLK_PREDIV_MASK,
+                                       pll_ratio_table[i].sclk_prediv
+                                       << CS42L42_SCLK_PREDIV_SHIFT);
+                               snd_soc_update_bits(codec,
+                                       CS42L42_PLL_DIV_INT,
+                                       CS42L42_PLL_DIV_INT_MASK,
+                                       pll_ratio_table[i].pll_div_int
+                                       << CS42L42_PLL_DIV_INT_SHIFT);
+                               snd_soc_update_bits(codec,
+                                       CS42L42_PLL_DIV_FRAC0,
+                                       CS42L42_PLL_DIV_FRAC_MASK,
+                                       CS42L42_FRAC0_VAL(
+                                       pll_ratio_table[i].pll_div_frac)
+                                       << CS42L42_PLL_DIV_FRAC_SHIFT);
+                               snd_soc_update_bits(codec,
+                                       CS42L42_PLL_DIV_FRAC1,
+                                       CS42L42_PLL_DIV_FRAC_MASK,
+                                       CS42L42_FRAC1_VAL(
+                                       pll_ratio_table[i].pll_div_frac)
+                                       << CS42L42_PLL_DIV_FRAC_SHIFT);
+                               snd_soc_update_bits(codec,
+                                       CS42L42_PLL_DIV_FRAC2,
+                                       CS42L42_PLL_DIV_FRAC_MASK,
+                                       CS42L42_FRAC2_VAL(
+                                       pll_ratio_table[i].pll_div_frac)
+                                       << CS42L42_PLL_DIV_FRAC_SHIFT);
+                               snd_soc_update_bits(codec,
+                                       CS42L42_PLL_CTL4,
+                                       CS42L42_PLL_MODE_MASK,
+                                       pll_ratio_table[i].pll_mode
+                                       << CS42L42_PLL_MODE_SHIFT);
+                               snd_soc_update_bits(codec,
+                                       CS42L42_PLL_CTL3,
+                                       CS42L42_PLL_DIVOUT_MASK,
+                                       pll_ratio_table[i].pll_divout
+                                       << CS42L42_PLL_DIVOUT_SHIFT);
+                               snd_soc_update_bits(codec,
+                                       CS42L42_PLL_CAL_RATIO,
+                                       CS42L42_PLL_CAL_RATIO_MASK,
+                                       pll_ratio_table[i].pll_cal_ratio
+                                       << CS42L42_PLL_CAL_RATIO_SHIFT);
+                       }
+                       return 0;
+               }
+       }
+
+       return -EINVAL;
+}
+
+static int cs42l42_set_dai_fmt(struct snd_soc_dai *codec_dai, unsigned int fmt)
+{
+       struct snd_soc_codec *codec = codec_dai->codec;
+       u32 asp_cfg_val = 0;
+
+       switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
+       case SND_SOC_DAIFMT_CBS_CFM:
+               asp_cfg_val |= CS42L42_ASP_MASTER_MODE <<
+                               CS42L42_ASP_MODE_SHIFT;
+               break;
+       case SND_SOC_DAIFMT_CBS_CFS:
+               asp_cfg_val |= CS42L42_ASP_SLAVE_MODE <<
+                               CS42L42_ASP_MODE_SHIFT;
+               break;
+       default:
+               return -EINVAL;
+       }
+
+       /* interface format */
+       switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
+       case SND_SOC_DAIFMT_I2S:
+       case SND_SOC_DAIFMT_LEFT_J:
+               break;
+       default:
+               return -EINVAL;
+       }
+
+       /* Bitclock/frame inversion */
+       switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
+       case SND_SOC_DAIFMT_NB_NF:
+               break;
+       case SND_SOC_DAIFMT_NB_IF:
+               asp_cfg_val |= CS42L42_ASP_POL_INV <<
+                               CS42L42_ASP_LCPOL_IN_SHIFT;
+               break;
+       case SND_SOC_DAIFMT_IB_NF:
+               asp_cfg_val |= CS42L42_ASP_POL_INV <<
+                               CS42L42_ASP_SCPOL_IN_DAC_SHIFT;
+               break;
+       case SND_SOC_DAIFMT_IB_IF:
+               asp_cfg_val |= CS42L42_ASP_POL_INV <<
+                               CS42L42_ASP_LCPOL_IN_SHIFT;
+               asp_cfg_val |= CS42L42_ASP_POL_INV <<
+                               CS42L42_ASP_SCPOL_IN_DAC_SHIFT;
+               break;
+       }
+
+       snd_soc_update_bits(codec, CS42L42_ASP_CLK_CFG,
+                               CS42L42_ASP_MODE_MASK |
+                               CS42L42_ASP_SCPOL_IN_DAC_MASK |
+                               CS42L42_ASP_LCPOL_IN_MASK, asp_cfg_val);
+
+       return 0;
+}
+
+static int cs42l42_pcm_hw_params(struct snd_pcm_substream *substream,
+                               struct snd_pcm_hw_params *params,
+                               struct snd_soc_dai *dai)
+{
+       struct snd_soc_codec *codec = dai->codec;
+       struct cs42l42_private *cs42l42 = snd_soc_codec_get_drvdata(codec);
+       int retval;
+
+       cs42l42->srate = params_rate(params);
+       cs42l42->swidth = params_width(params);
+
+       retval = cs42l42_pll_config(codec);
+
+       return retval;
+}
+
+static int cs42l42_set_sysclk(struct snd_soc_dai *dai,
+                               int clk_id, unsigned int freq, int dir)
+{
+       struct snd_soc_codec *codec = dai->codec;
+       struct cs42l42_private *cs42l42 = snd_soc_codec_get_drvdata(codec);
+
+       cs42l42->sclk = freq;
+
+       return 0;
+}
+
+static int cs42l42_digital_mute(struct snd_soc_dai *dai, int mute)
+{
+       struct snd_soc_codec *codec = dai->codec;
+       unsigned int regval;
+       u8 fullScaleVol;
+
+       if (mute) {
+               /* Mark SCLK as not present to turn on the internal
+                * oscillator.
+                */
+               snd_soc_update_bits(codec, CS42L42_OSC_SWITCH,
+                                               CS42L42_SCLK_PRESENT_MASK, 0);
+
+               snd_soc_update_bits(codec, CS42L42_PLL_CTL1,
+                               CS42L42_PLL_START_MASK,
+                               0 << CS42L42_PLL_START_SHIFT);
+
+               /* Mute the headphone */
+               snd_soc_update_bits(codec, CS42L42_HP_CTL,
+                               CS42L42_HP_ANA_AMUTE_MASK |
+                               CS42L42_HP_ANA_BMUTE_MASK,
+                               CS42L42_HP_ANA_AMUTE_MASK |
+                               CS42L42_HP_ANA_BMUTE_MASK);
+       } else {
+               snd_soc_update_bits(codec, CS42L42_PLL_CTL1,
+                               CS42L42_PLL_START_MASK,
+                               1 << CS42L42_PLL_START_SHIFT);
+               /* Read the headphone load */
+               regval = snd_soc_read(codec, CS42L42_LOAD_DET_RCSTAT);
+               if (((regval & CS42L42_RLA_STAT_MASK) >>
+                       CS42L42_RLA_STAT_SHIFT) == CS42L42_RLA_STAT_15_OHM) {
+                       fullScaleVol = CS42L42_HP_FULL_SCALE_VOL_MASK;
+               } else {
+                       fullScaleVol = 0;
+               }
+
+               /* Un-mute the headphone, set the full scale volume flag */
+               snd_soc_update_bits(codec, CS42L42_HP_CTL,
+                               CS42L42_HP_ANA_AMUTE_MASK |
+                               CS42L42_HP_ANA_BMUTE_MASK |
+                               CS42L42_HP_FULL_SCALE_VOL_MASK, fullScaleVol);
+
+               /* Mark SCLK as present, turn off internal oscillator */
+               snd_soc_update_bits(codec, CS42L42_OSC_SWITCH,
+                               CS42L42_SCLK_PRESENT_MASK,
+                               CS42L42_SCLK_PRESENT_MASK);
+       }
+
+       return 0;
+}
+
+#define CS42L42_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S18_3LE | \
+                       SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S24_LE | \
+                       SNDRV_PCM_FMTBIT_S32_LE)
+
+
+static struct snd_soc_dai_ops cs42l42_ops = {
+       .hw_params      = cs42l42_pcm_hw_params,
+       .set_fmt        = cs42l42_set_dai_fmt,
+       .set_sysclk     = cs42l42_set_sysclk,
+       .digital_mute = cs42l42_digital_mute
+};
+
+static struct snd_soc_dai_driver cs42l42_dai = {
+               .name = "cs42l42",
+               .playback = {
+                       .stream_name = "Playback",
+                       .channels_min = 1,
+                       .channels_max = 2,
+                       .rates = SNDRV_PCM_RATE_8000_192000,
+                       .formats = CS42L42_FORMATS,
+               },
+               .capture = {
+                       .stream_name = "Capture",
+                       .channels_min = 1,
+                       .channels_max = 2,
+                       .rates = SNDRV_PCM_RATE_8000_192000,
+                       .formats = CS42L42_FORMATS,
+               },
+               .ops = &cs42l42_ops,
+};
+
+static void cs42l42_process_hs_type_detect(struct cs42l42_private *cs42l42)
+{
+       unsigned int hs_det_status;
+       unsigned int int_status;
+
+       /* Mask the auto detect interrupt */
+       regmap_update_bits(cs42l42->regmap,
+               CS42L42_CODEC_INT_MASK,
+               CS42L42_PDN_DONE_MASK |
+               CS42L42_HSDET_AUTO_DONE_MASK,
+               (1 << CS42L42_PDN_DONE_SHIFT) |
+               (1 << CS42L42_HSDET_AUTO_DONE_SHIFT));
+
+       /* Set hs detect to automatic, disabled mode */
+       regmap_update_bits(cs42l42->regmap,
+               CS42L42_HSDET_CTL2,
+               CS42L42_HSDET_CTRL_MASK |
+               CS42L42_HSDET_SET_MASK |
+               CS42L42_HSBIAS_REF_MASK |
+               CS42L42_HSDET_AUTO_TIME_MASK,
+               (2 << CS42L42_HSDET_CTRL_SHIFT) |
+               (2 << CS42L42_HSDET_SET_SHIFT) |
+               (0 << CS42L42_HSBIAS_REF_SHIFT) |
+               (3 << CS42L42_HSDET_AUTO_TIME_SHIFT));
+
+       /* Read and save the hs detection result */
+       regmap_read(cs42l42->regmap, CS42L42_HS_DET_STATUS, &hs_det_status);
+
+       cs42l42->hs_type = (hs_det_status & CS42L42_HSDET_TYPE_MASK) >>
+                               CS42L42_HSDET_TYPE_SHIFT;
+
+       /* Set up button detection */
+       if ((cs42l42->hs_type == CS42L42_PLUG_CTIA) ||
+             (cs42l42->hs_type == CS42L42_PLUG_OMTP)) {
+               /* Set auto HS bias settings to default */
+               regmap_update_bits(cs42l42->regmap,
+                       CS42L42_HSBIAS_SC_AUTOCTL,
+                       CS42L42_HSBIAS_SENSE_EN_MASK |
+                       CS42L42_AUTO_HSBIAS_HIZ_MASK |
+                       CS42L42_TIP_SENSE_EN_MASK |
+                       CS42L42_HSBIAS_SENSE_TRIP_MASK,
+                       (0 << CS42L42_HSBIAS_SENSE_EN_SHIFT) |
+                       (0 << CS42L42_AUTO_HSBIAS_HIZ_SHIFT) |
+                       (0 << CS42L42_TIP_SENSE_EN_SHIFT) |
+                       (3 << CS42L42_HSBIAS_SENSE_TRIP_SHIFT));
+
+               /* Set up hs detect level sensitivity */
+               regmap_update_bits(cs42l42->regmap,
+                       CS42L42_MIC_DET_CTL1,
+                       CS42L42_LATCH_TO_VP_MASK |
+                       CS42L42_EVENT_STAT_SEL_MASK |
+                       CS42L42_HS_DET_LEVEL_MASK,
+                       (1 << CS42L42_LATCH_TO_VP_SHIFT) |
+                       (0 << CS42L42_EVENT_STAT_SEL_SHIFT) |
+                       (cs42l42->bias_thresholds[0] <<
+                       CS42L42_HS_DET_LEVEL_SHIFT));
+
+               /* Set auto HS bias settings to default */
+               regmap_update_bits(cs42l42->regmap,
+                       CS42L42_HSBIAS_SC_AUTOCTL,
+                       CS42L42_HSBIAS_SENSE_EN_MASK |
+                       CS42L42_AUTO_HSBIAS_HIZ_MASK |
+                       CS42L42_TIP_SENSE_EN_MASK |
+                       CS42L42_HSBIAS_SENSE_TRIP_MASK,
+                       (1 << CS42L42_HSBIAS_SENSE_EN_SHIFT) |
+                       (1 << CS42L42_AUTO_HSBIAS_HIZ_SHIFT) |
+                       (0 << CS42L42_TIP_SENSE_EN_SHIFT) |
+                       (3 << CS42L42_HSBIAS_SENSE_TRIP_SHIFT));
+
+               /* Turn on level detect circuitry */
+               regmap_update_bits(cs42l42->regmap,
+                       CS42L42_MISC_DET_CTL,
+                       CS42L42_DETECT_MODE_MASK |
+                       CS42L42_HSBIAS_CTL_MASK |
+                       CS42L42_PDN_MIC_LVL_DET_MASK,
+                       (0 << CS42L42_DETECT_MODE_SHIFT) |
+                       (3 << CS42L42_HSBIAS_CTL_SHIFT) |
+                       (0 << CS42L42_PDN_MIC_LVL_DET_SHIFT));
+
+               msleep(cs42l42->btn_det_init_dbnce);
+
+               /* Clear any button interrupts before unmasking them */
+               regmap_read(cs42l42->regmap, CS42L42_DET_INT_STATUS2,
+                           &int_status);
+
+               /* Unmask button detect interrupts */
+               regmap_update_bits(cs42l42->regmap,
+                       CS42L42_DET_INT2_MASK,
+                       CS42L42_M_DETECT_TF_MASK |
+                       CS42L42_M_DETECT_FT_MASK |
+                       CS42L42_M_HSBIAS_HIZ_MASK |
+                       CS42L42_M_SHORT_RLS_MASK |
+                       CS42L42_M_SHORT_DET_MASK,
+                       (0 << CS42L42_M_DETECT_TF_SHIFT) |
+                       (0 << CS42L42_M_DETECT_FT_SHIFT) |
+                       (0 << CS42L42_M_HSBIAS_HIZ_SHIFT) |
+                       (1 << CS42L42_M_SHORT_RLS_SHIFT) |
+                       (1 << CS42L42_M_SHORT_DET_SHIFT));
+       } else {
+               /* Make sure button detect and HS bias circuits are off */
+               regmap_update_bits(cs42l42->regmap,
+                       CS42L42_MISC_DET_CTL,
+                       CS42L42_DETECT_MODE_MASK |
+                       CS42L42_HSBIAS_CTL_MASK |
+                       CS42L42_PDN_MIC_LVL_DET_MASK,
+                       (0 << CS42L42_DETECT_MODE_SHIFT) |
+                       (1 << CS42L42_HSBIAS_CTL_SHIFT) |
+                       (1 << CS42L42_PDN_MIC_LVL_DET_SHIFT));
+       }
+
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_DAC_CTL2,
+                               CS42L42_HPOUT_PULLDOWN_MASK |
+                               CS42L42_HPOUT_LOAD_MASK |
+                               CS42L42_HPOUT_CLAMP_MASK |
+                               CS42L42_DAC_HPF_EN_MASK |
+                               CS42L42_DAC_MON_EN_MASK,
+                               (0 << CS42L42_HPOUT_PULLDOWN_SHIFT) |
+                               (0 << CS42L42_HPOUT_LOAD_SHIFT) |
+                               (0 << CS42L42_HPOUT_CLAMP_SHIFT) |
+                               (1 << CS42L42_DAC_HPF_EN_SHIFT) |
+                               (0 << CS42L42_DAC_MON_EN_SHIFT));
+
+       /* Unmask tip sense interrupts */
+       regmap_update_bits(cs42l42->regmap,
+               CS42L42_TSRS_PLUG_INT_MASK,
+               CS42L42_RS_PLUG_MASK |
+               CS42L42_RS_UNPLUG_MASK |
+               CS42L42_TS_PLUG_MASK |
+               CS42L42_TS_UNPLUG_MASK,
+               (1 << CS42L42_RS_PLUG_SHIFT) |
+               (1 << CS42L42_RS_UNPLUG_SHIFT) |
+               (0 << CS42L42_TS_PLUG_SHIFT) |
+               (0 << CS42L42_TS_UNPLUG_SHIFT));
+}
+
+static void cs42l42_init_hs_type_detect(struct cs42l42_private *cs42l42)
+{
+       /* Mask tip sense interrupts */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_TSRS_PLUG_INT_MASK,
+                               CS42L42_RS_PLUG_MASK |
+                               CS42L42_RS_UNPLUG_MASK |
+                               CS42L42_TS_PLUG_MASK |
+                               CS42L42_TS_UNPLUG_MASK,
+                               (1 << CS42L42_RS_PLUG_SHIFT) |
+                               (1 << CS42L42_RS_UNPLUG_SHIFT) |
+                               (1 << CS42L42_TS_PLUG_SHIFT) |
+                               (1 << CS42L42_TS_UNPLUG_SHIFT));
+
+       /* Make sure button detect and HS bias circuits are off */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_MISC_DET_CTL,
+                               CS42L42_DETECT_MODE_MASK |
+                               CS42L42_HSBIAS_CTL_MASK |
+                               CS42L42_PDN_MIC_LVL_DET_MASK,
+                               (0 << CS42L42_DETECT_MODE_SHIFT) |
+                               (1 << CS42L42_HSBIAS_CTL_SHIFT) |
+                               (1 << CS42L42_PDN_MIC_LVL_DET_SHIFT));
+
+       /* Set auto HS bias settings to default */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_HSBIAS_SC_AUTOCTL,
+                               CS42L42_HSBIAS_SENSE_EN_MASK |
+                               CS42L42_AUTO_HSBIAS_HIZ_MASK |
+                               CS42L42_TIP_SENSE_EN_MASK |
+                               CS42L42_HSBIAS_SENSE_TRIP_MASK,
+                               (0 << CS42L42_HSBIAS_SENSE_EN_SHIFT) |
+                               (0 << CS42L42_AUTO_HSBIAS_HIZ_SHIFT) |
+                               (0 << CS42L42_TIP_SENSE_EN_SHIFT) |
+                               (3 << CS42L42_HSBIAS_SENSE_TRIP_SHIFT));
+
+       /* Set hs detect to manual, disabled mode */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_HSDET_CTL2,
+                               CS42L42_HSDET_CTRL_MASK |
+                               CS42L42_HSDET_SET_MASK |
+                               CS42L42_HSBIAS_REF_MASK |
+                               CS42L42_HSDET_AUTO_TIME_MASK,
+                               (0 << CS42L42_HSDET_CTRL_SHIFT) |
+                               (2 << CS42L42_HSDET_SET_SHIFT) |
+                               (0 << CS42L42_HSBIAS_REF_SHIFT) |
+                               (3 << CS42L42_HSDET_AUTO_TIME_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_DAC_CTL2,
+                               CS42L42_HPOUT_PULLDOWN_MASK |
+                               CS42L42_HPOUT_LOAD_MASK |
+                               CS42L42_HPOUT_CLAMP_MASK |
+                               CS42L42_DAC_HPF_EN_MASK |
+                               CS42L42_DAC_MON_EN_MASK,
+                               (8 << CS42L42_HPOUT_PULLDOWN_SHIFT) |
+                               (0 << CS42L42_HPOUT_LOAD_SHIFT) |
+                               (1 << CS42L42_HPOUT_CLAMP_SHIFT) |
+                               (1 << CS42L42_DAC_HPF_EN_SHIFT) |
+                               (1 << CS42L42_DAC_MON_EN_SHIFT));
+
+       /* Power up HS bias to 2.7V */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_MISC_DET_CTL,
+                               CS42L42_DETECT_MODE_MASK |
+                               CS42L42_HSBIAS_CTL_MASK |
+                               CS42L42_PDN_MIC_LVL_DET_MASK,
+                               (0 << CS42L42_DETECT_MODE_SHIFT) |
+                               (3 << CS42L42_HSBIAS_CTL_SHIFT) |
+                               (1 << CS42L42_PDN_MIC_LVL_DET_SHIFT));
+
+       /* Wait for HS bias to ramp up */
+       msleep(cs42l42->hs_bias_ramp_time);
+
+       /* Unmask auto detect interrupt */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_CODEC_INT_MASK,
+                               CS42L42_PDN_DONE_MASK |
+                               CS42L42_HSDET_AUTO_DONE_MASK,
+                               (1 << CS42L42_PDN_DONE_SHIFT) |
+                               (0 << CS42L42_HSDET_AUTO_DONE_SHIFT));
+
+       /* Set hs detect to automatic, enabled mode */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_HSDET_CTL2,
+                               CS42L42_HSDET_CTRL_MASK |
+                               CS42L42_HSDET_SET_MASK |
+                               CS42L42_HSBIAS_REF_MASK |
+                               CS42L42_HSDET_AUTO_TIME_MASK,
+                               (3 << CS42L42_HSDET_CTRL_SHIFT) |
+                               (2 << CS42L42_HSDET_SET_SHIFT) |
+                               (0 << CS42L42_HSBIAS_REF_SHIFT) |
+                               (3 << CS42L42_HSDET_AUTO_TIME_SHIFT));
+}
+
+static void cs42l42_cancel_hs_type_detect(struct cs42l42_private *cs42l42)
+{
+       /* Mask button detect interrupts */
+       regmap_update_bits(cs42l42->regmap,
+               CS42L42_DET_INT2_MASK,
+               CS42L42_M_DETECT_TF_MASK |
+               CS42L42_M_DETECT_FT_MASK |
+               CS42L42_M_HSBIAS_HIZ_MASK |
+               CS42L42_M_SHORT_RLS_MASK |
+               CS42L42_M_SHORT_DET_MASK,
+               (1 << CS42L42_M_DETECT_TF_SHIFT) |
+               (1 << CS42L42_M_DETECT_FT_SHIFT) |
+               (1 << CS42L42_M_HSBIAS_HIZ_SHIFT) |
+               (1 << CS42L42_M_SHORT_RLS_SHIFT) |
+               (1 << CS42L42_M_SHORT_DET_SHIFT));
+
+       /* Ground HS bias */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_MISC_DET_CTL,
+                               CS42L42_DETECT_MODE_MASK |
+                               CS42L42_HSBIAS_CTL_MASK |
+                               CS42L42_PDN_MIC_LVL_DET_MASK,
+                               (0 << CS42L42_DETECT_MODE_SHIFT) |
+                               (1 << CS42L42_HSBIAS_CTL_SHIFT) |
+                               (1 << CS42L42_PDN_MIC_LVL_DET_SHIFT));
+
+       /* Set auto HS bias settings to default */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_HSBIAS_SC_AUTOCTL,
+                               CS42L42_HSBIAS_SENSE_EN_MASK |
+                               CS42L42_AUTO_HSBIAS_HIZ_MASK |
+                               CS42L42_TIP_SENSE_EN_MASK |
+                               CS42L42_HSBIAS_SENSE_TRIP_MASK,
+                               (0 << CS42L42_HSBIAS_SENSE_EN_SHIFT) |
+                               (0 << CS42L42_AUTO_HSBIAS_HIZ_SHIFT) |
+                               (0 << CS42L42_TIP_SENSE_EN_SHIFT) |
+                               (3 << CS42L42_HSBIAS_SENSE_TRIP_SHIFT));
+
+       /* Set hs detect to manual, disabled mode */
+       regmap_update_bits(cs42l42->regmap,
+                               CS42L42_HSDET_CTL2,
+                               CS42L42_HSDET_CTRL_MASK |
+                               CS42L42_HSDET_SET_MASK |
+                               CS42L42_HSBIAS_REF_MASK |
+                               CS42L42_HSDET_AUTO_TIME_MASK,
+                               (0 << CS42L42_HSDET_CTRL_SHIFT) |
+                               (2 << CS42L42_HSDET_SET_SHIFT) |
+                               (0 << CS42L42_HSBIAS_REF_SHIFT) |
+                               (3 << CS42L42_HSDET_AUTO_TIME_SHIFT));
+}
+
+static void cs42l42_handle_button_press(struct cs42l42_private *cs42l42)
+{
+       int bias_level;
+       unsigned int detect_status;
+
+       /* Mask button detect interrupts */
+       regmap_update_bits(cs42l42->regmap,
+               CS42L42_DET_INT2_MASK,
+               CS42L42_M_DETECT_TF_MASK |
+               CS42L42_M_DETECT_FT_MASK |
+               CS42L42_M_HSBIAS_HIZ_MASK |
+               CS42L42_M_SHORT_RLS_MASK |
+               CS42L42_M_SHORT_DET_MASK,
+               (1 << CS42L42_M_DETECT_TF_SHIFT) |
+               (1 << CS42L42_M_DETECT_FT_SHIFT) |
+               (1 << CS42L42_M_HSBIAS_HIZ_SHIFT) |
+               (1 << CS42L42_M_SHORT_RLS_SHIFT) |
+               (1 << CS42L42_M_SHORT_DET_SHIFT));
+
+       usleep_range(cs42l42->btn_det_event_dbnce * 1000,
+                    cs42l42->btn_det_event_dbnce * 2000);
+
+       /* Test all 4 level detect biases */
+       bias_level = 1;
+       do {
+               /* Adjust button detect level sensitivity */
+               regmap_update_bits(cs42l42->regmap,
+                       CS42L42_MIC_DET_CTL1,
+                       CS42L42_LATCH_TO_VP_MASK |
+                       CS42L42_EVENT_STAT_SEL_MASK |
+                       CS42L42_HS_DET_LEVEL_MASK,
+                       (1 << CS42L42_LATCH_TO_VP_SHIFT) |
+                       (0 << CS42L42_EVENT_STAT_SEL_SHIFT) |
+                       (cs42l42->bias_thresholds[bias_level] <<
+                       CS42L42_HS_DET_LEVEL_SHIFT));
+
+               regmap_read(cs42l42->regmap, CS42L42_DET_STATUS2,
+                               &detect_status);
+       } while ((detect_status & CS42L42_HS_TRUE_MASK) &&
+               (++bias_level < CS42L42_NUM_BIASES));
+
+       switch (bias_level) {
+       case 1: /* Function C button press */
+               dev_dbg(cs42l42->codec->dev, "Function C button press\n");
+               break;
+       case 2: /* Function B button press */
+               dev_dbg(cs42l42->codec->dev, "Function B button press\n");
+               break;
+       case 3: /* Function D button press */
+               dev_dbg(cs42l42->codec->dev, "Function D button press\n");
+               break;
+       case 4: /* Function A button press */
+               dev_dbg(cs42l42->codec->dev, "Function A button press\n");
+               break;
+       }
+
+       /* Set button detect level sensitivity back to default */
+       regmap_update_bits(cs42l42->regmap,
+               CS42L42_MIC_DET_CTL1,
+               CS42L42_LATCH_TO_VP_MASK |
+               CS42L42_EVENT_STAT_SEL_MASK |
+               CS42L42_HS_DET_LEVEL_MASK,
+               (1 << CS42L42_LATCH_TO_VP_SHIFT) |
+               (0 << CS42L42_EVENT_STAT_SEL_SHIFT) |
+               (cs42l42->bias_thresholds[0] << CS42L42_HS_DET_LEVEL_SHIFT));
+
+       /* Clear any button interrupts before unmasking them */
+       regmap_read(cs42l42->regmap, CS42L42_DET_INT_STATUS2,
+                   &detect_status);
+
+       /* Unmask button detect interrupts */
+       regmap_update_bits(cs42l42->regmap,
+               CS42L42_DET_INT2_MASK,
+               CS42L42_M_DETECT_TF_MASK |
+               CS42L42_M_DETECT_FT_MASK |
+               CS42L42_M_HSBIAS_HIZ_MASK |
+               CS42L42_M_SHORT_RLS_MASK |
+               CS42L42_M_SHORT_DET_MASK,
+               (0 << CS42L42_M_DETECT_TF_SHIFT) |
+               (0 << CS42L42_M_DETECT_FT_SHIFT) |
+               (0 << CS42L42_M_HSBIAS_HIZ_SHIFT) |
+               (1 << CS42L42_M_SHORT_RLS_SHIFT) |
+               (1 << CS42L42_M_SHORT_DET_SHIFT));
+}
+
+struct cs42l42_irq_params {
+       u16 status_addr;
+       u16 mask_addr;
+       u8 mask;
+};
+
+static const struct cs42l42_irq_params irq_params_table[] = {
+       {CS42L42_ADC_OVFL_STATUS, CS42L42_ADC_OVFL_INT_MASK,
+               CS42L42_ADC_OVFL_VAL_MASK},
+       {CS42L42_MIXER_STATUS, CS42L42_MIXER_INT_MASK,
+               CS42L42_MIXER_VAL_MASK},
+       {CS42L42_SRC_STATUS, CS42L42_SRC_INT_MASK,
+               CS42L42_SRC_VAL_MASK},
+       {CS42L42_ASP_RX_STATUS, CS42L42_ASP_RX_INT_MASK,
+               CS42L42_ASP_RX_VAL_MASK},
+       {CS42L42_ASP_TX_STATUS, CS42L42_ASP_TX_INT_MASK,
+               CS42L42_ASP_TX_VAL_MASK},
+       {CS42L42_CODEC_STATUS, CS42L42_CODEC_INT_MASK,
+               CS42L42_CODEC_VAL_MASK},
+       {CS42L42_DET_INT_STATUS1, CS42L42_DET_INT1_MASK,
+               CS42L42_DET_INT_VAL1_MASK},
+       {CS42L42_DET_INT_STATUS2, CS42L42_DET_INT2_MASK,
+               CS42L42_DET_INT_VAL2_MASK},
+       {CS42L42_SRCPL_INT_STATUS, CS42L42_SRCPL_INT_MASK,
+               CS42L42_SRCPL_VAL_MASK},
+       {CS42L42_VPMON_STATUS, CS42L42_VPMON_INT_MASK,
+               CS42L42_VPMON_VAL_MASK},
+       {CS42L42_PLL_LOCK_STATUS, CS42L42_PLL_LOCK_INT_MASK,
+               CS42L42_PLL_LOCK_VAL_MASK},
+       {CS42L42_TSRS_PLUG_STATUS, CS42L42_TSRS_PLUG_INT_MASK,
+               CS42L42_TSRS_PLUG_VAL_MASK}
+};
+
+static irqreturn_t cs42l42_irq_thread(int irq, void *data)
+{
+       struct cs42l42_private *cs42l42 = (struct cs42l42_private *)data;
+       struct snd_soc_codec *codec = cs42l42->codec;
+       unsigned int stickies[12];
+       unsigned int masks[12];
+       unsigned int current_plug_status;
+       unsigned int current_button_status;
+       unsigned int i;
+
+       /* Read sticky registers to clear interurpt */
+       for (i = 0; i < ARRAY_SIZE(stickies); i++) {
+               regmap_read(cs42l42->regmap, irq_params_table[i].status_addr,
+                               &(stickies[i]));
+               regmap_read(cs42l42->regmap, irq_params_table[i].mask_addr,
+                               &(masks[i]));
+               stickies[i] = stickies[i] & (~masks[i]) &
+                               irq_params_table[i].mask;
+       }
+
+       /* Read tip sense status before handling type detect */
+       current_plug_status = (stickies[11] &
+               (CS42L42_TS_PLUG_MASK | CS42L42_TS_UNPLUG_MASK)) >>
+               CS42L42_TS_PLUG_SHIFT;
+
+       /* Read button sense status */
+       current_button_status = stickies[7] &
+               (CS42L42_M_DETECT_TF_MASK |
+               CS42L42_M_DETECT_FT_MASK |
+               CS42L42_M_HSBIAS_HIZ_MASK);
+
+       /* Check auto-detect status */
+       if ((~masks[5]) & irq_params_table[5].mask) {
+               if (stickies[5] & CS42L42_HSDET_AUTO_DONE_MASK) {
+                       cs42l42_process_hs_type_detect(cs42l42);
+                       dev_dbg(codec->dev,
+                               "Auto detect done (%d)\n",
+                               cs42l42->hs_type);
+               }
+       }
+
+       /* Check tip sense status */
+       if ((~masks[11]) & irq_params_table[11].mask) {
+               switch (current_plug_status) {
+               case CS42L42_TS_PLUG:
+                       if (cs42l42->plug_state != CS42L42_TS_PLUG) {
+                               cs42l42->plug_state = CS42L42_TS_PLUG;
+                               cs42l42_init_hs_type_detect(cs42l42);
+                       }
+                       break;
+
+               case CS42L42_TS_UNPLUG:
+                       if (cs42l42->plug_state != CS42L42_TS_UNPLUG) {
+                               cs42l42->plug_state = CS42L42_TS_UNPLUG;
+                               cs42l42_cancel_hs_type_detect(cs42l42);
+                               dev_dbg(codec->dev,
+                                       "Unplug event\n");
+                       }
+                       break;
+
+               default:
+                       if (cs42l42->plug_state != CS42L42_TS_TRANS)
+                               cs42l42->plug_state = CS42L42_TS_TRANS;
+               }
+       }
+
+       /* Check button detect status */
+       if ((~masks[7]) & irq_params_table[7].mask) {
+               if (!(current_button_status &
+                       CS42L42_M_HSBIAS_HIZ_MASK)) {
+
+                       if (current_button_status &
+                               CS42L42_M_DETECT_TF_MASK) {
+                               dev_dbg(codec->dev,
+                                       "Button released\n");
+                       } else if (current_button_status &
+                               CS42L42_M_DETECT_FT_MASK) {
+                               cs42l42_handle_button_press(cs42l42);
+                       }
+               }
+       }
+
+       return IRQ_HANDLED;
+}
+
+static void cs42l42_set_interrupt_masks(struct cs42l42_private *cs42l42)
+{
+       regmap_update_bits(cs42l42->regmap, CS42L42_ADC_OVFL_INT_MASK,
+                       CS42L42_ADC_OVFL_MASK,
+                       (1 << CS42L42_ADC_OVFL_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_MIXER_INT_MASK,
+                       CS42L42_MIX_CHB_OVFL_MASK |
+                       CS42L42_MIX_CHA_OVFL_MASK |
+                       CS42L42_EQ_OVFL_MASK |
+                       CS42L42_EQ_BIQUAD_OVFL_MASK,
+                       (1 << CS42L42_MIX_CHB_OVFL_SHIFT) |
+                       (1 << CS42L42_MIX_CHA_OVFL_SHIFT) |
+                       (1 << CS42L42_EQ_OVFL_SHIFT) |
+                       (1 << CS42L42_EQ_BIQUAD_OVFL_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_SRC_INT_MASK,
+                       CS42L42_SRC_ILK_MASK |
+                       CS42L42_SRC_OLK_MASK |
+                       CS42L42_SRC_IUNLK_MASK |
+                       CS42L42_SRC_OUNLK_MASK,
+                       (1 << CS42L42_SRC_ILK_SHIFT) |
+                       (1 << CS42L42_SRC_OLK_SHIFT) |
+                       (1 << CS42L42_SRC_IUNLK_SHIFT) |
+                       (1 << CS42L42_SRC_OUNLK_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_ASP_RX_INT_MASK,
+                       CS42L42_ASPRX_NOLRCK_MASK |
+                       CS42L42_ASPRX_EARLY_MASK |
+                       CS42L42_ASPRX_LATE_MASK |
+                       CS42L42_ASPRX_ERROR_MASK |
+                       CS42L42_ASPRX_OVLD_MASK,
+                       (1 << CS42L42_ASPRX_NOLRCK_SHIFT) |
+                       (1 << CS42L42_ASPRX_EARLY_SHIFT) |
+                       (1 << CS42L42_ASPRX_LATE_SHIFT) |
+                       (1 << CS42L42_ASPRX_ERROR_SHIFT) |
+                       (1 << CS42L42_ASPRX_OVLD_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_ASP_TX_INT_MASK,
+                       CS42L42_ASPTX_NOLRCK_MASK |
+                       CS42L42_ASPTX_EARLY_MASK |
+                       CS42L42_ASPTX_LATE_MASK |
+                       CS42L42_ASPTX_SMERROR_MASK,
+                       (1 << CS42L42_ASPTX_NOLRCK_SHIFT) |
+                       (1 << CS42L42_ASPTX_EARLY_SHIFT) |
+                       (1 << CS42L42_ASPTX_LATE_SHIFT) |
+                       (1 << CS42L42_ASPTX_SMERROR_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_CODEC_INT_MASK,
+                       CS42L42_PDN_DONE_MASK |
+                       CS42L42_HSDET_AUTO_DONE_MASK,
+                       (1 << CS42L42_PDN_DONE_SHIFT) |
+                       (1 << CS42L42_HSDET_AUTO_DONE_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_SRCPL_INT_MASK,
+                       CS42L42_SRCPL_ADC_LK_MASK |
+                       CS42L42_SRCPL_DAC_LK_MASK |
+                       CS42L42_SRCPL_ADC_UNLK_MASK |
+                       CS42L42_SRCPL_DAC_UNLK_MASK,
+                       (1 << CS42L42_SRCPL_ADC_LK_SHIFT) |
+                       (1 << CS42L42_SRCPL_DAC_LK_SHIFT) |
+                       (1 << CS42L42_SRCPL_ADC_UNLK_SHIFT) |
+                       (1 << CS42L42_SRCPL_DAC_UNLK_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_DET_INT1_MASK,
+                       CS42L42_TIP_SENSE_UNPLUG_MASK |
+                       CS42L42_TIP_SENSE_PLUG_MASK |
+                       CS42L42_HSBIAS_SENSE_MASK,
+                       (1 << CS42L42_TIP_SENSE_UNPLUG_SHIFT) |
+                       (1 << CS42L42_TIP_SENSE_PLUG_SHIFT) |
+                       (1 << CS42L42_HSBIAS_SENSE_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_DET_INT2_MASK,
+                       CS42L42_M_DETECT_TF_MASK |
+                       CS42L42_M_DETECT_FT_MASK |
+                       CS42L42_M_HSBIAS_HIZ_MASK |
+                       CS42L42_M_SHORT_RLS_MASK |
+                       CS42L42_M_SHORT_DET_MASK,
+                       (1 << CS42L42_M_DETECT_TF_SHIFT) |
+                       (1 << CS42L42_M_DETECT_FT_SHIFT) |
+                       (1 << CS42L42_M_HSBIAS_HIZ_SHIFT) |
+                       (1 << CS42L42_M_SHORT_RLS_SHIFT) |
+                       (1 << CS42L42_M_SHORT_DET_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_VPMON_INT_MASK,
+                       CS42L42_VPMON_MASK,
+                       (1 << CS42L42_VPMON_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_PLL_LOCK_INT_MASK,
+                       CS42L42_PLL_LOCK_MASK,
+                       (1 << CS42L42_PLL_LOCK_SHIFT));
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_TSRS_PLUG_INT_MASK,
+                       CS42L42_RS_PLUG_MASK |
+                       CS42L42_RS_UNPLUG_MASK |
+                       CS42L42_TS_PLUG_MASK |
+                       CS42L42_TS_UNPLUG_MASK,
+                       (1 << CS42L42_RS_PLUG_SHIFT) |
+                       (1 << CS42L42_RS_UNPLUG_SHIFT) |
+                       (0 << CS42L42_TS_PLUG_SHIFT) |
+                       (0 << CS42L42_TS_UNPLUG_SHIFT));
+}
+
+static void cs42l42_setup_hs_type_detect(struct cs42l42_private *cs42l42)
+{
+       unsigned int reg;
+
+       cs42l42->hs_type = CS42L42_PLUG_INVALID;
+
+       /* Latch analog controls to VP power domain */
+       regmap_update_bits(cs42l42->regmap, CS42L42_MIC_DET_CTL1,
+                       CS42L42_LATCH_TO_VP_MASK |
+                       CS42L42_EVENT_STAT_SEL_MASK |
+                       CS42L42_HS_DET_LEVEL_MASK,
+                       (1 << CS42L42_LATCH_TO_VP_SHIFT) |
+                       (0 << CS42L42_EVENT_STAT_SEL_SHIFT) |
+                       (cs42l42->bias_thresholds[0] <<
+                       CS42L42_HS_DET_LEVEL_SHIFT));
+
+       /* Remove ground noise-suppression clamps */
+       regmap_update_bits(cs42l42->regmap,
+                       CS42L42_HS_CLAMP_DISABLE,
+                       CS42L42_HS_CLAMP_DISABLE_MASK,
+                       (1 << CS42L42_HS_CLAMP_DISABLE_SHIFT));
+
+       /* Enable the tip sense circuit */
+       regmap_update_bits(cs42l42->regmap, CS42L42_TIPSENSE_CTL,
+                       CS42L42_TIP_SENSE_CTRL_MASK |
+                       CS42L42_TIP_SENSE_INV_MASK |
+                       CS42L42_TIP_SENSE_DEBOUNCE_MASK,
+                       (3 << CS42L42_TIP_SENSE_CTRL_SHIFT) |
+                       (0 << CS42L42_TIP_SENSE_INV_SHIFT) |
+                       (2 << CS42L42_TIP_SENSE_DEBOUNCE_SHIFT));
+
+       /* Save the initial status of the tip sense */
+       regmap_read(cs42l42->regmap,
+                         CS42L42_TSRS_PLUG_STATUS,
+                         &reg);
+       cs42l42->plug_state = (((char) reg) &
+                     (CS42L42_TS_PLUG_MASK | CS42L42_TS_UNPLUG_MASK)) >>
+                     CS42L42_TS_PLUG_SHIFT;
+}
+
+static const unsigned int threshold_defaults[] = {
+       CS42L42_HS_DET_LEVEL_15,
+       CS42L42_HS_DET_LEVEL_8,
+       CS42L42_HS_DET_LEVEL_4,
+       CS42L42_HS_DET_LEVEL_1
+};
+
+static int cs42l42_handle_device_data(struct i2c_client *i2c_client,
+                                       struct cs42l42_private *cs42l42)
+{
+       struct device_node *np = i2c_client->dev.of_node;
+       unsigned int val;
+       unsigned int thresholds[CS42L42_NUM_BIASES];
+       int ret;
+       int i;
+
+       ret = of_property_read_u32(np, "cirrus,ts-inv", &val);
+
+       if (!ret) {
+               switch (val) {
+               case CS42L42_TS_INV_EN:
+               case CS42L42_TS_INV_DIS:
+                       cs42l42->ts_inv = val;
+                       break;
+               default:
+                       dev_err(&i2c_client->dev,
+                               "Wrong cirrus,ts-inv DT value %d\n",
+                               val);
+                       cs42l42->ts_inv = CS42L42_TS_INV_DIS;
+               }
+       } else {
+               cs42l42->ts_inv = CS42L42_TS_INV_DIS;
+       }
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_TSENSE_CTL,
+                       CS42L42_TS_INV_MASK,
+                       (cs42l42->ts_inv << CS42L42_TS_INV_SHIFT));
+
+       ret = of_property_read_u32(np, "cirrus,ts-dbnc-rise", &val);
+
+       if (!ret) {
+               switch (val) {
+               case CS42L42_TS_DBNCE_0:
+               case CS42L42_TS_DBNCE_125:
+               case CS42L42_TS_DBNCE_250:
+               case CS42L42_TS_DBNCE_500:
+               case CS42L42_TS_DBNCE_750:
+               case CS42L42_TS_DBNCE_1000:
+               case CS42L42_TS_DBNCE_1250:
+               case CS42L42_TS_DBNCE_1500:
+                       cs42l42->ts_dbnc_rise = val;
+                       break;
+               default:
+                       dev_err(&i2c_client->dev,
+                               "Wrong cirrus,ts-dbnc-rise DT value %d\n",
+                               val);
+                       cs42l42->ts_dbnc_rise = CS42L42_TS_DBNCE_1000;
+               }
+       } else {
+               cs42l42->ts_dbnc_rise = CS42L42_TS_DBNCE_1000;
+       }
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_TSENSE_CTL,
+                       CS42L42_TS_RISE_DBNCE_TIME_MASK,
+                       (cs42l42->ts_dbnc_rise <<
+                       CS42L42_TS_RISE_DBNCE_TIME_SHIFT));
+
+       ret = of_property_read_u32(np, "cirrus,ts-dbnc-fall", &val);
+
+       if (!ret) {
+               switch (val) {
+               case CS42L42_TS_DBNCE_0:
+               case CS42L42_TS_DBNCE_125:
+               case CS42L42_TS_DBNCE_250:
+               case CS42L42_TS_DBNCE_500:
+               case CS42L42_TS_DBNCE_750:
+               case CS42L42_TS_DBNCE_1000:
+               case CS42L42_TS_DBNCE_1250:
+               case CS42L42_TS_DBNCE_1500:
+                       cs42l42->ts_dbnc_fall = val;
+                       break;
+               default:
+                       dev_err(&i2c_client->dev,
+                               "Wrong cirrus,ts-dbnc-fall DT value %d\n",
+                               val);
+                       cs42l42->ts_dbnc_fall = CS42L42_TS_DBNCE_0;
+               }
+       } else {
+               cs42l42->ts_dbnc_fall = CS42L42_TS_DBNCE_0;
+       }
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_TSENSE_CTL,
+                       CS42L42_TS_FALL_DBNCE_TIME_MASK,
+                       (cs42l42->ts_dbnc_fall <<
+                       CS42L42_TS_FALL_DBNCE_TIME_SHIFT));
+
+       ret = of_property_read_u32(np, "cirrus,btn-det-init-dbnce", &val);
+
+       if (!ret) {
+               if ((val >= CS42L42_BTN_DET_INIT_DBNCE_MIN) &&
+                       (val <= CS42L42_BTN_DET_INIT_DBNCE_MAX))
+                       cs42l42->btn_det_init_dbnce = val;
+               else {
+                       dev_err(&i2c_client->dev,
+                               "Wrong cirrus,btn-det-init-dbnce DT value %d\n",
+                               val);
+                       cs42l42->btn_det_init_dbnce =
+                               CS42L42_BTN_DET_INIT_DBNCE_DEFAULT;
+               }
+       } else {
+               cs42l42->btn_det_init_dbnce =
+                       CS42L42_BTN_DET_INIT_DBNCE_DEFAULT;
+       }
+
+       ret = of_property_read_u32(np, "cirrus,btn-det-event-dbnce", &val);
+
+       if (!ret) {
+               if ((val >= CS42L42_BTN_DET_EVENT_DBNCE_MIN) &&
+                       (val <= CS42L42_BTN_DET_EVENT_DBNCE_MAX))
+                       cs42l42->btn_det_event_dbnce = val;
+               else {
+                       dev_err(&i2c_client->dev,
+                       "Wrong cirrus,btn-det-event-dbnce DT value %d\n", val);
+                       cs42l42->btn_det_event_dbnce =
+                               CS42L42_BTN_DET_EVENT_DBNCE_DEFAULT;
+               }
+       } else {
+               cs42l42->btn_det_event_dbnce =
+                       CS42L42_BTN_DET_EVENT_DBNCE_DEFAULT;
+       }
+
+       ret = of_property_read_u32_array(np, "cirrus,bias-lvls",
+                                  (u32 *)thresholds, CS42L42_NUM_BIASES);
+
+       if (!ret) {
+               for (i = 0; i < CS42L42_NUM_BIASES; i++) {
+                       if ((thresholds[i] >= CS42L42_HS_DET_LEVEL_MIN) &&
+                               (thresholds[i] <= CS42L42_HS_DET_LEVEL_MAX))
+                               cs42l42->bias_thresholds[i] = thresholds[i];
+                       else {
+                               dev_err(&i2c_client->dev,
+                               "Wrong cirrus,bias-lvls[%d] DT value %d\n", i,
+                                       thresholds[i]);
+                               cs42l42->bias_thresholds[i] =
+                                       threshold_defaults[i];
+                       }
+               }
+       } else {
+               for (i = 0; i < CS42L42_NUM_BIASES; i++)
+                       cs42l42->bias_thresholds[i] = threshold_defaults[i];
+       }
+
+       ret = of_property_read_u32(np, "cirrus,hs-bias-ramp-rate", &val);
+
+       if (!ret) {
+               switch (val) {
+               case CS42L42_HSBIAS_RAMP_FAST_RISE_SLOW_FALL:
+                       cs42l42->hs_bias_ramp_rate = val;
+                       cs42l42->hs_bias_ramp_time = CS42L42_HSBIAS_RAMP_TIME0;
+                       break;
+               case CS42L42_HSBIAS_RAMP_FAST:
+                       cs42l42->hs_bias_ramp_rate = val;
+                       cs42l42->hs_bias_ramp_time = CS42L42_HSBIAS_RAMP_TIME1;
+                       break;
+               case CS42L42_HSBIAS_RAMP_SLOW:
+                       cs42l42->hs_bias_ramp_rate = val;
+                       cs42l42->hs_bias_ramp_time = CS42L42_HSBIAS_RAMP_TIME2;
+                       break;
+               case CS42L42_HSBIAS_RAMP_SLOWEST:
+                       cs42l42->hs_bias_ramp_rate = val;
+                       cs42l42->hs_bias_ramp_time = CS42L42_HSBIAS_RAMP_TIME3;
+                       break;
+               default:
+                       dev_err(&i2c_client->dev,
+                               "Wrong cirrus,hs-bias-ramp-rate DT value %d\n",
+                               val);
+                       cs42l42->hs_bias_ramp_rate = CS42L42_HSBIAS_RAMP_SLOW;
+                       cs42l42->hs_bias_ramp_time = CS42L42_HSBIAS_RAMP_TIME2;
+               }
+       } else {
+               cs42l42->hs_bias_ramp_rate = CS42L42_HSBIAS_RAMP_SLOW;
+               cs42l42->hs_bias_ramp_time = CS42L42_HSBIAS_RAMP_TIME2;
+       }
+
+       regmap_update_bits(cs42l42->regmap, CS42L42_HS_BIAS_CTL,
+                       CS42L42_HSBIAS_RAMP_MASK,
+                       (cs42l42->hs_bias_ramp_rate <<
+                       CS42L42_HSBIAS_RAMP_SHIFT));
+
+       return 0;
+}
+
+static int cs42l42_i2c_probe(struct i2c_client *i2c_client,
+                                      const struct i2c_device_id *id)
+{
+       struct cs42l42_private *cs42l42;
+       int ret, i;
+       unsigned int devid = 0;
+       unsigned int reg;
+
+       cs42l42 = devm_kzalloc(&i2c_client->dev, sizeof(struct cs42l42_private),
+                              GFP_KERNEL);
+       if (!cs42l42)
+               return -ENOMEM;
+
+       i2c_set_clientdata(i2c_client, cs42l42);
+
+       cs42l42->regmap = devm_regmap_init_i2c(i2c_client, &cs42l42_regmap);
+       if (IS_ERR(cs42l42->regmap)) {
+               ret = PTR_ERR(cs42l42->regmap);
+               dev_err(&i2c_client->dev, "regmap_init() failed: %d\n", ret);
+               return ret;
+       }
+
+       for (i = 0; i < ARRAY_SIZE(cs42l42->supplies); i++)
+               cs42l42->supplies[i].supply = cs42l42_supply_names[i];
+
+       ret = devm_regulator_bulk_get(&i2c_client->dev,
+                                     ARRAY_SIZE(cs42l42->supplies),
+                                     cs42l42->supplies);
+       if (ret != 0) {
+               dev_err(&i2c_client->dev,
+                       "Failed to request supplies: %d\n", ret);
+               return ret;
+       }
+
+       ret = regulator_bulk_enable(ARRAY_SIZE(cs42l42->supplies),
+                                   cs42l42->supplies);
+       if (ret != 0) {
+               dev_err(&i2c_client->dev,
+                       "Failed to enable supplies: %d\n", ret);
+               return ret;
+       }
+
+       /* Reset the Device */
+       cs42l42->reset_gpio = devm_gpiod_get_optional(&i2c_client->dev,
+               "reset", GPIOD_OUT_LOW);
+       if (IS_ERR(cs42l42->reset_gpio))
+               return PTR_ERR(cs42l42->reset_gpio);
+
+       if (cs42l42->reset_gpio) {
+               dev_dbg(&i2c_client->dev, "Found reset GPIO\n");
+               gpiod_set_value_cansleep(cs42l42->reset_gpio, 1);
+       }
+       mdelay(3);
+
+       /* Request IRQ */
+       ret = devm_request_threaded_irq(&i2c_client->dev,
+                       i2c_client->irq,
+                       NULL, cs42l42_irq_thread,
+                       IRQF_ONESHOT | IRQF_TRIGGER_LOW,
+                       "cs42l42", cs42l42);
+
+       if (ret != 0)
+               dev_err(&i2c_client->dev,
+                       "Failed to request IRQ: %d\n", ret);
+
+       /* initialize codec */
+       ret = regmap_read(cs42l42->regmap, CS42L42_DEVID_AB, &reg);
+       devid = (reg & 0xFF) << 12;
+
+       ret = regmap_read(cs42l42->regmap, CS42L42_DEVID_CD, &reg);
+       devid |= (reg & 0xFF) << 4;
+
+       ret = regmap_read(cs42l42->regmap, CS42L42_DEVID_E, &reg);
+       devid |= (reg & 0xF0) >> 4;
+
+       if (devid != CS42L42_CHIP_ID) {
+               ret = -ENODEV;
+               dev_err(&i2c_client->dev,
+                       "CS42L42 Device ID (%X). Expected %X\n",
+                       devid, CS42L42_CHIP_ID);
+               return ret;
+       }
+
+       ret = regmap_read(cs42l42->regmap, CS42L42_REVID, &reg);
+       if (ret < 0) {
+               dev_err(&i2c_client->dev, "Get Revision ID failed\n");
+               return ret;
+       }
+
+       dev_info(&i2c_client->dev,
+                "Cirrus Logic CS42L42, Revision: %02X\n", reg & 0xFF);
+
+       /* Power up the codec */
+       regmap_update_bits(cs42l42->regmap, CS42L42_PWR_CTL1,
+                       CS42L42_ASP_DAO_PDN_MASK |
+                       CS42L42_ASP_DAI_PDN_MASK |
+                       CS42L42_MIXER_PDN_MASK |
+                       CS42L42_EQ_PDN_MASK |
+                       CS42L42_HP_PDN_MASK |
+                       CS42L42_ADC_PDN_MASK |
+                       CS42L42_PDN_ALL_MASK,
+                       (1 << CS42L42_ASP_DAO_PDN_SHIFT) |
+                       (1 << CS42L42_ASP_DAI_PDN_SHIFT) |
+                       (1 << CS42L42_MIXER_PDN_SHIFT) |
+                       (1 << CS42L42_EQ_PDN_SHIFT) |
+                       (1 << CS42L42_HP_PDN_SHIFT) |
+                       (1 << CS42L42_ADC_PDN_SHIFT) |
+                       (0 << CS42L42_PDN_ALL_SHIFT));
+
+       if (i2c_client->dev.of_node) {
+               ret = cs42l42_handle_device_data(i2c_client, cs42l42);
+               if (ret != 0)
+                       return ret;
+       }
+
+       /* Setup headset detection */
+       cs42l42_setup_hs_type_detect(cs42l42);
+
+       /* Mask/Unmask Interrupts */
+       cs42l42_set_interrupt_masks(cs42l42);
+
+       /* Register codec for machine driver */
+       ret =  snd_soc_register_codec(&i2c_client->dev,
+                       &soc_codec_dev_cs42l42, &cs42l42_dai, 1);
+       if (ret < 0)
+               goto err_disable;
+       return 0;
+
+err_disable:
+       regulator_bulk_disable(ARRAY_SIZE(cs42l42->supplies),
+                               cs42l42->supplies);
+       return ret;
+}
+
+static int cs42l42_i2c_remove(struct i2c_client *i2c_client)
+{
+       struct cs42l42_private *cs42l42 = i2c_get_clientdata(i2c_client);
+
+       snd_soc_unregister_codec(&i2c_client->dev);
+
+       /* Hold down reset */
+       if (cs42l42->reset_gpio)
+               gpiod_set_value_cansleep(cs42l42->reset_gpio, 0);
+
+       return 0;
+}
+
+#ifdef CONFIG_PM
+static int cs42l42_runtime_suspend(struct device *dev)
+{
+       struct cs42l42_private *cs42l42 = dev_get_drvdata(dev);
+
+       regcache_cache_only(cs42l42->regmap, true);
+       regcache_mark_dirty(cs42l42->regmap);
+
+       /* Hold down reset */
+       if (cs42l42->reset_gpio)
+               gpiod_set_value_cansleep(cs42l42->reset_gpio, 0);
+
+       /* remove power */
+       regulator_bulk_disable(ARRAY_SIZE(cs42l42->supplies),
+                               cs42l42->supplies);
+
+       return 0;
+}
+
+static int cs42l42_runtime_resume(struct device *dev)
+{
+       struct cs42l42_private *cs42l42 = dev_get_drvdata(dev);
+       int ret;
+
+       /* Enable power */
+       ret = regulator_bulk_enable(ARRAY_SIZE(cs42l42->supplies),
+                                       cs42l42->supplies);
+       if (ret != 0) {
+               dev_err(dev, "Failed to enable supplies: %d\n",
+                       ret);
+               return ret;
+       }
+
+       if (cs42l42->reset_gpio)
+               gpiod_set_value_cansleep(cs42l42->reset_gpio, 1);
+
+       regcache_cache_only(cs42l42->regmap, false);
+       regcache_sync(cs42l42->regmap);
+
+       return 0;
+}
+#endif
+
+static const struct dev_pm_ops cs42l42_runtime_pm = {
+       SET_RUNTIME_PM_OPS(cs42l42_runtime_suspend, cs42l42_runtime_resume,
+                          NULL)
+};
+
+static const struct of_device_id cs42l42_of_match[] = {
+       { .compatible = "cirrus,cs42l42", },
+       {},
+};
+MODULE_DEVICE_TABLE(of, cs42l42_of_match);
+
+
+static const struct i2c_device_id cs42l42_id[] = {
+       {"cs42l42", 0},
+       {}
+};
+
+MODULE_DEVICE_TABLE(i2c, cs42l42_id);
+
+static struct i2c_driver cs42l42_i2c_driver = {
+       .driver = {
+               .name = "cs42l42",
+               .owner = THIS_MODULE,
+               .pm = &cs42l42_runtime_pm,
+               .of_match_table = cs42l42_of_match,
+               },
+       .id_table = cs42l42_id,
+       .probe = cs42l42_i2c_probe,
+       .remove = cs42l42_i2c_remove,
+};
+
+module_i2c_driver(cs42l42_i2c_driver);
+
+MODULE_DESCRIPTION("ASoC CS42L42 driver");
+MODULE_AUTHOR("James Schulman, Cirrus Logic Inc, <james.schulman@cirrus.com>");
+MODULE_AUTHOR("Brian Austin, Cirrus Logic Inc, <brian.austin@cirrus.com>");
+MODULE_AUTHOR("Michael White, Cirrus Logic Inc, <michael.white@cirrus.com>");
+MODULE_LICENSE("GPL");
diff --git a/sound/soc/codecs/cs42l42.h b/sound/soc/codecs/cs42l42.h
new file mode 100644 (file)
index 0000000..d87a0a5
--- /dev/null
@@ -0,0 +1,776 @@
+/*
+ * cs42l42.h -- CS42L42 ALSA SoC audio driver header
+ *
+ * Copyright 2016 Cirrus Logic, Inc.
+ *
+ * Author: James Schulman <james.schulman@cirrus.com>
+ * Author: Brian Austin <brian.austin@cirrus.com>
+ * Author: Michael White <michael.white@cirrus.com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ */
+
+#ifndef __CS42L42_H__
+#define __CS42L42_H__
+
+#define CS42L42_PAGE_REGISTER  0x00    /* Page Select Register */
+#define CS42L42_WIN_START      0x00
+#define CS42L42_WIN_LEN                0x100
+#define CS42L42_RANGE_MIN      0x00
+#define CS42L42_RANGE_MAX      0x7F
+
+#define CS42L42_PAGE_10                0x1000
+#define CS42L42_PAGE_11                0x1100
+#define CS42L42_PAGE_12                0x1200
+#define CS42L42_PAGE_13                0x1300
+#define CS42L42_PAGE_15                0x1500
+#define CS42L42_PAGE_19                0x1900
+#define CS42L42_PAGE_1B                0x1B00
+#define CS42L42_PAGE_1C                0x1C00
+#define CS42L42_PAGE_1D                0x1D00
+#define CS42L42_PAGE_1F                0x1F00
+#define CS42L42_PAGE_20                0x2000
+#define CS42L42_PAGE_21                0x2100
+#define CS42L42_PAGE_23                0x2300
+#define CS42L42_PAGE_24                0x2400
+#define CS42L42_PAGE_25                0x2500
+#define CS42L42_PAGE_26                0x2600
+#define CS42L42_PAGE_28                0x2800
+#define CS42L42_PAGE_29                0x2900
+#define CS42L42_PAGE_2A                0x2A00
+#define CS42L42_PAGE_30                0x3000
+
+#define CS42L42_CHIP_ID                0x42A42
+
+/* Page 0x10 Global Registers */
+#define CS42L42_DEVID_AB               (CS42L42_PAGE_10 + 0x01)
+#define CS42L42_DEVID_CD               (CS42L42_PAGE_10 + 0x02)
+#define CS42L42_DEVID_E                        (CS42L42_PAGE_10 + 0x03)
+#define CS42L42_FABID                  (CS42L42_PAGE_10 + 0x04)
+#define CS42L42_REVID                  (CS42L42_PAGE_10 + 0x05)
+#define CS42L42_FRZ_CTL                        (CS42L42_PAGE_10 + 0x06)
+
+#define CS42L42_SRC_CTL                        (CS42L42_PAGE_10 + 0x07)
+#define CS42L42_SRC_BYPASS_DAC_SHIFT   1
+#define CS42L42_SRC_BYPASS_DAC_MASK    (1 << CS42L42_SRC_BYPASS_DAC_SHIFT)
+
+#define CS42L42_MCLK_STATUS            (CS42L42_PAGE_10 + 0x08)
+
+#define CS42L42_MCLK_CTL               (CS42L42_PAGE_10 + 0x09)
+#define CS42L42_INTERNAL_FS_SHIFT      1
+#define CS42L42_INTERNAL_FS_MASK       (1 << CS42L42_INTERNAL_FS_SHIFT)
+
+#define CS42L42_SFTRAMP_RATE           (CS42L42_PAGE_10 + 0x0A)
+#define CS42L42_I2C_DEBOUNCE           (CS42L42_PAGE_10 + 0x0E)
+#define CS42L42_I2C_STRETCH            (CS42L42_PAGE_10 + 0x0F)
+#define CS42L42_I2C_TIMEOUT            (CS42L42_PAGE_10 + 0x10)
+
+/* Page 0x11 Power and Headset Detect Registers */
+#define CS42L42_PWR_CTL1               (CS42L42_PAGE_11 + 0x01)
+#define CS42L42_ASP_DAO_PDN_SHIFT      7
+#define CS42L42_ASP_DAO_PDN_MASK       (1 << CS42L42_ASP_DAO_PDN_SHIFT)
+#define CS42L42_ASP_DAI_PDN_SHIFT      6
+#define CS42L42_ASP_DAI_PDN_MASK       (1 << CS42L42_ASP_DAI_PDN_SHIFT)
+#define CS42L42_MIXER_PDN_SHIFT                5
+#define CS42L42_MIXER_PDN_MASK         (1 << CS42L42_MIXER_PDN_SHIFT)
+#define CS42L42_EQ_PDN_SHIFT           4
+#define CS42L42_EQ_PDN_MASK            (1 << CS42L42_EQ_PDN_SHIFT)
+#define CS42L42_HP_PDN_SHIFT           3
+#define CS42L42_HP_PDN_MASK            (1 << CS42L42_HP_PDN_SHIFT)
+#define CS42L42_ADC_PDN_SHIFT          2
+#define CS42L42_ADC_PDN_MASK           (1 << CS42L42_HP_PDN_SHIFT)
+#define CS42L42_PDN_ALL_SHIFT          0
+#define CS42L42_PDN_ALL_MASK           (1 << CS42L42_PDN_ALL_SHIFT)
+
+#define CS42L42_PWR_CTL2               (CS42L42_PAGE_11 + 0x02)
+#define CS42L42_ADC_SRC_PDNB_SHIFT     0
+#define CS42L42_ADC_SRC_PDNB_MASK      (1 << CS42L42_ADC_SRC_PDNB_SHIFT)
+#define CS42L42_DAC_SRC_PDNB_SHIFT     1
+#define CS42L42_DAC_SRC_PDNB_MASK      (1 << CS42L42_DAC_SRC_PDNB_SHIFT)
+#define CS42L42_ASP_DAI1_PDN_SHIFT     2
+#define CS42L42_ASP_DAI1_PDN_MASK      (1 << CS42L42_ASP_DAI1_PDN_SHIFT)
+#define CS42L42_SRC_PDN_OVERRIDE_SHIFT 3
+#define CS42L42_SRC_PDN_OVERRIDE_MASK  (1 << CS42L42_SRC_PDN_OVERRIDE_SHIFT)
+#define CS42L42_DISCHARGE_FILT_SHIFT   4
+#define CS42L42_DISCHARGE_FILT_MASK    (1 << CS42L42_DISCHARGE_FILT_SHIFT)
+
+#define CS42L42_PWR_CTL3                       (CS42L42_PAGE_11 + 0x03)
+#define CS42L42_RING_SENSE_PDNB_SHIFT          1
+#define CS42L42_RING_SENSE_PDNB_MASK           (1 << \
+                                       CS42L42_RING_SENSE_PDNB_SHIFT)
+#define CS42L42_VPMON_PDNB_SHIFT               2
+#define CS42L42_VPMON_PDNB_MASK                        (1 << \
+                                       CS42L42_VPMON_PDNB_SHIFT)
+#define CS42L42_SW_CLK_STP_STAT_SEL_SHIFT      5
+#define CS42L42_SW_CLK_STP_STAT_SEL_MASK       (3 << \
+                                       CS42L42_SW_CLK_STP_STAT_SEL_SHIFT)
+
+#define CS42L42_RSENSE_CTL1                    (CS42L42_PAGE_11 + 0x04)
+#define CS42L42_RS_TRIM_R_SHIFT                        0
+#define CS42L42_RS_TRIM_R_MASK                 (1 << \
+                                       CS42L42_RS_TRIM_R_SHIFT)
+#define CS42L42_RS_TRIM_T_SHIFT                        1
+#define CS42L42_RS_TRIM_T_MASK                 (1 << \
+                                       CS42L42_RS_TRIM_T_SHIFT)
+#define CS42L42_HPREF_RS_SHIFT                 2
+#define CS42L42_HPREF_RS_MASK                  (1 << \
+                                       CS42L42_HPREF_RS_SHIFT)
+#define CS42L42_HSBIAS_FILT_REF_RS_SHIFT       3
+#define CS42L42_HSBIAS_FILT_REF_RS_MASK                (1 << \
+                                       CS42L42_HSBIAS_FILT_REF_RS_SHIFT)
+#define CS42L42_RING_SENSE_PU_HIZ_SHIFT                6
+#define CS42L42_RING_SENSE_PU_HIZ_MASK         (1 << \
+                                       CS42L42_RING_SENSE_PU_HIZ_SHIFT)
+
+#define CS42L42_RSENSE_CTL2            (CS42L42_PAGE_11 + 0x05)
+#define CS42L42_TS_RS_GATE_SHIFT       7
+#define CS42L42_TS_RS_GATE_MAS         (1 << CS42L42_TS_RS_GATE_SHIFT)
+
+#define CS42L42_OSC_SWITCH             (CS42L42_PAGE_11 + 0x07)
+#define CS42L42_SCLK_PRESENT_SHIFT     0
+#define CS42L42_SCLK_PRESENT_MASK      (1 << CS42L42_SCLK_PRESENT_SHIFT)
+
+#define CS42L42_OSC_SWITCH_STATUS      (CS42L42_PAGE_11 + 0x09)
+#define CS42L42_OSC_SW_SEL_STAT_SHIFT  0
+#define CS42L42_OSC_SW_SEL_STAT_MASK   (3 << CS42L42_OSC_SW_SEL_STAT_SHIFT)
+#define CS42L42_OSC_PDNB_STAT_SHIFT    2
+#define CS42L42_OSC_PDNB_STAT_MASK     (1 << CS42L42_OSC_SW_SEL_STAT_SHIFT)
+
+#define CS42L42_RSENSE_CTL3                    (CS42L42_PAGE_11 + 0x12)
+#define CS42L42_RS_RISE_DBNCE_TIME_SHIFT       0
+#define CS42L42_RS_RISE_DBNCE_TIME_MASK                (7 << \
+                                       CS42L42_RS_RISE_DBNCE_TIME_SHIFT)
+#define CS42L42_RS_FALL_DBNCE_TIME_SHIFT       3
+#define CS42L42_RS_FALL_DBNCE_TIME_MASK                (7 << \
+                                       CS42L42_RS_FALL_DBNCE_TIME_SHIFT)
+#define CS42L42_RS_PU_EN_SHIFT                 6
+#define CS42L42_RS_PU_EN_MASK                  (1 << \
+                                       CS42L42_RS_PU_EN_SHIFT)
+#define CS42L42_RS_INV_SHIFT                   7
+#define CS42L42_RS_INV_MASK                    (1 << \
+                                       CS42L42_RS_INV_SHIFT)
+
+#define CS42L42_TSENSE_CTL                     (CS42L42_PAGE_11 + 0x13)
+#define CS42L42_TS_RISE_DBNCE_TIME_SHIFT       0
+#define CS42L42_TS_RISE_DBNCE_TIME_MASK                (7 << \
+                                       CS42L42_TS_RISE_DBNCE_TIME_SHIFT)
+#define CS42L42_TS_FALL_DBNCE_TIME_SHIFT       3
+#define CS42L42_TS_FALL_DBNCE_TIME_MASK                (7 << \
+                                       CS42L42_TS_FALL_DBNCE_TIME_SHIFT)
+#define CS42L42_TS_INV_SHIFT                   7
+#define CS42L42_TS_INV_MASK                    (1 << \
+                                       CS42L42_TS_INV_SHIFT)
+
+#define CS42L42_TSRS_INT_DISABLE       (CS42L42_PAGE_11 + 0x14)
+#define CS42L42_D_RS_PLUG_DBNC_SHIFT   0
+#define CS42L42_D_RS_PLUG_DBNC_MASK    (1 << CS42L42_D_RS_PLUG_DBNC_SHIFT)
+#define CS42L42_D_RS_UNPLUG_DBNC_SHIFT 1
+#define CS42L42_D_RS_UNPLUG_DBNC_MASK  (1 << CS42L42_D_RS_UNPLUG_DBNC_SHIFT)
+#define CS42L42_D_TS_PLUG_DBNC_SHIFT   2
+#define CS42L42_D_TS_PLUG_DBNC_MASK    (1 << CS42L42_D_TS_PLUG_DBNC_SHIFT)
+#define CS42L42_D_TS_UNPLUG_DBNC_SHIFT 3
+#define CS42L42_D_TS_UNPLUG_DBNC_MASK  (1 << CS42L42_D_TS_UNPLUG_DBNC_SHIFT)
+
+#define CS42L42_TRSENSE_STATUS         (CS42L42_PAGE_11 + 0x15)
+#define CS42L42_RS_PLUG_DBNC_SHIFT     0
+#define CS42L42_RS_PLUG_DBNC_MASK      (1 << CS42L42_RS_PLUG_DBNC_SHIFT)
+#define CS42L42_RS_UNPLUG_DBNC_SHIFT   1
+#define CS42L42_RS_UNPLUG_DBNC_MASK    (1 << CS42L42_RS_UNPLUG_DBNC_SHIFT)
+#define CS42L42_TS_PLUG_DBNC_SHIFT     2
+#define CS42L42_TS_PLUG_DBNC_MASK      (1 << CS42L42_TS_PLUG_DBNC_SHIFT)
+#define CS42L42_TS_UNPLUG_DBNC_SHIFT   3
+#define CS42L42_TS_UNPLUG_DBNC_MASK    (1 << CS42L42_TS_UNPLUG_DBNC_SHIFT)
+
+#define CS42L42_HSDET_CTL1             (CS42L42_PAGE_11 + 0x1F)
+#define CS42L42_HSDET_COMP1_LVL_SHIFT  0
+#define CS42L42_HSDET_COMP1_LVL_MASK   (15 << CS42L42_HSDET_COMP1_LVL_SHIFT)
+#define CS42L42_HSDET_COMP2_LVL_SHIFT  4
+#define CS42L42_HSDET_COMP2_LVL_MASK   (15 << CS42L42_HSDET_COMP2_LVL_SHIFT)
+
+#define CS42L42_HSDET_CTL2             (CS42L42_PAGE_11 + 0x20)
+#define CS42L42_HSDET_AUTO_TIME_SHIFT  0
+#define CS42L42_HSDET_AUTO_TIME_MASK   (3 << CS42L42_HSDET_AUTO_TIME_SHIFT)
+#define CS42L42_HSBIAS_REF_SHIFT       3
+#define CS42L42_HSBIAS_REF_MASK                (1 << CS42L42_HSBIAS_REF_SHIFT)
+#define CS42L42_HSDET_SET_SHIFT                4
+#define CS42L42_HSDET_SET_MASK         (3 << CS42L42_HSDET_SET_SHIFT)
+#define CS42L42_HSDET_CTRL_SHIFT       6
+#define CS42L42_HSDET_CTRL_MASK                (3 << CS42L42_HSDET_CTRL_SHIFT)
+
+#define CS42L42_HS_SWITCH_CTL          (CS42L42_PAGE_11 + 0x21)
+#define CS42L42_SW_GNDHS_HS4_SHIFT     0
+#define CS42L42_SW_GNDHS_HS4_MASK      (1 << CS42L42_SW_GNDHS_HS4_SHIFT)
+#define CS42L42_SW_GNDHS_HS3_SHIFT     1
+#define CS42L42_SW_GNDHS_HS3_MASK      (1 << CS42L42_SW_GNDHS_HS3_SHIFT)
+#define CS42L42_SW_HSB_HS4_SHIFT       2
+#define CS42L42_SW_HSB_HS4_MASK                (1 << CS42L42_SW_HSB_HS4_SHIFT)
+#define CS42L42_SW_HSB_HS3_SHIFT       3
+#define CS42L42_SW_HSB_HS3_MASK                (1 << CS42L42_SW_HSB_HS3_SHIFT)
+#define CS42L42_SW_HSB_FILT_HS4_SHIFT  4
+#define CS42L42_SW_HSB_FILT_HS4_MASK   (1 << CS42L42_SW_HSB_FILT_HS4_SHIFT)
+#define CS42L42_SW_HSB_FILT_HS3_SHIFT  5
+#define CS42L42_SW_HSB_FILT_HS3_MASK   (1 << CS42L42_SW_HSB_FILT_HS3_SHIFT)
+#define CS42L42_SW_REF_HS4_SHIFT       6
+#define CS42L42_SW_REF_HS4_MASK                (1 << CS42L42_SW_REF_HS4_SHIFT)
+#define CS42L42_SW_REF_HS3_SHIFT       7
+#define CS42L42_SW_REF_HS3_MASK                (1 << CS42L42_SW_REF_HS3_SHIFT)
+
+#define CS42L42_HS_DET_STATUS          (CS42L42_PAGE_11 + 0x24)
+#define CS42L42_HSDET_TYPE_SHIFT       0
+#define CS42L42_HSDET_TYPE_MASK                (3 << CS42L42_HSDET_TYPE_SHIFT)
+#define CS42L42_HSDET_COMP1_OUT_SHIFT  6
+#define CS42L42_HSDET_COMP1_OUT_MASK   (1 << CS42L42_HSDET_COMP1_OUT_SHIFT)
+#define CS42L42_HSDET_COMP2_OUT_SHIFT  7
+#define CS42L42_HSDET_COMP2_OUT_MASK   (1 << CS42L42_HSDET_COMP2_OUT_SHIFT)
+#define CS42L42_PLUG_CTIA              0
+#define CS42L42_PLUG_OMTP              1
+#define CS42L42_PLUG_HEADPHONE         2
+#define CS42L42_PLUG_INVALID           3
+
+#define CS42L42_HS_CLAMP_DISABLE       (CS42L42_PAGE_11 + 0x29)
+#define CS42L42_HS_CLAMP_DISABLE_SHIFT 0
+#define CS42L42_HS_CLAMP_DISABLE_MASK  (1 << CS42L42_HS_CLAMP_DISABLE_SHIFT)
+
+/* Page 0x12 Clocking Registers */
+#define CS42L42_MCLK_SRC_SEL           (CS42L42_PAGE_12 + 0x01)
+#define CS42L42_MCLKDIV_SHIFT          1
+#define CS42L42_MCLKDIV_MASK           (1 << CS42L42_MCLKDIV_SHIFT)
+#define CS42L42_MCLK_SRC_SEL_SHIFT     0
+#define CS42L42_MCLK_SRC_SEL_MASK      (1 << CS42L42_MCLK_SRC_SEL_SHIFT)
+
+#define CS42L42_SPDIF_CLK_CFG          (CS42L42_PAGE_12 + 0x02)
+#define CS42L42_FSYNC_PW_LOWER         (CS42L42_PAGE_12 + 0x03)
+
+#define CS42L42_FSYNC_PW_UPPER                 (CS42L42_PAGE_12 + 0x04)
+#define CS42L42_FSYNC_PULSE_WIDTH_SHIFT                0
+#define CS42L42_FSYNC_PULSE_WIDTH_MASK         (0xff << \
+                                       CS42L42_FSYNC_PULSE_WIDTH_SHIFT)
+
+#define CS42L42_FSYNC_P_LOWER          (CS42L42_PAGE_12 + 0x05)
+
+#define CS42L42_FSYNC_P_UPPER          (CS42L42_PAGE_12 + 0x06)
+#define CS42L42_FSYNC_PERIOD_SHIFT     0
+#define CS42L42_FSYNC_PERIOD_MASK      (0xff << CS42L42_FSYNC_PERIOD_SHIFT)
+
+#define CS42L42_ASP_CLK_CFG            (CS42L42_PAGE_12 + 0x07)
+#define CS42L42_ASP_SCLK_EN_SHIFT      5
+#define CS42L42_ASP_SCLK_EN_MASK       (1 << CS42L42_ASP_SCLK_EN_SHIFT)
+#define CS42L42_ASP_MASTER_MODE                0x01
+#define CS42L42_ASP_SLAVE_MODE         0x00
+#define CS42L42_ASP_MODE_SHIFT         4
+#define CS42L42_ASP_MODE_MASK          (1 << CS42L42_ASP_MODE_SHIFT)
+#define CS42L42_ASP_SCPOL_IN_DAC_SHIFT 2
+#define CS42L42_ASP_SCPOL_IN_DAC_MASK  (1 << CS42L42_ASP_SCPOL_IN_DAC_SHIFT)
+#define CS42L42_ASP_LCPOL_IN_SHIFT     0
+#define CS42L42_ASP_LCPOL_IN_MASK      (1 << CS42L42_ASP_LCPOL_IN_SHIFT)
+#define CS42L42_ASP_POL_INV            1
+
+#define CS42L42_ASP_FRM_CFG            (CS42L42_PAGE_12 + 0x08)
+#define CS42L42_ASP_STP_SHIFT          4
+#define CS42L42_ASP_STP_MASK           (1 << CS42L42_ASP_STP_SHIFT)
+#define CS42L42_ASP_5050_SHIFT         3
+#define CS42L42_ASP_5050_MASK          (1 << CS42L42_ASP_5050_SHIFT)
+#define CS42L42_ASP_FSD_SHIFT          0
+#define CS42L42_ASP_FSD_MASK           (7 << CS42L42_ASP_FSD_SHIFT)
+#define CS42L42_ASP_FSD_0_5            1
+#define CS42L42_ASP_FSD_1_0            2
+#define CS42L42_ASP_FSD_1_5            3
+#define CS42L42_ASP_FSD_2_0            4
+
+#define CS42L42_FS_RATE_EN             (CS42L42_PAGE_12 + 0x09)
+#define CS42L42_FS_EN_SHIFT            0
+#define CS42L42_FS_EN_MASK             (0xf << CS42L42_FS_EN_SHIFT)
+#define CS42L42_FS_EN_IASRC_96K                0x1
+#define CS42L42_FS_EN_OASRC_96K                0x2
+
+#define CS42L42_IN_ASRC_CLK            (CS42L42_PAGE_12 + 0x0A)
+#define CS42L42_CLK_IASRC_SEL_SHIFT    0
+#define CS42L42_CLK_IASRC_SEL_MASK     (1 << CS42L42_CLK_IASRC_SEL_SHIFT)
+#define CS42L42_CLK_IASRC_SEL_12       1
+
+#define CS42L42_OUT_ASRC_CLK           (CS42L42_PAGE_12 + 0x0B)
+#define CS42L42_CLK_OASRC_SEL_SHIFT    0
+#define CS42L42_CLK_OASRC_SEL_MASK     (1 << CS42L42_CLK_OASRC_SEL_SHIFT)
+#define CS42L42_CLK_OASRC_SEL_12       1
+
+#define CS42L42_PLL_DIV_CFG1           (CS42L42_PAGE_12 + 0x0C)
+#define CS42L42_SCLK_PREDIV_SHIFT      0
+#define CS42L42_SCLK_PREDIV_MASK       (3 << CS42L42_SCLK_PREDIV_SHIFT)
+
+/* Page 0x13 Interrupt Registers */
+/* Interrupts */
+#define CS42L42_ADC_OVFL_STATUS                (CS42L42_PAGE_13 + 0x01)
+#define CS42L42_MIXER_STATUS           (CS42L42_PAGE_13 + 0x02)
+#define CS42L42_SRC_STATUS             (CS42L42_PAGE_13 + 0x03)
+#define CS42L42_ASP_RX_STATUS          (CS42L42_PAGE_13 + 0x04)
+#define CS42L42_ASP_TX_STATUS          (CS42L42_PAGE_13 + 0x05)
+#define CS42L42_CODEC_STATUS           (CS42L42_PAGE_13 + 0x08)
+#define CS42L42_DET_INT_STATUS1                (CS42L42_PAGE_13 + 0x09)
+#define CS42L42_DET_INT_STATUS2                (CS42L42_PAGE_13 + 0x0A)
+#define CS42L42_SRCPL_INT_STATUS       (CS42L42_PAGE_13 + 0x0B)
+#define CS42L42_VPMON_STATUS           (CS42L42_PAGE_13 + 0x0D)
+#define CS42L42_PLL_LOCK_STATUS                (CS42L42_PAGE_13 + 0x0E)
+#define CS42L42_TSRS_PLUG_STATUS       (CS42L42_PAGE_13 + 0x0F)
+/* Masks */
+#define CS42L42_ADC_OVFL_INT_MASK      (CS42L42_PAGE_13 + 0x16)
+#define CS42L42_ADC_OVFL_SHIFT         0
+#define CS42L42_ADC_OVFL_MASK          (1 << CS42L42_ADC_OVFL_SHIFT)
+#define CS42L42_ADC_OVFL_VAL_MASK      CS42L42_ADC_OVFL_MASK
+
+#define CS42L42_MIXER_INT_MASK         (CS42L42_PAGE_13 + 0x17)
+#define CS42L42_MIX_CHB_OVFL_SHIFT     0
+#define CS42L42_MIX_CHB_OVFL_MASK      (1 << CS42L42_MIX_CHB_OVFL_SHIFT)
+#define CS42L42_MIX_CHA_OVFL_SHIFT     1
+#define CS42L42_MIX_CHA_OVFL_MASK      (1 << CS42L42_MIX_CHA_OVFL_SHIFT)
+#define CS42L42_EQ_OVFL_SHIFT          2
+#define CS42L42_EQ_OVFL_MASK           (1 << CS42L42_EQ_OVFL_SHIFT)
+#define CS42L42_EQ_BIQUAD_OVFL_SHIFT   3
+#define CS42L42_EQ_BIQUAD_OVFL_MASK    (1 << CS42L42_EQ_BIQUAD_OVFL_SHIFT)
+#define CS42L42_MIXER_VAL_MASK         (CS42L42_MIX_CHB_OVFL_MASK | \
+                                       CS42L42_MIX_CHA_OVFL_MASK | \
+                                       CS42L42_EQ_OVFL_MASK | \
+                                       CS42L42_EQ_BIQUAD_OVFL_MASK)
+
+#define CS42L42_SRC_INT_MASK           (CS42L42_PAGE_13 + 0x18)
+#define CS42L42_SRC_ILK_SHIFT          0
+#define CS42L42_SRC_ILK_MASK           (1 << CS42L42_SRC_ILK_SHIFT)
+#define CS42L42_SRC_OLK_SHIFT          1
+#define CS42L42_SRC_OLK_MASK           (1 << CS42L42_SRC_OLK_SHIFT)
+#define CS42L42_SRC_IUNLK_SHIFT                2
+#define CS42L42_SRC_IUNLK_MASK         (1 << CS42L42_SRC_IUNLK_SHIFT)
+#define CS42L42_SRC_OUNLK_SHIFT                3
+#define CS42L42_SRC_OUNLK_MASK         (1 << CS42L42_SRC_OUNLK_SHIFT)
+#define CS42L42_SRC_VAL_MASK           (CS42L42_SRC_ILK_MASK | \
+                                       CS42L42_SRC_OLK_MASK | \
+                                       CS42L42_SRC_IUNLK_MASK | \
+                                       CS42L42_SRC_OUNLK_MASK)
+
+#define CS42L42_ASP_RX_INT_MASK                (CS42L42_PAGE_13 + 0x19)
+#define CS42L42_ASPRX_NOLRCK_SHIFT     0
+#define CS42L42_ASPRX_NOLRCK_MASK      (1 << CS42L42_ASPRX_NOLRCK_SHIFT)
+#define CS42L42_ASPRX_EARLY_SHIFT      1
+#define CS42L42_ASPRX_EARLY_MASK       (1 << CS42L42_ASPRX_EARLY_SHIFT)
+#define CS42L42_ASPRX_LATE_SHIFT       2
+#define CS42L42_ASPRX_LATE_MASK                (1 << CS42L42_ASPRX_LATE_SHIFT)
+#define CS42L42_ASPRX_ERROR_SHIFT      3
+#define CS42L42_ASPRX_ERROR_MASK       (1 << CS42L42_ASPRX_ERROR_SHIFT)
+#define CS42L42_ASPRX_OVLD_SHIFT       4
+#define CS42L42_ASPRX_OVLD_MASK                (1 << CS42L42_ASPRX_OVLD_SHIFT)
+#define CS42L42_ASP_RX_VAL_MASK                (CS42L42_ASPRX_NOLRCK_MASK | \
+                                       CS42L42_ASPRX_EARLY_MASK | \
+                                       CS42L42_ASPRX_LATE_MASK | \
+                                       CS42L42_ASPRX_ERROR_MASK | \
+                                       CS42L42_ASPRX_OVLD_MASK)
+
+#define CS42L42_ASP_TX_INT_MASK                (CS42L42_PAGE_13 + 0x1A)
+#define CS42L42_ASPTX_NOLRCK_SHIFT     0
+#define CS42L42_ASPTX_NOLRCK_MASK      (1 << CS42L42_ASPTX_NOLRCK_SHIFT)
+#define CS42L42_ASPTX_EARLY_SHIFT      1
+#define CS42L42_ASPTX_EARLY_MASK       (1 << CS42L42_ASPTX_EARLY_SHIFT)
+#define CS42L42_ASPTX_LATE_SHIFT       2
+#define CS42L42_ASPTX_LATE_MASK                (1 << CS42L42_ASPTX_LATE_SHIFT)
+#define CS42L42_ASPTX_SMERROR_SHIFT    3
+#define CS42L42_ASPTX_SMERROR_MASK     (1 << CS42L42_ASPTX_SMERROR_SHIFT)
+#define CS42L42_ASP_TX_VAL_MASK                (CS42L42_ASPTX_NOLRCK_MASK | \
+                                       CS42L42_ASPTX_EARLY_MASK | \
+                                       CS42L42_ASPTX_LATE_MASK | \
+                                       CS42L42_ASPTX_SMERROR_MASK)
+
+#define CS42L42_CODEC_INT_MASK         (CS42L42_PAGE_13 + 0x1B)
+#define CS42L42_PDN_DONE_SHIFT         0
+#define CS42L42_PDN_DONE_MASK          (1 << CS42L42_PDN_DONE_SHIFT)
+#define CS42L42_HSDET_AUTO_DONE_SHIFT  1
+#define CS42L42_HSDET_AUTO_DONE_MASK   (1 << CS42L42_HSDET_AUTO_DONE_SHIFT)
+#define CS42L42_CODEC_VAL_MASK         (CS42L42_PDN_DONE_MASK | \
+                                       CS42L42_HSDET_AUTO_DONE_MASK)
+
+#define CS42L42_SRCPL_INT_MASK         (CS42L42_PAGE_13 + 0x1C)
+#define CS42L42_SRCPL_ADC_LK_SHIFT     0
+#define CS42L42_SRCPL_ADC_LK_MASK      (1 << CS42L42_SRCPL_ADC_LK_SHIFT)
+#define CS42L42_SRCPL_DAC_LK_SHIFT     2
+#define CS42L42_SRCPL_DAC_LK_MASK      (1 << CS42L42_SRCPL_DAC_LK_SHIFT)
+#define CS42L42_SRCPL_ADC_UNLK_SHIFT   5
+#define CS42L42_SRCPL_ADC_UNLK_MASK    (1 << CS42L42_SRCPL_ADC_UNLK_SHIFT)
+#define CS42L42_SRCPL_DAC_UNLK_SHIFT   6
+#define CS42L42_SRCPL_DAC_UNLK_MASK    (1 << CS42L42_SRCPL_DAC_UNLK_SHIFT)
+#define CS42L42_SRCPL_VAL_MASK         (CS42L42_SRCPL_ADC_LK_MASK | \
+                                       CS42L42_SRCPL_DAC_LK_MASK | \
+                                       CS42L42_SRCPL_ADC_UNLK_MASK | \
+                                       CS42L42_SRCPL_DAC_UNLK_MASK)
+
+#define CS42L42_VPMON_INT_MASK         (CS42L42_PAGE_13 + 0x1E)
+#define CS42L42_VPMON_SHIFT            0
+#define CS42L42_VPMON_MASK             (1 << CS42L42_VPMON_SHIFT)
+#define CS42L42_VPMON_VAL_MASK         CS42L42_VPMON_MASK
+
+#define CS42L42_PLL_LOCK_INT_MASK      (CS42L42_PAGE_13 + 0x1F)
+#define CS42L42_PLL_LOCK_SHIFT         0
+#define CS42L42_PLL_LOCK_MASK          (1 << CS42L42_PLL_LOCK_SHIFT)
+#define CS42L42_PLL_LOCK_VAL_MASK      CS42L42_PLL_LOCK_MASK
+
+#define CS42L42_TSRS_PLUG_INT_MASK     (CS42L42_PAGE_13 + 0x20)
+#define CS42L42_RS_PLUG_SHIFT          0
+#define CS42L42_RS_PLUG_MASK           (1 << CS42L42_RS_PLUG_SHIFT)
+#define CS42L42_RS_UNPLUG_SHIFT                1
+#define CS42L42_RS_UNPLUG_MASK         (1 << CS42L42_RS_UNPLUG_SHIFT)
+#define CS42L42_TS_PLUG_SHIFT          2
+#define CS42L42_TS_PLUG_MASK           (1 << CS42L42_TS_PLUG_SHIFT)
+#define CS42L42_TS_UNPLUG_SHIFT                3
+#define CS42L42_TS_UNPLUG_MASK         (1 << CS42L42_TS_UNPLUG_SHIFT)
+#define CS42L42_TSRS_PLUG_VAL_MASK     (CS42L42_RS_PLUG_MASK | \
+                                       CS42L42_RS_UNPLUG_MASK | \
+                                       CS42L42_TS_PLUG_MASK | \
+                                       CS42L42_TS_UNPLUG_MASK)
+#define CS42L42_TS_PLUG                        3
+#define CS42L42_TS_UNPLUG              0
+#define CS42L42_TS_TRANS               1
+
+/* Page 0x15 Fractional-N PLL Registers */
+#define CS42L42_PLL_CTL1               (CS42L42_PAGE_15 + 0x01)
+#define CS42L42_PLL_START_SHIFT                0
+#define CS42L42_PLL_START_MASK         (1 << CS42L42_PLL_START_SHIFT)
+
+#define CS42L42_PLL_DIV_FRAC0          (CS42L42_PAGE_15 + 0x02)
+#define CS42L42_PLL_DIV_FRAC_SHIFT     0
+#define CS42L42_PLL_DIV_FRAC_MASK      (0xff << CS42L42_PLL_DIV_FRAC_SHIFT)
+
+#define CS42L42_PLL_DIV_FRAC1          (CS42L42_PAGE_15 + 0x03)
+#define CS42L42_PLL_DIV_FRAC2          (CS42L42_PAGE_15 + 0x04)
+
+#define CS42L42_PLL_DIV_INT            (CS42L42_PAGE_15 + 0x05)
+#define CS42L42_PLL_DIV_INT_SHIFT      0
+#define CS42L42_PLL_DIV_INT_MASK       (0xff << CS42L42_PLL_DIV_INT_SHIFT)
+
+#define CS42L42_PLL_CTL3               (CS42L42_PAGE_15 + 0x08)
+#define CS42L42_PLL_DIVOUT_SHIFT       0
+#define CS42L42_PLL_DIVOUT_MASK                (0xff << CS42L42_PLL_DIVOUT_SHIFT)
+
+#define CS42L42_PLL_CAL_RATIO          (CS42L42_PAGE_15 + 0x0A)
+#define CS42L42_PLL_CAL_RATIO_SHIFT    0
+#define CS42L42_PLL_CAL_RATIO_MASK     (0xff << CS42L42_PLL_CAL_RATIO_SHIFT)
+
+#define CS42L42_PLL_CTL4               (CS42L42_PAGE_15 + 0x1B)
+#define CS42L42_PLL_MODE_SHIFT         0
+#define CS42L42_PLL_MODE_MASK          (3 << CS42L42_PLL_MODE_SHIFT)
+
+/* Page 0x19 HP Load Detect Registers */
+#define CS42L42_LOAD_DET_RCSTAT                (CS42L42_PAGE_19 + 0x25)
+#define CS42L42_RLA_STAT_SHIFT         0
+#define CS42L42_RLA_STAT_MASK          (3 << CS42L42_RLA_STAT_SHIFT)
+#define CS42L42_RLA_STAT_15_OHM                0
+
+#define CS42L42_LOAD_DET_DONE          (CS42L42_PAGE_19 + 0x26)
+#define CS42L42_HPLOAD_DET_DONE_SHIFT  0
+#define CS42L42_HPLOAD_DET_DONE_MASK   (1 << CS42L42_HPLOAD_DET_DONE_SHIFT)
+
+#define CS42L42_LOAD_DET_EN            (CS42L42_PAGE_19 + 0x27)
+#define CS42L42_HP_LD_EN_SHIFT         0
+#define CS42L42_HP_LD_EN_MASK          (1 << CS42L42_HP_LD_EN_SHIFT)
+
+/* Page 0x1B Headset Interface Registers */
+#define CS42L42_HSBIAS_SC_AUTOCTL              (CS42L42_PAGE_1B + 0x70)
+#define CS42L42_HSBIAS_SENSE_TRIP_SHIFT                0
+#define CS42L42_HSBIAS_SENSE_TRIP_MASK         (7 << \
+                                       CS42L42_HSBIAS_SENSE_TRIP_SHIFT)
+#define CS42L42_TIP_SENSE_EN_SHIFT             5
+#define CS42L42_TIP_SENSE_EN_MASK              (1 << \
+                                       CS42L42_TIP_SENSE_EN_SHIFT)
+#define CS42L42_AUTO_HSBIAS_HIZ_SHIFT          6
+#define CS42L42_AUTO_HSBIAS_HIZ_MASK           (1 << \
+                                       CS42L42_AUTO_HSBIAS_HIZ_SHIFT)
+#define CS42L42_HSBIAS_SENSE_EN_SHIFT          7
+#define CS42L42_HSBIAS_SENSE_EN_MASK           (1 << \
+                                       CS42L42_HSBIAS_SENSE_EN_SHIFT)
+
+#define CS42L42_WAKE_CTL               (CS42L42_PAGE_1B + 0x71)
+#define CS42L42_WAKEB_CLEAR_SHIFT      0
+#define CS42L42_WAKEB_CLEAR_MASK       (1 << CS42L42_WAKEB_CLEAR_SHIFT)
+#define CS42L42_WAKEB_MODE_SHIFT       5
+#define CS42L42_WAKEB_MODE_MASK                (1 << CS42L42_WAKEB_MODE_SHIFT)
+#define CS42L42_M_HP_WAKE_SHIFT                6
+#define CS42L42_M_HP_WAKE_MASK         (1 << CS42L42_M_HP_WAKE_SHIFT)
+#define CS42L42_M_MIC_WAKE_SHIFT       7
+#define CS42L42_M_MIC_WAKE_MASK                (1 << CS42L42_M_MIC_WAKE_SHIFT)
+
+#define CS42L42_ADC_DISABLE_MUTE               (CS42L42_PAGE_1B + 0x72)
+#define CS42L42_ADC_DISABLE_S0_MUTE_SHIFT      7
+#define CS42L42_ADC_DISABLE_S0_MUTE_MASK       (1 << \
+                                       CS42L42_ADC_DISABLE_S0_MUTE_SHIFT)
+
+#define CS42L42_TIPSENSE_CTL                   (CS42L42_PAGE_1B + 0x73)
+#define CS42L42_TIP_SENSE_DEBOUNCE_SHIFT       0
+#define CS42L42_TIP_SENSE_DEBOUNCE_MASK                (3 << \
+                                       CS42L42_TIP_SENSE_DEBOUNCE_SHIFT)
+#define CS42L42_TIP_SENSE_INV_SHIFT            5
+#define CS42L42_TIP_SENSE_INV_MASK             (1 << \
+                                       CS42L42_TIP_SENSE_INV_SHIFT)
+#define CS42L42_TIP_SENSE_CTRL_SHIFT           6
+#define CS42L42_TIP_SENSE_CTRL_MASK            (3 << \
+                                       CS42L42_TIP_SENSE_CTRL_SHIFT)
+
+#define CS42L42_MISC_DET_CTL           (CS42L42_PAGE_1B + 0x74)
+#define CS42L42_PDN_MIC_LVL_DET_SHIFT  0
+#define CS42L42_PDN_MIC_LVL_DET_MASK   (1 << CS42L42_PDN_MIC_LVL_DET_SHIFT)
+#define CS42L42_HSBIAS_CTL_SHIFT       1
+#define CS42L42_HSBIAS_CTL_MASK                (3 << CS42L42_HSBIAS_CTL_SHIFT)
+#define CS42L42_DETECT_MODE_SHIFT      3
+#define CS42L42_DETECT_MODE_MASK       (3 << CS42L42_DETECT_MODE_SHIFT)
+
+#define CS42L42_MIC_DET_CTL1           (CS42L42_PAGE_1B + 0x75)
+#define CS42L42_HS_DET_LEVEL_SHIFT     0
+#define CS42L42_HS_DET_LEVEL_MASK      (0x3F << CS42L42_HS_DET_LEVEL_SHIFT)
+#define CS42L42_EVENT_STAT_SEL_SHIFT   6
+#define CS42L42_EVENT_STAT_SEL_MASK    (1 << CS42L42_EVENT_STAT_SEL_SHIFT)
+#define CS42L42_LATCH_TO_VP_SHIFT      7
+#define CS42L42_LATCH_TO_VP_MASK       (1 << CS42L42_LATCH_TO_VP_SHIFT)
+
+#define CS42L42_MIC_DET_CTL2           (CS42L42_PAGE_1B + 0x76)
+#define CS42L42_DEBOUNCE_TIME_SHIFT    5
+#define CS42L42_DEBOUNCE_TIME_MASK     (0x07 << CS42L42_DEBOUNCE_TIME_SHIFT)
+
+#define CS42L42_DET_STATUS1            (CS42L42_PAGE_1B + 0x77)
+#define CS42L42_HSBIAS_HIZ_MODE_SHIFT  6
+#define CS42L42_HSBIAS_HIZ_MODE_MASK   (1 << CS42L42_HSBIAS_HIZ_MODE_SHIFT)
+#define CS42L42_TIP_SENSE_SHIFT                7
+#define CS42L42_TIP_SENSE_MASK         (1 << CS42L42_TIP_SENSE_SHIFT)
+
+#define CS42L42_DET_STATUS2            (CS42L42_PAGE_1B + 0x78)
+#define CS42L42_SHORT_TRUE_SHIFT       0
+#define CS42L42_SHORT_TRUE_MASK                (1 << CS42L42_SHORT_TRUE_SHIFT)
+#define CS42L42_HS_TRUE_SHIFT  1
+#define CS42L42_HS_TRUE_MASK           (1 << CS42L42_HS_TRUE_SHIFT)
+
+#define CS42L42_DET_INT1_MASK          (CS42L42_PAGE_1B + 0x79)
+#define CS42L42_TIP_SENSE_UNPLUG_SHIFT 5
+#define CS42L42_TIP_SENSE_UNPLUG_MASK  (1 << CS42L42_TIP_SENSE_UNPLUG_SHIFT)
+#define CS42L42_TIP_SENSE_PLUG_SHIFT   6
+#define CS42L42_TIP_SENSE_PLUG_MASK    (1 << CS42L42_TIP_SENSE_PLUG_SHIFT)
+#define CS42L42_HSBIAS_SENSE_SHIFT     7
+#define CS42L42_HSBIAS_SENSE_MASK      (1 << CS42L42_HSBIAS_SENSE_SHIFT)
+#define CS42L42_DET_INT_VAL1_MASK      (CS42L42_TIP_SENSE_UNPLUG_MASK | \
+                                       CS42L42_TIP_SENSE_PLUG_MASK | \
+                                       CS42L42_HSBIAS_SENSE_MASK)
+
+#define CS42L42_DET_INT2_MASK          (CS42L42_PAGE_1B + 0x7A)
+#define CS42L42_M_SHORT_DET_SHIFT      0
+#define CS42L42_M_SHORT_DET_MASK       (1 << \
+                                       CS42L42_M_SHORT_DET_SHIFT)
+#define CS42L42_M_SHORT_RLS_SHIFT      1
+#define CS42L42_M_SHORT_RLS_MASK       (1 << \
+                                       CS42L42_M_SHORT_RLS_SHIFT)
+#define CS42L42_M_HSBIAS_HIZ_SHIFT     2
+#define CS42L42_M_HSBIAS_HIZ_MASK      (1 << \
+                                       CS42L42_M_HSBIAS_HIZ_SHIFT)
+#define CS42L42_M_DETECT_FT_SHIFT      6
+#define CS42L42_M_DETECT_FT_MASK       (1 << \
+                                       CS42L42_M_DETECT_FT_SHIFT)
+#define CS42L42_M_DETECT_TF_SHIFT      7
+#define CS42L42_M_DETECT_TF_MASK       (1 << \
+                                       CS42L42_M_DETECT_TF_SHIFT)
+#define CS42L42_DET_INT_VAL2_MASK      (CS42L42_M_SHORT_DET_MASK | \
+                                       CS42L42_M_SHORT_RLS_MASK | \
+                                       CS42L42_M_HSBIAS_HIZ_MASK | \
+                                       CS42L42_M_DETECT_FT_MASK | \
+                                       CS42L42_M_DETECT_TF_MASK)
+
+/* Page 0x1C Headset Bias Registers */
+#define CS42L42_HS_BIAS_CTL            (CS42L42_PAGE_1C + 0x03)
+#define CS42L42_HSBIAS_RAMP_SHIFT      0
+#define CS42L42_HSBIAS_RAMP_MASK       (3 << CS42L42_HSBIAS_RAMP_SHIFT)
+#define CS42L42_HSBIAS_PD_SHIFT                4
+#define CS42L42_HSBIAS_PD_MASK         (1 << CS42L42_HSBIAS_PD_SHIFT)
+#define CS42L42_HSBIAS_CAPLESS_SHIFT   7
+#define CS42L42_HSBIAS_CAPLESS_MASK    (1 << CS42L42_HSBIAS_CAPLESS_SHIFT)
+
+/* Page 0x1D ADC Registers */
+#define CS42L42_ADC_CTL                        (CS42L42_PAGE_1D + 0x01)
+#define CS42L42_ADC_NOTCH_DIS_SHIFT            5
+#define CS42L42_ADC_FORCE_WEAK_VCM_SHIFT       4
+#define CS42L42_ADC_INV_SHIFT                  2
+#define CS42L42_ADC_DIG_BOOST_SHIFT            0
+
+#define CS42L42_ADC_VOLUME             (CS42L42_PAGE_1D + 0x03)
+#define CS42L42_ADC_VOL_SHIFT          0
+
+#define CS42L42_ADC_WNF_HPF_CTL                (CS42L42_PAGE_1D + 0x04)
+#define CS42L42_ADC_WNF_CF_SHIFT       4
+#define CS42L42_ADC_WNF_EN_SHIFT       3
+#define CS42L42_ADC_HPF_CF_SHIFT       1
+#define CS42L42_ADC_HPF_EN_SHIFT       0
+
+/* Page 0x1F DAC Registers */
+#define CS42L42_DAC_CTL1               (CS42L42_PAGE_1F + 0x01)
+#define CS42L42_DACB_INV_SHIFT         1
+#define CS42L42_DACA_INV_SHIFT         0
+
+#define CS42L42_DAC_CTL2               (CS42L42_PAGE_1F + 0x06)
+#define CS42L42_HPOUT_PULLDOWN_SHIFT   4
+#define CS42L42_HPOUT_PULLDOWN_MASK    (15 << CS42L42_HPOUT_PULLDOWN_SHIFT)
+#define CS42L42_HPOUT_LOAD_SHIFT       3
+#define CS42L42_HPOUT_LOAD_MASK                (1 << CS42L42_HPOUT_LOAD_SHIFT)
+#define CS42L42_HPOUT_CLAMP_SHIFT      2
+#define CS42L42_HPOUT_CLAMP_MASK       (1 << CS42L42_HPOUT_CLAMP_SHIFT)
+#define CS42L42_DAC_HPF_EN_SHIFT       1
+#define CS42L42_DAC_HPF_EN_MASK                (1 << CS42L42_DAC_HPF_EN_SHIFT)
+#define CS42L42_DAC_MON_EN_SHIFT       0
+#define CS42L42_DAC_MON_EN_MASK                (1 << CS42L42_DAC_MON_EN_SHIFT)
+
+/* Page 0x20 HP CTL Registers */
+#define CS42L42_HP_CTL                 (CS42L42_PAGE_20 + 0x01)
+#define CS42L42_HP_ANA_BMUTE_SHIFT     3
+#define CS42L42_HP_ANA_BMUTE_MASK      (1 << CS42L42_HP_ANA_BMUTE_SHIFT)
+#define CS42L42_HP_ANA_AMUTE_SHIFT     2
+#define CS42L42_HP_ANA_AMUTE_MASK      (1 << CS42L42_HP_ANA_AMUTE_SHIFT)
+#define CS42L42_HP_FULL_SCALE_VOL_SHIFT        1
+#define CS42L42_HP_FULL_SCALE_VOL_MASK (1 << CS42L42_HP_FULL_SCALE_VOL_SHIFT)
+
+/* Page 0x21 Class H Registers */
+#define CS42L42_CLASSH_CTL             (CS42L42_PAGE_21 + 0x01)
+
+/* Page 0x23 Mixer Volume Registers */
+#define CS42L42_MIXER_CHA_VOL          (CS42L42_PAGE_23 + 0x01)
+#define CS42L42_MIXER_ADC_VOL          (CS42L42_PAGE_23 + 0x02)
+
+#define CS42L42_MIXER_CHB_VOL          (CS42L42_PAGE_23 + 0x03)
+#define CS42L42_MIXER_CH_VOL_SHIFT     0
+#define CS42L42_MIXER_CH_VOL_MASK      (0x3f << CS42L42_MIXER_CH_VOL_SHIFT)
+
+/* Page 0x24 EQ Registers */
+#define CS42L42_EQ_COEF_IN0            (CS42L42_PAGE_24 + 0x01)
+#define CS42L42_EQ_COEF_IN1            (CS42L42_PAGE_24 + 0x02)
+#define CS42L42_EQ_COEF_IN2            (CS42L42_PAGE_24 + 0x03)
+#define CS42L42_EQ_COEF_IN3            (CS42L42_PAGE_24 + 0x04)
+#define CS42L42_EQ_COEF_RW             (CS42L42_PAGE_24 + 0x06)
+#define CS42L42_EQ_COEF_OUT0           (CS42L42_PAGE_24 + 0x07)
+#define CS42L42_EQ_COEF_OUT1           (CS42L42_PAGE_24 + 0x08)
+#define CS42L42_EQ_COEF_OUT2           (CS42L42_PAGE_24 + 0x09)
+#define CS42L42_EQ_COEF_OUT3           (CS42L42_PAGE_24 + 0x0A)
+#define CS42L42_EQ_INIT_STAT           (CS42L42_PAGE_24 + 0x0B)
+#define CS42L42_EQ_START_FILT          (CS42L42_PAGE_24 + 0x0C)
+#define CS42L42_EQ_MUTE_CTL            (CS42L42_PAGE_24 + 0x0E)
+
+/* Page 0x25 Audio Port Registers */
+#define CS42L42_SP_RX_CH_SEL           (CS42L42_PAGE_25 + 0x01)
+
+#define CS42L42_SP_RX_ISOC_CTL         (CS42L42_PAGE_25 + 0x02)
+#define CS42L42_SP_RX_RSYNC_SHIFT      6
+#define CS42L42_SP_RX_RSYNC_MASK       (1 << CS42L42_SP_RX_RSYNC_SHIFT)
+#define CS42L42_SP_RX_NSB_POS_SHIFT    3
+#define CS42L42_SP_RX_NSB_POS_MASK     (7 << CS42L42_SP_RX_NSB_POS_SHIFT)
+#define CS42L42_SP_RX_NFS_NSBB_SHIFT   2
+#define CS42L42_SP_RX_NFS_NSBB_MASK    (1 << CS42L42_SP_RX_NFS_NSBB_SHIFT)
+#define CS42L42_SP_RX_ISOC_MODE_SHIFT  0
+#define CS42L42_SP_RX_ISOC_MODE_MASK   (3 << CS42L42_SP_RX_ISOC_MODE_SHIFT)
+
+#define CS42L42_SP_RX_FS               (CS42L42_PAGE_25 + 0x03)
+#define CS42l42_SPDIF_CH_SEL           (CS42L42_PAGE_25 + 0x04)
+#define CS42L42_SP_TX_ISOC_CTL         (CS42L42_PAGE_25 + 0x05)
+#define CS42L42_SP_TX_FS               (CS42L42_PAGE_25 + 0x06)
+#define CS42L42_SPDIF_SW_CTL1          (CS42L42_PAGE_25 + 0x07)
+
+/* Page 0x26 SRC Registers */
+#define CS42L42_SRC_SDIN_FS            (CS42L42_PAGE_26 + 0x01)
+#define CS42L42_SRC_SDIN_FS_SHIFT      0
+#define CS42L42_SRC_SDIN_FS_MASK       (0x1f << CS42L42_SRC_SDIN_FS_SHIFT)
+
+#define CS42L42_SRC_SDOUT_FS           (CS42L42_PAGE_26 + 0x09)
+
+/* Page 0x28 S/PDIF Registers */
+#define CS42L42_SPDIF_CTL1             (CS42L42_PAGE_28 + 0x01)
+#define CS42L42_SPDIF_CTL2             (CS42L42_PAGE_28 + 0x02)
+#define CS42L42_SPDIF_CTL3             (CS42L42_PAGE_28 + 0x03)
+#define CS42L42_SPDIF_CTL4             (CS42L42_PAGE_28 + 0x04)
+
+/* Page 0x29 Serial Port TX Registers */
+#define CS42L42_ASP_TX_SZ_EN           (CS42L42_PAGE_29 + 0x01)
+#define CS42L42_ASP_TX_CH_EN           (CS42L42_PAGE_29 + 0x02)
+#define CS42L42_ASP_TX_CH_AP_RES       (CS42L42_PAGE_29 + 0x03)
+#define CS42L42_ASP_TX_CH1_BIT_MSB     (CS42L42_PAGE_29 + 0x04)
+#define CS42L42_ASP_TX_CH1_BIT_LSB     (CS42L42_PAGE_29 + 0x05)
+#define CS42L42_ASP_TX_HIZ_DLY_CFG     (CS42L42_PAGE_29 + 0x06)
+#define CS42L42_ASP_TX_CH2_BIT_MSB     (CS42L42_PAGE_29 + 0x0A)
+#define CS42L42_ASP_TX_CH2_BIT_LSB     (CS42L42_PAGE_29 + 0x0B)
+
+/* Page 0x2A Serial Port RX Registers */
+#define CS42L42_ASP_RX_DAI0_EN         (CS42L42_PAGE_2A + 0x01)
+#define CS42L42_ASP_RX0_CH_EN_SHIFT    2
+#define CS42L42_ASP_RX0_CH_EN_MASK     (0xf << CS42L42_ASP_RX0_CH_EN_SHIFT)
+#define CS42L42_ASP_RX0_CH1_EN         1
+#define CS42L42_ASP_RX0_CH2_EN         2
+#define CS42L42_ASP_RX0_CH3_EN         4
+#define CS42L42_ASP_RX0_CH4_EN         8
+
+#define CS42L42_ASP_RX_DAI0_CH1_AP_RES (CS42L42_PAGE_2A + 0x02)
+#define CS42L42_ASP_RX_DAI0_CH1_BIT_MSB        (CS42L42_PAGE_2A + 0x03)
+#define CS42L42_ASP_RX_DAI0_CH1_BIT_LSB        (CS42L42_PAGE_2A + 0x04)
+#define CS42L42_ASP_RX_DAI0_CH2_AP_RES (CS42L42_PAGE_2A + 0x05)
+#define CS42L42_ASP_RX_DAI0_CH2_BIT_MSB        (CS42L42_PAGE_2A + 0x06)
+#define CS42L42_ASP_RX_DAI0_CH2_BIT_LSB        (CS42L42_PAGE_2A + 0x07)
+#define CS42L42_ASP_RX_DAI0_CH3_AP_RES (CS42L42_PAGE_2A + 0x08)
+#define CS42L42_ASP_RX_DAI0_CH3_BIT_MSB        (CS42L42_PAGE_2A + 0x09)
+#define CS42L42_ASP_RX_DAI0_CH3_BIT_LSB        (CS42L42_PAGE_2A + 0x0A)
+#define CS42L42_ASP_RX_DAI0_CH4_AP_RES (CS42L42_PAGE_2A + 0x0B)
+#define CS42L42_ASP_RX_DAI0_CH4_BIT_MSB        (CS42L42_PAGE_2A + 0x0C)
+#define CS42L42_ASP_RX_DAI0_CH4_BIT_LSB        (CS42L42_PAGE_2A + 0x0D)
+#define CS42L42_ASP_RX_DAI1_CH1_AP_RES (CS42L42_PAGE_2A + 0x0E)
+#define CS42L42_ASP_RX_DAI1_CH1_BIT_MSB        (CS42L42_PAGE_2A + 0x0F)
+#define CS42L42_ASP_RX_DAI1_CH1_BIT_LSB        (CS42L42_PAGE_2A + 0x10)
+#define CS42L42_ASP_RX_DAI1_CH2_AP_RES (CS42L42_PAGE_2A + 0x11)
+#define CS42L42_ASP_RX_DAI1_CH2_BIT_MSB        (CS42L42_PAGE_2A + 0x12)
+#define CS42L42_ASP_RX_DAI1_CH2_BIT_LSB        (CS42L42_PAGE_2A + 0x13)
+
+#define CS42L42_ASP_RX_CH_AP_SHIFT     6
+#define CS42L42_ASP_RX_CH_AP_MASK      (1 << CS42L42_ASP_RX_CH_AP_SHIFT)
+#define CS42L42_ASP_RX_CH_AP_LOW       0
+#define CS42L42_ASP_RX_CH_AP_HI                1
+#define CS42L42_ASP_RX_CH_RES_SHIFT    0
+#define CS42L42_ASP_RX_CH_RES_MASK     (3 << CS42L42_ASP_RX_CH_RES_SHIFT)
+#define CS42L42_ASP_RX_CH_RES_32       3
+#define CS42L42_ASP_RX_CH_RES_16       1
+#define CS42L42_ASP_RX_CH_BIT_ST_SHIFT 0
+#define CS42L42_ASP_RX_CH_BIT_ST_MASK  (0xff << CS42L42_ASP_RX_CH_BIT_ST_SHIFT)
+
+/* Page 0x30 ID Registers */
+#define CS42L42_SUB_REVID              (CS42L42_PAGE_30 + 0x14)
+#define CS42L42_MAX_REGISTER           (CS42L42_PAGE_30 + 0x14)
+
+/* Defines for fracturing values spread across multiple registers */
+#define CS42L42_FRAC0_VAL(val) ((val) & 0x0000ff)
+#define CS42L42_FRAC1_VAL(val) (((val) & 0x00ff00) >> 8)
+#define CS42L42_FRAC2_VAL(val) (((val) & 0xff0000) >> 16)
+
+#define CS42L42_NUM_SUPPLIES   5
+
+static const char *const cs42l42_supply_names[CS42L42_NUM_SUPPLIES] = {
+       "VA",
+       "VP",
+       "VCP",
+       "VD_FILT",
+       "VL",
+};
+
+struct  cs42l42_private {
+       struct regmap *regmap;
+       struct snd_soc_codec *codec;
+       struct regulator_bulk_data supplies[CS42L42_NUM_SUPPLIES];
+       struct gpio_desc *reset_gpio;
+       struct completion pdn_done;
+       u32 sclk;
+       u32 srate;
+       u32 swidth;
+       u8 plug_state;
+       u8 hs_type;
+       u8 ts_inv;
+       u8 ts_dbnc_rise;
+       u8 ts_dbnc_fall;
+       u8 btn_det_init_dbnce;
+       u8 btn_det_event_dbnce;
+       u8 bias_thresholds[CS42L42_NUM_BIASES];
+       u8 hs_bias_ramp_rate;
+       u8 hs_bias_ramp_time;
+};
+
+#endif /* __CS42L42_H__ */