return 0;
 }
 
+static int cs35l45_activate_ctl(struct snd_soc_component *component,
+                               const char *ctl_name, bool active)
+{
+       struct snd_card *card = component->card->snd_card;
+       struct snd_kcontrol *kcontrol;
+       struct snd_kcontrol_volatile *vd;
+       unsigned int index_offset;
+       char name[SNDRV_CTL_ELEM_ID_NAME_MAXLEN];
+
+       if (component->name_prefix)
+               snprintf(name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN, "%s %s",
+                        component->name_prefix, ctl_name);
+       else
+               snprintf(name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN, "%s", ctl_name);
+
+       kcontrol = snd_soc_card_get_kcontrol(component->card, name);
+       if (!kcontrol) {
+               dev_err(component->dev, "Can't find kcontrol %s\n", name);
+               return -EINVAL;
+       }
+
+       index_offset = snd_ctl_get_ioff(kcontrol, &kcontrol->id);
+       vd = &kcontrol->vd[index_offset];
+       if (active)
+               vd->access |= SNDRV_CTL_ELEM_ACCESS_WRITE;
+       else
+               vd->access &= ~SNDRV_CTL_ELEM_ACCESS_WRITE;
+
+       snd_ctl_notify(card, SNDRV_CTL_EVENT_MASK_INFO, &kcontrol->id);
+
+       return 0;
+}
+
+static int cs35l45_amplifier_mode_get(struct snd_kcontrol *kcontrol,
+                                     struct snd_ctl_elem_value *ucontrol)
+{
+       struct snd_soc_component *component =
+                       snd_soc_kcontrol_component(kcontrol);
+       struct cs35l45_private *cs35l45 =
+                       snd_soc_component_get_drvdata(component);
+
+       ucontrol->value.integer.value[0] = cs35l45->amplifier_mode;
+
+       return 0;
+}
+
+static int cs35l45_amplifier_mode_put(struct snd_kcontrol *kcontrol,
+                                     struct snd_ctl_elem_value *ucontrol)
+{
+       struct snd_soc_component *component =
+                       snd_soc_kcontrol_component(kcontrol);
+       struct cs35l45_private *cs35l45 =
+                       snd_soc_component_get_drvdata(component);
+       struct snd_soc_dapm_context *dapm =
+                       snd_soc_component_get_dapm(component);
+       unsigned int amp_state;
+       int ret;
+
+       if ((ucontrol->value.integer.value[0] == cs35l45->amplifier_mode) ||
+           (ucontrol->value.integer.value[0] > AMP_MODE_RCV))
+               return 0;
+
+       snd_soc_dapm_mutex_lock(dapm);
+
+       ret = regmap_read(cs35l45->regmap, CS35L45_BLOCK_ENABLES, &_state);
+       if (ret < 0) {
+               dev_err(cs35l45->dev, "Failed to read AMP state: %d\n", ret);
+               snd_soc_dapm_mutex_unlock(dapm);
+               return ret;
+       }
+
+       regmap_clear_bits(cs35l45->regmap, CS35L45_BLOCK_ENABLES,
+                                 CS35L45_AMP_EN_MASK);
+       snd_soc_component_disable_pin_unlocked(component, "SPK");
+       snd_soc_dapm_sync_unlocked(dapm);
+
+       if (ucontrol->value.integer.value[0] == AMP_MODE_SPK) {
+               regmap_clear_bits(cs35l45->regmap, CS35L45_BLOCK_ENABLES,
+                                 CS35L45_RCV_EN_MASK);
+
+               regmap_update_bits(cs35l45->regmap, CS35L45_BLOCK_ENABLES,
+                                  CS35L45_BST_EN_MASK,
+                                  CS35L45_BST_ENABLE << CS35L45_BST_EN_SHIFT);
+
+               regmap_update_bits(cs35l45->regmap, CS35L45_HVLV_CONFIG,
+                                  CS35L45_HVLV_MODE_MASK,
+                                  CS35L45_HVLV_OPERATION <<
+                                  CS35L45_HVLV_MODE_SHIFT);
+
+               ret = cs35l45_activate_ctl(component, "Analog PCM Volume", true);
+               if (ret < 0)
+                       dev_err(cs35l45->dev,
+                               "Unable to deactivate ctl (%d)\n", ret);
+
+       } else  /* AMP_MODE_RCV */ {
+               regmap_set_bits(cs35l45->regmap, CS35L45_BLOCK_ENABLES,
+                               CS35L45_RCV_EN_MASK);
+
+               regmap_update_bits(cs35l45->regmap, CS35L45_BLOCK_ENABLES,
+                                  CS35L45_BST_EN_MASK,
+                                  CS35L45_BST_DISABLE_FET_OFF <<
+                                  CS35L45_BST_EN_SHIFT);
+
+               regmap_update_bits(cs35l45->regmap, CS35L45_HVLV_CONFIG,
+                                  CS35L45_HVLV_MODE_MASK,
+                                  CS35L45_FORCE_LV_OPERATION <<
+                                  CS35L45_HVLV_MODE_SHIFT);
+
+               regmap_clear_bits(cs35l45->regmap,
+                                 CS35L45_BLOCK_ENABLES2,
+                                 CS35L45_AMP_DRE_EN_MASK);
+
+               regmap_update_bits(cs35l45->regmap, CS35L45_AMP_GAIN,
+                                  CS35L45_AMP_GAIN_PCM_MASK,
+                                  CS35L45_AMP_GAIN_PCM_13DBV <<
+                                  CS35L45_AMP_GAIN_PCM_SHIFT);
+
+               ret = cs35l45_activate_ctl(component, "Analog PCM Volume", false);
+               if (ret < 0)
+                       dev_err(cs35l45->dev,
+                               "Unable to deactivate ctl (%d)\n", ret);
+       }
+
+       if (amp_state & CS35L45_AMP_EN_MASK)
+               regmap_set_bits(cs35l45->regmap, CS35L45_BLOCK_ENABLES,
+                               CS35L45_AMP_EN_MASK);
+
+       snd_soc_component_enable_pin_unlocked(component, "SPK");
+       snd_soc_dapm_sync_unlocked(dapm);
+       snd_soc_dapm_mutex_unlock(dapm);
+
+       cs35l45->amplifier_mode = ucontrol->value.integer.value[0];
+
+       return 1;
+}
+
 static const char * const cs35l45_asp_tx_txt[] = {
        "Zero", "ASP_RX1", "ASP_RX2",
        "VMON", "IMON", "ERR_VOL",
        { "SPK", NULL, "AMP"},
 };
 
+static const char * const amplifier_mode_texts[] = {"SPK", "RCV"};
+static SOC_ENUM_SINGLE_DECL(amplifier_mode_enum, SND_SOC_NOPM, 0,
+                           amplifier_mode_texts);
+static DECLARE_TLV_DB_SCALE(amp_gain_tlv, 1000, 300, 0);
 static const DECLARE_TLV_DB_SCALE(cs35l45_dig_pcm_vol_tlv, -10225, 25, true);
 
 static const struct snd_kcontrol_new cs35l45_controls[] = {
+       SOC_ENUM_EXT("Amplifier Mode", amplifier_mode_enum,
+                    cs35l45_amplifier_mode_get, cs35l45_amplifier_mode_put),
+       SOC_SINGLE_TLV("Analog PCM Volume", CS35L45_AMP_GAIN,
+                       CS35L45_AMP_GAIN_PCM_SHIFT,
+                       CS35L45_AMP_GAIN_PCM_MASK >> CS35L45_AMP_GAIN_PCM_SHIFT,
+                       0, amp_gain_tlv),
        /* Ignore bit 0: it is beyond the resolution of TLV_DB_SCALE */
        SOC_SINGLE_S_TLV("Digital PCM Volume",
                         CS35L45_AMP_PCM_CONTROL,
        if (ret < 0)
                return ret;
 
+       cs35l45->amplifier_mode = AMP_MODE_SPK;
+
        return 0;
 }
 
 
 #define CS35L45_DSP1RX6_INPUT                  0x00004C54
 #define CS35L45_DSP1RX7_INPUT                  0x00004C58
 #define CS35L45_DSP1RX8_INPUT                  0x00004C5C
+#define CS35L45_HVLV_CONFIG                    0x00006400
 #define CS35L45_LDPM_CONFIG                    0x00006404
 #define CS35L45_AMP_PCM_CONTROL                        0x00007000
 #define CS35L45_AMP_PCM_HPF_TST                        0x00007004
+#define CS35L45_AMP_GAIN                       0x00007800
 #define CS35L45_IRQ1_CFG                       0x0000E000
 #define CS35L45_IRQ1_STATUS                    0x0000E004
 #define CS35L45_IRQ1_EINT_1                    0x0000E010
 #define CS35L45_VDD_BATTMON_EN_SHIFT           8
 #define CS35L45_BST_EN_SHIFT                   4
 #define CS35L45_BST_EN_MASK                    GENMASK(5, 4)
+#define CS35L45_RCV_EN_SHIFT                   2
+#define CS35L45_RCV_EN_MASK                    BIT(2)
+#define CS35L45_AMP_EN_SHIFT                   0
+#define CS35L45_AMP_EN_MASK                    BIT(0)
 
-#define CS35L45_BST_DISABLE_FET_ON              0x01
+#define CS35L45_BST_DISABLE_FET_OFF            0x00
+#define CS35L45_BST_DISABLE_FET_ON             0x01
+#define CS35L45_BST_ENABLE                     0x02
 
 /* BLOCK_ENABLES2 */
 #define CS35L45_ASP_EN_SHIFT                   27
-
+#define CS35L45_AMP_DRE_EN_SHIFT               20
+#define CS35L45_AMP_DRE_EN_MASK                BIT(20)
 #define CS35L45_MEM_RDY_SHIFT                  1
 #define CS35L45_MEM_RDY_MASK                   BIT(1)
 
 #define CS35L45_ASP_WL_SHIFT                   0
 #define CS35L45_ASP_WL_MASK                    GENMASK(5, 0)
 
+/* HVLV_CONFIG */
+#define CS35L45_FORCE_LV_OPERATION             0x01
+#define CS35L45_FORCE_HV_OPERATION             0x02
+#define CS35L45_HVLV_OPERATION                 0x03
+#define CS35L45_HVLV_MODE_SHIFT                0
+#define CS35L45_HVLV_MODE_MASK                 GENMASK(1, 0)
+
 /* AMP_PCM_CONTROL */
 #define CS35L45_AMP_VOL_PCM_SHIFT              0
 #define CS35L45_AMP_VOL_PCM_WIDTH              11
 #define CS35L45_HPF_44P1                       0x000108BD
 #define CS35L45_HPF_88P2                       0x0001045F
 
+/* AMP_GAIN_PCM */
+#define CS35L45_AMP_GAIN_PCM_10DBV             0x00
+#define CS35L45_AMP_GAIN_PCM_13DBV             0x01
+#define CS35L45_AMP_GAIN_PCM_16DBV             0x02
+#define CS35L45_AMP_GAIN_PCM_19DBV             0x03
+
+#define CS35L45_AMP_GAIN_PCM_SHIFT             8
+#define CS35L45_AMP_GAIN_PCM_MASK              GENMASK(9, 8)
+
 /* IRQ1_EINT_4 */
 #define CS35L45_OTP_BOOT_DONE_STS_MASK         BIT(1)
 #define CS35L45_OTP_BUSY_MASK                  BIT(0)
        CONTROL_BUS_SPI = 1,
 };
 
+enum amp_mode {
+       AMP_MODE_SPK  = 0,
+       AMP_MODE_RCV  = 1,
+};
+
 #define CS35L45_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | \
                         SNDRV_PCM_FMTBIT_S24_3LE| \
                         SNDRV_PCM_FMTBIT_S24_LE)
        bool sysclk_set;
        u8 slot_width;
        u8 slot_count;
+       int amplifier_mode;
        int irq_invert;
        int irq;
        unsigned int i2c_addr;