|  | // SPDX-License-Identifier: GPL-2.0+ | 
|  | /* | 
|  | * This driver supports the analog controls for the internal codec | 
|  | * found in Allwinner's A64 SoC. | 
|  | * | 
|  | * Copyright (C) 2016 Chen-Yu Tsai <wens@csie.org> | 
|  | * Copyright (C) 2017 Marcus Cooper <codekipper@gmail.com> | 
|  | * Copyright (C) 2018 Vasily Khoruzhick <anarsoul@gmail.com> | 
|  | * | 
|  | * Based on sun8i-codec-analog.c | 
|  | * | 
|  | */ | 
|  |  | 
|  | #include <linux/io.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/mod_devicetable.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/platform_device.h> | 
|  | #include <linux/regmap.h> | 
|  |  | 
|  | #include <sound/soc.h> | 
|  | #include <sound/soc-dapm.h> | 
|  | #include <sound/tlv.h> | 
|  |  | 
|  | #include "sun8i-adda-pr-regmap.h" | 
|  |  | 
|  | /* Codec analog control register offsets and bit fields */ | 
|  | #define SUN50I_ADDA_HP_CTRL		0x00 | 
|  | #define SUN50I_ADDA_HP_CTRL_PA_CLK_GATE		7 | 
|  | #define SUN50I_ADDA_HP_CTRL_HPPA_EN		6 | 
|  | #define SUN50I_ADDA_HP_CTRL_HPVOL		0 | 
|  |  | 
|  | #define SUN50I_ADDA_OL_MIX_CTRL		0x01 | 
|  | #define SUN50I_ADDA_OL_MIX_CTRL_MIC1		6 | 
|  | #define SUN50I_ADDA_OL_MIX_CTRL_MIC2		5 | 
|  | #define SUN50I_ADDA_OL_MIX_CTRL_PHONE		4 | 
|  | #define SUN50I_ADDA_OL_MIX_CTRL_PHONEN		3 | 
|  | #define SUN50I_ADDA_OL_MIX_CTRL_LINEINL		2 | 
|  | #define SUN50I_ADDA_OL_MIX_CTRL_DACL		1 | 
|  | #define SUN50I_ADDA_OL_MIX_CTRL_DACR		0 | 
|  |  | 
|  | #define SUN50I_ADDA_OR_MIX_CTRL		0x02 | 
|  | #define SUN50I_ADDA_OR_MIX_CTRL_MIC1		6 | 
|  | #define SUN50I_ADDA_OR_MIX_CTRL_MIC2		5 | 
|  | #define SUN50I_ADDA_OR_MIX_CTRL_PHONE		4 | 
|  | #define SUN50I_ADDA_OR_MIX_CTRL_PHONEP		3 | 
|  | #define SUN50I_ADDA_OR_MIX_CTRL_LINEINR		2 | 
|  | #define SUN50I_ADDA_OR_MIX_CTRL_DACR		1 | 
|  | #define SUN50I_ADDA_OR_MIX_CTRL_DACL		0 | 
|  |  | 
|  | #define SUN50I_ADDA_EARPIECE_CTRL0	0x03 | 
|  | #define SUN50I_ADDA_EARPIECE_CTRL0_EAR_RAMP_TIME	4 | 
|  | #define SUN50I_ADDA_EARPIECE_CTRL0_ESPSR		0 | 
|  |  | 
|  | #define SUN50I_ADDA_EARPIECE_CTRL1	0x04 | 
|  | #define SUN50I_ADDA_EARPIECE_CTRL1_ESPPA_EN	7 | 
|  | #define SUN50I_ADDA_EARPIECE_CTRL1_ESPPA_MUTE	6 | 
|  | #define SUN50I_ADDA_EARPIECE_CTRL1_ESP_VOL	0 | 
|  |  | 
|  | #define SUN50I_ADDA_LINEOUT_CTRL0	0x05 | 
|  | #define SUN50I_ADDA_LINEOUT_CTRL0_LEN		7 | 
|  | #define SUN50I_ADDA_LINEOUT_CTRL0_REN		6 | 
|  | #define SUN50I_ADDA_LINEOUT_CTRL0_LSRC_SEL	5 | 
|  | #define SUN50I_ADDA_LINEOUT_CTRL0_RSRC_SEL	4 | 
|  |  | 
|  | #define SUN50I_ADDA_LINEOUT_CTRL1	0x06 | 
|  | #define SUN50I_ADDA_LINEOUT_CTRL1_VOL		0 | 
|  |  | 
|  | #define SUN50I_ADDA_MIC1_CTRL		0x07 | 
|  | #define SUN50I_ADDA_MIC1_CTRL_MIC1G		4 | 
|  | #define SUN50I_ADDA_MIC1_CTRL_MIC1AMPEN		3 | 
|  | #define SUN50I_ADDA_MIC1_CTRL_MIC1BOOST		0 | 
|  |  | 
|  | #define SUN50I_ADDA_MIC2_CTRL		0x08 | 
|  | #define SUN50I_ADDA_MIC2_CTRL_MIC2G		4 | 
|  | #define SUN50I_ADDA_MIC2_CTRL_MIC2AMPEN		3 | 
|  | #define SUN50I_ADDA_MIC2_CTRL_MIC2BOOST		0 | 
|  |  | 
|  | #define SUN50I_ADDA_LINEIN_CTRL		0x09 | 
|  | #define SUN50I_ADDA_LINEIN_CTRL_LINEING		0 | 
|  |  | 
|  | #define SUN50I_ADDA_MIX_DAC_CTRL	0x0a | 
|  | #define SUN50I_ADDA_MIX_DAC_CTRL_DACAREN	7 | 
|  | #define SUN50I_ADDA_MIX_DAC_CTRL_DACALEN	6 | 
|  | #define SUN50I_ADDA_MIX_DAC_CTRL_RMIXEN		5 | 
|  | #define SUN50I_ADDA_MIX_DAC_CTRL_LMIXEN		4 | 
|  | #define SUN50I_ADDA_MIX_DAC_CTRL_RHPPAMUTE	3 | 
|  | #define SUN50I_ADDA_MIX_DAC_CTRL_LHPPAMUTE	2 | 
|  | #define SUN50I_ADDA_MIX_DAC_CTRL_RHPIS		1 | 
|  | #define SUN50I_ADDA_MIX_DAC_CTRL_LHPIS		0 | 
|  |  | 
|  | #define SUN50I_ADDA_L_ADCMIX_SRC	0x0b | 
|  | #define SUN50I_ADDA_L_ADCMIX_SRC_MIC1		6 | 
|  | #define SUN50I_ADDA_L_ADCMIX_SRC_MIC2		5 | 
|  | #define SUN50I_ADDA_L_ADCMIX_SRC_PHONE		4 | 
|  | #define SUN50I_ADDA_L_ADCMIX_SRC_PHONEN		3 | 
|  | #define SUN50I_ADDA_L_ADCMIX_SRC_LINEINL	2 | 
|  | #define SUN50I_ADDA_L_ADCMIX_SRC_OMIXRL		1 | 
|  | #define SUN50I_ADDA_L_ADCMIX_SRC_OMIXRR		0 | 
|  |  | 
|  | #define SUN50I_ADDA_R_ADCMIX_SRC	0x0c | 
|  | #define SUN50I_ADDA_R_ADCMIX_SRC_MIC1		6 | 
|  | #define SUN50I_ADDA_R_ADCMIX_SRC_MIC2		5 | 
|  | #define SUN50I_ADDA_R_ADCMIX_SRC_PHONE		4 | 
|  | #define SUN50I_ADDA_R_ADCMIX_SRC_PHONEP		3 | 
|  | #define SUN50I_ADDA_R_ADCMIX_SRC_LINEINR	2 | 
|  | #define SUN50I_ADDA_R_ADCMIX_SRC_OMIXR		1 | 
|  | #define SUN50I_ADDA_R_ADCMIX_SRC_OMIXL		0 | 
|  |  | 
|  | #define SUN50I_ADDA_ADC_CTRL		0x0d | 
|  | #define SUN50I_ADDA_ADC_CTRL_ADCREN		7 | 
|  | #define SUN50I_ADDA_ADC_CTRL_ADCLEN		6 | 
|  | #define SUN50I_ADDA_ADC_CTRL_ADCG		0 | 
|  |  | 
|  | #define SUN50I_ADDA_HS_MBIAS_CTRL	0x0e | 
|  | #define SUN50I_ADDA_HS_MBIAS_CTRL_MMICBIASEN	7 | 
|  |  | 
|  | #define SUN50I_ADDA_MDET_CTRL		0x1c | 
|  | #define SUN50I_ADDA_MDET_CTRL_SELDETADC_FS	4 | 
|  | #define SUN50I_ADDA_MDET_CTRL_SELDETADC_DB	2 | 
|  | #define SUN50I_ADDA_MDET_CTRL_SELDETADC_BF	0 | 
|  |  | 
|  | #define SUN50I_ADDA_JACK_MIC_CTRL	0x1d | 
|  | #define SUN50I_ADDA_JACK_MIC_CTRL_JACKDETEN	7 | 
|  | #define SUN50I_ADDA_JACK_MIC_CTRL_INNERRESEN	6 | 
|  | #define SUN50I_ADDA_JACK_MIC_CTRL_HMICBIASEN	5 | 
|  | #define SUN50I_ADDA_JACK_MIC_CTRL_MICADCEN	4 | 
|  |  | 
|  | /* mixer controls */ | 
|  | static const struct snd_kcontrol_new sun50i_a64_codec_mixer_controls[] = { | 
|  | SOC_DAPM_DOUBLE_R("Mic1 Playback Switch", | 
|  | SUN50I_ADDA_OL_MIX_CTRL, | 
|  | SUN50I_ADDA_OR_MIX_CTRL, | 
|  | SUN50I_ADDA_OL_MIX_CTRL_MIC1, 1, 0), | 
|  | SOC_DAPM_DOUBLE_R("Mic2 Playback Switch", | 
|  | SUN50I_ADDA_OL_MIX_CTRL, | 
|  | SUN50I_ADDA_OR_MIX_CTRL, | 
|  | SUN50I_ADDA_OL_MIX_CTRL_MIC2, 1, 0), | 
|  | SOC_DAPM_DOUBLE_R("Line In Playback Switch", | 
|  | SUN50I_ADDA_OL_MIX_CTRL, | 
|  | SUN50I_ADDA_OR_MIX_CTRL, | 
|  | SUN50I_ADDA_OL_MIX_CTRL_LINEINL, 1, 0), | 
|  | SOC_DAPM_DOUBLE_R("DAC Playback Switch", | 
|  | SUN50I_ADDA_OL_MIX_CTRL, | 
|  | SUN50I_ADDA_OR_MIX_CTRL, | 
|  | SUN50I_ADDA_OL_MIX_CTRL_DACL, 1, 0), | 
|  | SOC_DAPM_DOUBLE_R("DAC Reversed Playback Switch", | 
|  | SUN50I_ADDA_OL_MIX_CTRL, | 
|  | SUN50I_ADDA_OR_MIX_CTRL, | 
|  | SUN50I_ADDA_OL_MIX_CTRL_DACR, 1, 0), | 
|  | }; | 
|  |  | 
|  | /* ADC mixer controls */ | 
|  | static const struct snd_kcontrol_new sun50i_codec_adc_mixer_controls[] = { | 
|  | SOC_DAPM_DOUBLE_R("Mic1 Capture Switch", | 
|  | SUN50I_ADDA_L_ADCMIX_SRC, | 
|  | SUN50I_ADDA_R_ADCMIX_SRC, | 
|  | SUN50I_ADDA_L_ADCMIX_SRC_MIC1, 1, 0), | 
|  | SOC_DAPM_DOUBLE_R("Mic2 Capture Switch", | 
|  | SUN50I_ADDA_L_ADCMIX_SRC, | 
|  | SUN50I_ADDA_R_ADCMIX_SRC, | 
|  | SUN50I_ADDA_L_ADCMIX_SRC_MIC2, 1, 0), | 
|  | SOC_DAPM_DOUBLE_R("Line In Capture Switch", | 
|  | SUN50I_ADDA_L_ADCMIX_SRC, | 
|  | SUN50I_ADDA_R_ADCMIX_SRC, | 
|  | SUN50I_ADDA_L_ADCMIX_SRC_LINEINL, 1, 0), | 
|  | SOC_DAPM_DOUBLE_R("Mixer Capture Switch", | 
|  | SUN50I_ADDA_L_ADCMIX_SRC, | 
|  | SUN50I_ADDA_R_ADCMIX_SRC, | 
|  | SUN50I_ADDA_L_ADCMIX_SRC_OMIXRL, 1, 0), | 
|  | SOC_DAPM_DOUBLE_R("Mixer Reversed Capture Switch", | 
|  | SUN50I_ADDA_L_ADCMIX_SRC, | 
|  | SUN50I_ADDA_R_ADCMIX_SRC, | 
|  | SUN50I_ADDA_L_ADCMIX_SRC_OMIXRR, 1, 0), | 
|  | }; | 
|  |  | 
|  | static const DECLARE_TLV_DB_SCALE(sun50i_codec_out_mixer_pregain_scale, | 
|  | -450, 150, 0); | 
|  | static const DECLARE_TLV_DB_RANGE(sun50i_codec_mic_gain_scale, | 
|  | 0, 0, TLV_DB_SCALE_ITEM(0, 0, 0), | 
|  | 1, 7, TLV_DB_SCALE_ITEM(2400, 300, 0), | 
|  | ); | 
|  |  | 
|  | static const DECLARE_TLV_DB_SCALE(sun50i_codec_hp_vol_scale, -6300, 100, 1); | 
|  |  | 
|  | static const DECLARE_TLV_DB_RANGE(sun50i_codec_lineout_vol_scale, | 
|  | 0, 1, TLV_DB_SCALE_ITEM(TLV_DB_GAIN_MUTE, 0, 1), | 
|  | 2, 31, TLV_DB_SCALE_ITEM(-4350, 150, 0), | 
|  | ); | 
|  |  | 
|  | static const DECLARE_TLV_DB_RANGE(sun50i_codec_earpiece_vol_scale, | 
|  | 0, 1, TLV_DB_SCALE_ITEM(TLV_DB_GAIN_MUTE, 0, 1), | 
|  | 2, 31, TLV_DB_SCALE_ITEM(-4350, 150, 0), | 
|  | ); | 
|  |  | 
|  | /* volume / mute controls */ | 
|  | static const struct snd_kcontrol_new sun50i_a64_codec_controls[] = { | 
|  | SOC_SINGLE_TLV("Headphone Playback Volume", | 
|  | SUN50I_ADDA_HP_CTRL, | 
|  | SUN50I_ADDA_HP_CTRL_HPVOL, 0x3f, 0, | 
|  | sun50i_codec_hp_vol_scale), | 
|  |  | 
|  | /* Mixer pre-gain */ | 
|  | SOC_SINGLE_TLV("Mic1 Playback Volume", SUN50I_ADDA_MIC1_CTRL, | 
|  | SUN50I_ADDA_MIC1_CTRL_MIC1G, | 
|  | 0x7, 0, sun50i_codec_out_mixer_pregain_scale), | 
|  |  | 
|  | /* Microphone Amp boost gain */ | 
|  | SOC_SINGLE_TLV("Mic1 Boost Volume", SUN50I_ADDA_MIC1_CTRL, | 
|  | SUN50I_ADDA_MIC1_CTRL_MIC1BOOST, 0x7, 0, | 
|  | sun50i_codec_mic_gain_scale), | 
|  |  | 
|  | /* Mixer pre-gain */ | 
|  | SOC_SINGLE_TLV("Mic2 Playback Volume", | 
|  | SUN50I_ADDA_MIC2_CTRL, SUN50I_ADDA_MIC2_CTRL_MIC2G, | 
|  | 0x7, 0, sun50i_codec_out_mixer_pregain_scale), | 
|  |  | 
|  | /* Microphone Amp boost gain */ | 
|  | SOC_SINGLE_TLV("Mic2 Boost Volume", SUN50I_ADDA_MIC2_CTRL, | 
|  | SUN50I_ADDA_MIC2_CTRL_MIC2BOOST, 0x7, 0, | 
|  | sun50i_codec_mic_gain_scale), | 
|  |  | 
|  | /* ADC */ | 
|  | SOC_SINGLE_TLV("ADC Gain Capture Volume", SUN50I_ADDA_ADC_CTRL, | 
|  | SUN50I_ADDA_ADC_CTRL_ADCG, 0x7, 0, | 
|  | sun50i_codec_out_mixer_pregain_scale), | 
|  |  | 
|  | /* Mixer pre-gain */ | 
|  | SOC_SINGLE_TLV("Line In Playback Volume", SUN50I_ADDA_LINEIN_CTRL, | 
|  | SUN50I_ADDA_LINEIN_CTRL_LINEING, | 
|  | 0x7, 0, sun50i_codec_out_mixer_pregain_scale), | 
|  |  | 
|  | SOC_SINGLE_TLV("Line Out Playback Volume", | 
|  | SUN50I_ADDA_LINEOUT_CTRL1, | 
|  | SUN50I_ADDA_LINEOUT_CTRL1_VOL, 0x1f, 0, | 
|  | sun50i_codec_lineout_vol_scale), | 
|  |  | 
|  | SOC_SINGLE_TLV("Earpiece Playback Volume", | 
|  | SUN50I_ADDA_EARPIECE_CTRL1, | 
|  | SUN50I_ADDA_EARPIECE_CTRL1_ESP_VOL, 0x1f, 0, | 
|  | sun50i_codec_earpiece_vol_scale), | 
|  | }; | 
|  |  | 
|  | static const char * const sun50i_codec_hp_src_enum_text[] = { | 
|  | "DAC", "Mixer", | 
|  | }; | 
|  |  | 
|  | static SOC_ENUM_DOUBLE_DECL(sun50i_codec_hp_src_enum, | 
|  | SUN50I_ADDA_MIX_DAC_CTRL, | 
|  | SUN50I_ADDA_MIX_DAC_CTRL_LHPIS, | 
|  | SUN50I_ADDA_MIX_DAC_CTRL_RHPIS, | 
|  | sun50i_codec_hp_src_enum_text); | 
|  |  | 
|  | static const struct snd_kcontrol_new sun50i_codec_hp_src[] = { | 
|  | SOC_DAPM_ENUM("Headphone Source Playback Route", | 
|  | sun50i_codec_hp_src_enum), | 
|  | }; | 
|  |  | 
|  | static const struct snd_kcontrol_new sun50i_codec_hp_switch = | 
|  | SOC_DAPM_DOUBLE("Headphone Playback Switch", | 
|  | SUN50I_ADDA_MIX_DAC_CTRL, | 
|  | SUN50I_ADDA_MIX_DAC_CTRL_LHPPAMUTE, | 
|  | SUN50I_ADDA_MIX_DAC_CTRL_RHPPAMUTE, 1, 0); | 
|  |  | 
|  | static const char * const sun50i_codec_lineout_src_enum_text[] = { | 
|  | "Stereo", "Mono Differential", | 
|  | }; | 
|  |  | 
|  | static SOC_ENUM_DOUBLE_DECL(sun50i_codec_lineout_src_enum, | 
|  | SUN50I_ADDA_LINEOUT_CTRL0, | 
|  | SUN50I_ADDA_LINEOUT_CTRL0_LSRC_SEL, | 
|  | SUN50I_ADDA_LINEOUT_CTRL0_RSRC_SEL, | 
|  | sun50i_codec_lineout_src_enum_text); | 
|  |  | 
|  | static const struct snd_kcontrol_new sun50i_codec_lineout_src[] = { | 
|  | SOC_DAPM_ENUM("Line Out Source Playback Route", | 
|  | sun50i_codec_lineout_src_enum), | 
|  | }; | 
|  |  | 
|  | static const struct snd_kcontrol_new sun50i_codec_lineout_switch = | 
|  | SOC_DAPM_DOUBLE("Line Out Playback Switch", | 
|  | SUN50I_ADDA_LINEOUT_CTRL0, | 
|  | SUN50I_ADDA_LINEOUT_CTRL0_LEN, | 
|  | SUN50I_ADDA_LINEOUT_CTRL0_REN, 1, 0); | 
|  |  | 
|  | static const char * const sun50i_codec_earpiece_src_enum_text[] = { | 
|  | "DACR", "DACL", "Right Mixer", "Left Mixer", | 
|  | }; | 
|  |  | 
|  | static SOC_ENUM_SINGLE_DECL(sun50i_codec_earpiece_src_enum, | 
|  | SUN50I_ADDA_EARPIECE_CTRL0, | 
|  | SUN50I_ADDA_EARPIECE_CTRL0_ESPSR, | 
|  | sun50i_codec_earpiece_src_enum_text); | 
|  |  | 
|  | static const struct snd_kcontrol_new sun50i_codec_earpiece_src[] = { | 
|  | SOC_DAPM_ENUM("Earpiece Source Playback Route", | 
|  | sun50i_codec_earpiece_src_enum), | 
|  | }; | 
|  |  | 
|  | static const struct snd_kcontrol_new sun50i_codec_earpiece_switch[] = { | 
|  | SOC_DAPM_SINGLE("Earpiece Playback Switch", | 
|  | SUN50I_ADDA_EARPIECE_CTRL1, | 
|  | SUN50I_ADDA_EARPIECE_CTRL1_ESPPA_MUTE, 1, 0), | 
|  | }; | 
|  |  | 
|  | static int sun50i_codec_hbias_event(struct snd_soc_dapm_widget *w, | 
|  | struct snd_kcontrol *kcontrol, int event) | 
|  | { | 
|  | struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); | 
|  | u32 value = !!SND_SOC_DAPM_EVENT_ON(event); | 
|  |  | 
|  | regmap_update_bits(component->regmap, SUN50I_ADDA_JACK_MIC_CTRL, | 
|  | BIT(SUN50I_ADDA_JACK_MIC_CTRL_MICADCEN), | 
|  | value << SUN50I_ADDA_JACK_MIC_CTRL_MICADCEN); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct snd_soc_dapm_widget sun50i_a64_codec_widgets[] = { | 
|  | /* DAC */ | 
|  | SND_SOC_DAPM_DAC("Left DAC", NULL, SUN50I_ADDA_MIX_DAC_CTRL, | 
|  | SUN50I_ADDA_MIX_DAC_CTRL_DACALEN, 0), | 
|  | SND_SOC_DAPM_DAC("Right DAC", NULL, SUN50I_ADDA_MIX_DAC_CTRL, | 
|  | SUN50I_ADDA_MIX_DAC_CTRL_DACAREN, 0), | 
|  | /* ADC */ | 
|  | SND_SOC_DAPM_ADC("Left ADC", NULL, SUN50I_ADDA_ADC_CTRL, | 
|  | SUN50I_ADDA_ADC_CTRL_ADCLEN, 0), | 
|  | SND_SOC_DAPM_ADC("Right ADC", NULL, SUN50I_ADDA_ADC_CTRL, | 
|  | SUN50I_ADDA_ADC_CTRL_ADCREN, 0), | 
|  | /* | 
|  | * Due to this component and the codec belonging to separate DAPM | 
|  | * contexts, we need to manually link the above widgets to their | 
|  | * stream widgets at the card level. | 
|  | */ | 
|  |  | 
|  | SND_SOC_DAPM_REGULATOR_SUPPLY("cpvdd", 0, 0), | 
|  | SND_SOC_DAPM_MUX("Left Headphone Source", | 
|  | SND_SOC_NOPM, 0, 0, sun50i_codec_hp_src), | 
|  | SND_SOC_DAPM_MUX("Right Headphone Source", | 
|  | SND_SOC_NOPM, 0, 0, sun50i_codec_hp_src), | 
|  | SND_SOC_DAPM_SWITCH("Left Headphone Switch", | 
|  | SND_SOC_NOPM, 0, 0, &sun50i_codec_hp_switch), | 
|  | SND_SOC_DAPM_SWITCH("Right Headphone Switch", | 
|  | SND_SOC_NOPM, 0, 0, &sun50i_codec_hp_switch), | 
|  | SND_SOC_DAPM_OUT_DRV("Left Headphone Amp", | 
|  | SND_SOC_NOPM, 0, 0, NULL, 0), | 
|  | SND_SOC_DAPM_OUT_DRV("Right Headphone Amp", | 
|  | SND_SOC_NOPM, 0, 0, NULL, 0), | 
|  | SND_SOC_DAPM_SUPPLY("Headphone Amp", SUN50I_ADDA_HP_CTRL, | 
|  | SUN50I_ADDA_HP_CTRL_HPPA_EN, 0, NULL, 0), | 
|  | SND_SOC_DAPM_OUTPUT("HP"), | 
|  |  | 
|  | SND_SOC_DAPM_MUX("Left Line Out Source", | 
|  | SND_SOC_NOPM, 0, 0, sun50i_codec_lineout_src), | 
|  | SND_SOC_DAPM_MUX("Right Line Out Source", | 
|  | SND_SOC_NOPM, 0, 0, sun50i_codec_lineout_src), | 
|  | SND_SOC_DAPM_SWITCH("Left Line Out Switch", | 
|  | SND_SOC_NOPM, 0, 0, &sun50i_codec_lineout_switch), | 
|  | SND_SOC_DAPM_SWITCH("Right Line Out Switch", | 
|  | SND_SOC_NOPM, 0, 0, &sun50i_codec_lineout_switch), | 
|  | SND_SOC_DAPM_OUTPUT("LINEOUT"), | 
|  |  | 
|  | SND_SOC_DAPM_MUX("Earpiece Source Playback Route", | 
|  | SND_SOC_NOPM, 0, 0, sun50i_codec_earpiece_src), | 
|  | SOC_MIXER_NAMED_CTL_ARRAY("Earpiece Switch", | 
|  | SND_SOC_NOPM, 0, 0, | 
|  | sun50i_codec_earpiece_switch), | 
|  | SND_SOC_DAPM_OUT_DRV("Earpiece Amp", SUN50I_ADDA_EARPIECE_CTRL1, | 
|  | SUN50I_ADDA_EARPIECE_CTRL1_ESPPA_EN, 0, NULL, 0), | 
|  | SND_SOC_DAPM_OUTPUT("EARPIECE"), | 
|  |  | 
|  | /* Microphone inputs */ | 
|  | SND_SOC_DAPM_INPUT("MIC1"), | 
|  |  | 
|  | /* Microphone Bias */ | 
|  | SND_SOC_DAPM_SUPPLY("MBIAS", SUN50I_ADDA_HS_MBIAS_CTRL, | 
|  | SUN50I_ADDA_HS_MBIAS_CTRL_MMICBIASEN, | 
|  | 0, NULL, 0), | 
|  |  | 
|  | /* Mic input path */ | 
|  | SND_SOC_DAPM_PGA("Mic1 Amplifier", SUN50I_ADDA_MIC1_CTRL, | 
|  | SUN50I_ADDA_MIC1_CTRL_MIC1AMPEN, 0, NULL, 0), | 
|  |  | 
|  | /* Microphone input */ | 
|  | SND_SOC_DAPM_INPUT("MIC2"), | 
|  |  | 
|  | /* Microphone Bias */ | 
|  | SND_SOC_DAPM_SUPPLY("HBIAS", SUN50I_ADDA_JACK_MIC_CTRL, | 
|  | SUN50I_ADDA_JACK_MIC_CTRL_HMICBIASEN, | 
|  | 0, sun50i_codec_hbias_event, | 
|  | SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD), | 
|  |  | 
|  | /* Mic input path */ | 
|  | SND_SOC_DAPM_PGA("Mic2 Amplifier", SUN50I_ADDA_MIC2_CTRL, | 
|  | SUN50I_ADDA_MIC2_CTRL_MIC2AMPEN, 0, NULL, 0), | 
|  |  | 
|  | /* Line input */ | 
|  | SND_SOC_DAPM_INPUT("LINEIN"), | 
|  |  | 
|  | /* Mixers */ | 
|  | SND_SOC_DAPM_MIXER("Left Mixer", SUN50I_ADDA_MIX_DAC_CTRL, | 
|  | SUN50I_ADDA_MIX_DAC_CTRL_LMIXEN, 0, | 
|  | sun50i_a64_codec_mixer_controls, | 
|  | ARRAY_SIZE(sun50i_a64_codec_mixer_controls)), | 
|  | SND_SOC_DAPM_MIXER("Right Mixer", SUN50I_ADDA_MIX_DAC_CTRL, | 
|  | SUN50I_ADDA_MIX_DAC_CTRL_RMIXEN, 0, | 
|  | sun50i_a64_codec_mixer_controls, | 
|  | ARRAY_SIZE(sun50i_a64_codec_mixer_controls)), | 
|  | SND_SOC_DAPM_MIXER("Left ADC Mixer", SND_SOC_NOPM, 0, 0, | 
|  | sun50i_codec_adc_mixer_controls, | 
|  | ARRAY_SIZE(sun50i_codec_adc_mixer_controls)), | 
|  | SND_SOC_DAPM_MIXER("Right ADC Mixer", SND_SOC_NOPM, 0, 0, | 
|  | sun50i_codec_adc_mixer_controls, | 
|  | ARRAY_SIZE(sun50i_codec_adc_mixer_controls)), | 
|  | }; | 
|  |  | 
|  | static const struct snd_soc_dapm_route sun50i_a64_codec_routes[] = { | 
|  | /* Left Mixer Routes */ | 
|  | { "Left Mixer", "Mic1 Playback Switch", "Mic1 Amplifier" }, | 
|  | { "Left Mixer", "Mic2 Playback Switch", "Mic2 Amplifier" }, | 
|  | { "Left Mixer", "Line In Playback Switch", "LINEIN" }, | 
|  | { "Left Mixer", "DAC Playback Switch", "Left DAC" }, | 
|  | { "Left Mixer", "DAC Reversed Playback Switch", "Right DAC" }, | 
|  |  | 
|  | /* Right Mixer Routes */ | 
|  | { "Right Mixer", "Mic1 Playback Switch", "Mic1 Amplifier" }, | 
|  | { "Right Mixer", "Mic2 Playback Switch", "Mic2 Amplifier" }, | 
|  | { "Right Mixer", "Line In Playback Switch", "LINEIN" }, | 
|  | { "Right Mixer", "DAC Playback Switch", "Right DAC" }, | 
|  | { "Right Mixer", "DAC Reversed Playback Switch", "Left DAC" }, | 
|  |  | 
|  | /* Left ADC Mixer Routes */ | 
|  | { "Left ADC Mixer", "Mic1 Capture Switch", "Mic1 Amplifier" }, | 
|  | { "Left ADC Mixer", "Mic2 Capture Switch", "Mic2 Amplifier" }, | 
|  | { "Left ADC Mixer", "Line In Capture Switch", "LINEIN" }, | 
|  | { "Left ADC Mixer", "Mixer Capture Switch", "Left Mixer" }, | 
|  | { "Left ADC Mixer", "Mixer Reversed Capture Switch", "Right Mixer" }, | 
|  |  | 
|  | /* Right ADC Mixer Routes */ | 
|  | { "Right ADC Mixer", "Mic1 Capture Switch", "Mic1 Amplifier" }, | 
|  | { "Right ADC Mixer", "Mic2 Capture Switch", "Mic2 Amplifier" }, | 
|  | { "Right ADC Mixer", "Line In Capture Switch", "LINEIN" }, | 
|  | { "Right ADC Mixer", "Mixer Capture Switch", "Right Mixer" }, | 
|  | { "Right ADC Mixer", "Mixer Reversed Capture Switch", "Left Mixer" }, | 
|  |  | 
|  | /* ADC Routes */ | 
|  | { "Left ADC", NULL, "Left ADC Mixer" }, | 
|  | { "Right ADC", NULL, "Right ADC Mixer" }, | 
|  |  | 
|  | /* Headphone Routes */ | 
|  | { "Left Headphone Source", "DAC", "Left DAC" }, | 
|  | { "Left Headphone Source", "Mixer", "Left Mixer" }, | 
|  | { "Left Headphone Switch", "Headphone Playback Switch", "Left Headphone Source" }, | 
|  | { "Left Headphone Amp", NULL, "Left Headphone Switch" }, | 
|  | { "Left Headphone Amp", NULL, "Headphone Amp" }, | 
|  | { "HP", NULL, "Left Headphone Amp" }, | 
|  |  | 
|  | { "Right Headphone Source", "DAC", "Right DAC" }, | 
|  | { "Right Headphone Source", "Mixer", "Right Mixer" }, | 
|  | { "Right Headphone Switch", "Headphone Playback Switch", "Right Headphone Source" }, | 
|  | { "Right Headphone Amp", NULL, "Right Headphone Switch" }, | 
|  | { "Right Headphone Amp", NULL, "Headphone Amp" }, | 
|  | { "HP", NULL, "Right Headphone Amp" }, | 
|  |  | 
|  | { "Headphone Amp", NULL, "cpvdd" }, | 
|  |  | 
|  | /* Microphone Routes */ | 
|  | { "Mic1 Amplifier", NULL, "MIC1"}, | 
|  |  | 
|  | /* Microphone Routes */ | 
|  | { "Mic2 Amplifier", NULL, "MIC2"}, | 
|  |  | 
|  | /* Line-out Routes */ | 
|  | { "Left Line Out Source", "Stereo", "Left Mixer" }, | 
|  | { "Left Line Out Source", "Mono Differential", "Left Mixer" }, | 
|  | { "Left Line Out Source", "Mono Differential", "Right Mixer" }, | 
|  | { "Left Line Out Switch", "Line Out Playback Switch", "Left Line Out Source" }, | 
|  | { "LINEOUT", NULL, "Left Line Out Switch" }, | 
|  |  | 
|  | { "Right Line Out Switch", "Line Out Playback Switch", "Right Mixer" }, | 
|  | { "Right Line Out Source", "Stereo", "Right Line Out Switch" }, | 
|  | { "Right Line Out Source", "Mono Differential", "Left Line Out Switch" }, | 
|  | { "LINEOUT", NULL, "Right Line Out Source" }, | 
|  |  | 
|  | /* Earpiece Routes */ | 
|  | { "Earpiece Source Playback Route", "DACL", "Left DAC" }, | 
|  | { "Earpiece Source Playback Route", "DACR", "Right DAC" }, | 
|  | { "Earpiece Source Playback Route", "Left Mixer", "Left Mixer" }, | 
|  | { "Earpiece Source Playback Route", "Right Mixer", "Right Mixer" }, | 
|  | { "Earpiece Switch", "Earpiece Playback Switch", "Earpiece Source Playback Route" }, | 
|  | { "Earpiece Amp", NULL, "Earpiece Switch" }, | 
|  | { "EARPIECE", NULL, "Earpiece Amp" }, | 
|  | }; | 
|  |  | 
|  | static int sun50i_a64_codec_set_bias_level(struct snd_soc_component *component, | 
|  | enum snd_soc_bias_level level) | 
|  | { | 
|  | struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component); | 
|  | int hbias; | 
|  |  | 
|  | switch (level) { | 
|  | case SND_SOC_BIAS_OFF: | 
|  | regmap_clear_bits(component->regmap, SUN50I_ADDA_JACK_MIC_CTRL, | 
|  | BIT(SUN50I_ADDA_JACK_MIC_CTRL_JACKDETEN) | | 
|  | BIT(SUN50I_ADDA_JACK_MIC_CTRL_MICADCEN)); | 
|  |  | 
|  | regmap_set_bits(component->regmap, SUN50I_ADDA_HP_CTRL, | 
|  | BIT(SUN50I_ADDA_HP_CTRL_PA_CLK_GATE)); | 
|  | break; | 
|  | case SND_SOC_BIAS_STANDBY: | 
|  | regmap_clear_bits(component->regmap, SUN50I_ADDA_HP_CTRL, | 
|  | BIT(SUN50I_ADDA_HP_CTRL_PA_CLK_GATE)); | 
|  |  | 
|  | hbias = snd_soc_dapm_get_pin_status(dapm, "HBIAS"); | 
|  | regmap_update_bits(component->regmap, SUN50I_ADDA_JACK_MIC_CTRL, | 
|  | BIT(SUN50I_ADDA_JACK_MIC_CTRL_JACKDETEN) | | 
|  | BIT(SUN50I_ADDA_JACK_MIC_CTRL_MICADCEN), | 
|  | BIT(SUN50I_ADDA_JACK_MIC_CTRL_JACKDETEN) | | 
|  | hbias << SUN50I_ADDA_JACK_MIC_CTRL_MICADCEN); | 
|  | break; | 
|  | default: | 
|  | break; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct snd_soc_component_driver sun50i_codec_analog_cmpnt_drv = { | 
|  | .controls		= sun50i_a64_codec_controls, | 
|  | .num_controls		= ARRAY_SIZE(sun50i_a64_codec_controls), | 
|  | .dapm_widgets		= sun50i_a64_codec_widgets, | 
|  | .num_dapm_widgets	= ARRAY_SIZE(sun50i_a64_codec_widgets), | 
|  | .dapm_routes		= sun50i_a64_codec_routes, | 
|  | .num_dapm_routes	= ARRAY_SIZE(sun50i_a64_codec_routes), | 
|  | .set_bias_level		= sun50i_a64_codec_set_bias_level, | 
|  | .idle_bias_on		= true, | 
|  | .suspend_bias_off	= true, | 
|  | }; | 
|  |  | 
|  | static const struct of_device_id sun50i_codec_analog_of_match[] = { | 
|  | { | 
|  | .compatible = "allwinner,sun50i-a64-codec-analog", | 
|  | }, | 
|  | {} | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(of, sun50i_codec_analog_of_match); | 
|  |  | 
|  | static int sun50i_codec_analog_probe(struct platform_device *pdev) | 
|  | { | 
|  | struct regmap *regmap; | 
|  | void __iomem *base; | 
|  | bool enable; | 
|  |  | 
|  | base = devm_platform_ioremap_resource(pdev, 0); | 
|  | if (IS_ERR(base)) { | 
|  | dev_err(&pdev->dev, "Failed to map the registers\n"); | 
|  | return PTR_ERR(base); | 
|  | } | 
|  |  | 
|  | regmap = sun8i_adda_pr_regmap_init(&pdev->dev, base); | 
|  | if (IS_ERR(regmap)) { | 
|  | dev_err(&pdev->dev, "Failed to create regmap\n"); | 
|  | return PTR_ERR(regmap); | 
|  | } | 
|  |  | 
|  | enable = device_property_read_bool(&pdev->dev, | 
|  | "allwinner,internal-bias-resistor"); | 
|  | regmap_update_bits(regmap, SUN50I_ADDA_JACK_MIC_CTRL, | 
|  | BIT(SUN50I_ADDA_JACK_MIC_CTRL_INNERRESEN), | 
|  | enable << SUN50I_ADDA_JACK_MIC_CTRL_INNERRESEN); | 
|  |  | 
|  | /* Select sample interval of the ADC sample to 16ms */ | 
|  | regmap_update_bits(regmap, SUN50I_ADDA_MDET_CTRL, | 
|  | 0x7 << SUN50I_ADDA_MDET_CTRL_SELDETADC_FS | | 
|  | 0x3 << SUN50I_ADDA_MDET_CTRL_SELDETADC_BF, | 
|  | 0x3 << SUN50I_ADDA_MDET_CTRL_SELDETADC_FS | | 
|  | 0x3 << SUN50I_ADDA_MDET_CTRL_SELDETADC_BF); | 
|  |  | 
|  | return devm_snd_soc_register_component(&pdev->dev, | 
|  | &sun50i_codec_analog_cmpnt_drv, | 
|  | NULL, 0); | 
|  | } | 
|  |  | 
|  | static struct platform_driver sun50i_codec_analog_driver = { | 
|  | .driver = { | 
|  | .name = "sun50i-codec-analog", | 
|  | .of_match_table = sun50i_codec_analog_of_match, | 
|  | }, | 
|  | .probe = sun50i_codec_analog_probe, | 
|  | }; | 
|  | module_platform_driver(sun50i_codec_analog_driver); | 
|  |  | 
|  | MODULE_DESCRIPTION("Allwinner internal codec analog controls driver for A64"); | 
|  | MODULE_AUTHOR("Vasily Khoruzhick <anarsoul@gmail.com>"); | 
|  | MODULE_LICENSE("GPL"); | 
|  | MODULE_ALIAS("platform:sun50i-codec-analog"); |