mirror of
https://github.com/coolsnowwolf/lede.git
synced 2025-06-16 12:35:30 +08:00

* target: add phytium support * kernel/video: add phytium platform ARM GPU support * config: add EFI support to phytium armv8 * target: phytium: remove rtl8821cs driver * target: phytium: refresh dts
1082 lines
35 KiB
C
1082 lines
35 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* es8336.c -- es8336 ALSA SoC audio driver
|
|
* Copyright Everest Semiconductor Co.,Ltd
|
|
* Phytium Information Technology Co.,Ltd
|
|
*
|
|
* Author: David Yang <yangxiaohua@everest-semi.com>
|
|
* Yiqun Zhang <zhangyiqun@phytium.com.cn>
|
|
*
|
|
* 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/clk.h>
|
|
#include <linux/module.h>
|
|
#include <linux/moduleparam.h>
|
|
#include <linux/init.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/pm.h>
|
|
#include <linux/i2c.h>
|
|
#include <linux/spi/spi.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/of_gpio.h>
|
|
#include <sound/core.h>
|
|
#include <sound/pcm.h>
|
|
#include <sound/pcm_params.h>
|
|
#include <sound/tlv.h>
|
|
#include <sound/soc.h>
|
|
#include <sound/soc-dapm.h>
|
|
#include <sound/initval.h>
|
|
#include <linux/proc_fs.h>
|
|
#include <linux/gpio/consumer.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/irq.h>
|
|
#include <linux/regmap.h>
|
|
#include "es8336.h"
|
|
|
|
static struct snd_soc_component *es8336_component;
|
|
|
|
static const struct reg_default es8336_reg_defaults[] = {
|
|
{0x00, 0x03}, {0x01, 0x03}, {0x02, 0x00}, {0x03, 0x20},
|
|
{0x04, 0x11}, {0x05, 0x00}, {0x06, 0x11}, {0x07, 0x00},
|
|
{0x08, 0x00}, {0x09, 0x01}, {0x0a, 0x00}, {0x0b, 0x00},
|
|
{0x0c, 0xf8}, {0x0d, 0x3f}, {0x0e, 0x00}, {0x0f, 0x00},
|
|
{0x10, 0x01}, {0x11, 0xfc}, {0x12, 0x28}, {0x13, 0x00},
|
|
{0x14, 0x00}, {0x15, 0x33}, {0x16, 0x00}, {0x17, 0x00},
|
|
{0x18, 0x88}, {0x19, 0x06}, {0x1a, 0x22}, {0x1b, 0x03},
|
|
{0x1c, 0x0f}, {0x1d, 0x00}, {0x1e, 0x80}, {0x1f, 0x80},
|
|
{0x20, 0x00}, {0x21, 0x00}, {0x22, 0xc0}, {0x23, 0x00},
|
|
{0x24, 0x01}, {0x25, 0x08}, {0x26, 0x10}, {0x27, 0xc0},
|
|
{0x28, 0x00}, {0x29, 0x1c}, {0x2a, 0x00}, {0x2b, 0xb0},
|
|
{0x2c, 0x32}, {0x2d, 0x03}, {0x2e, 0x00}, {0x2f, 0x11},
|
|
{0x30, 0x10}, {0x31, 0x00}, {0x32, 0x00}, {0x33, 0xc0},
|
|
{0x34, 0xc0}, {0x35, 0x1f}, {0x36, 0xf7}, {0x37, 0xfd},
|
|
{0x38, 0xff}, {0x39, 0x1f}, {0x3a, 0xf7}, {0x3b, 0xfd},
|
|
{0x3c, 0xff}, {0x3d, 0x1f}, {0x3e, 0xf7}, {0x3f, 0xfd},
|
|
{0x40, 0xff}, {0x41, 0x1f}, {0x42, 0xf7}, {0x43, 0xfd},
|
|
{0x44, 0xff}, {0x45, 0x1f}, {0x46, 0xf7}, {0x47, 0xfd},
|
|
{0x48, 0xff}, {0x49, 0x1f}, {0x4a, 0xf7}, {0x4b, 0xfd},
|
|
{0x4c, 0xff}, {0x4d, 0x00}, {0x4e, 0x00}, {0x4f, 0xff},
|
|
{0x50, 0x00}, {0x51, 0x00}, {0x52, 0x00}, {0x53, 0x00},
|
|
};
|
|
|
|
/* codec private data */
|
|
struct es8336_priv {
|
|
struct regmap *regmap;
|
|
unsigned int dmic_amic;
|
|
unsigned int sysclk;
|
|
struct snd_pcm_hw_constraint_list *sysclk_constraints;
|
|
struct clk *mclk;
|
|
int debounce_time;
|
|
struct delayed_work work;
|
|
|
|
struct gpio_desc *spk_ctl_gpio;
|
|
struct gpio_desc *hp_det_gpio;
|
|
bool muted;
|
|
bool hp_inserted;
|
|
|
|
u8 mic_src;
|
|
int pwr_count;
|
|
};
|
|
|
|
/*
|
|
* es8336_reset
|
|
* write value 0xff to reg0x00, the chip will be in reset mode
|
|
* then, writer 0x00 to reg0x00, unreset the chip
|
|
*/
|
|
static int es8336_reset(struct snd_soc_component *component)
|
|
{
|
|
snd_soc_component_write(component, ES8336_RESET_REG00, 0x3F);
|
|
usleep_range(5000, 5500);
|
|
return snd_soc_component_write(component, ES8336_RESET_REG00, 0x03);
|
|
}
|
|
|
|
static void es8336_enable_spk(struct es8336_priv *es8336, bool enable)
|
|
{
|
|
gpiod_set_value(es8336->spk_ctl_gpio, enable);
|
|
}
|
|
|
|
static const DECLARE_TLV_DB_SCALE(dac_vol_tlv, -9600, 50, 1);
|
|
static const DECLARE_TLV_DB_SCALE(adc_vol_tlv, -9600, 50, 1);
|
|
static const DECLARE_TLV_DB_SCALE(hpmixer_gain_tlv, -1200, 150, 0);
|
|
static const DECLARE_TLV_DB_SCALE(mic_bst_tlv, 0, 1200, 0);
|
|
|
|
static unsigned int linin_pga_tlv[] = {
|
|
TLV_DB_RANGE_HEAD(9),
|
|
0, 0, TLV_DB_SCALE_ITEM(0, 0, 0),
|
|
1, 1, TLV_DB_SCALE_ITEM(300, 0, 0),
|
|
2, 2, TLV_DB_SCALE_ITEM(600, 0, 0),
|
|
3, 3, TLV_DB_SCALE_ITEM(900, 0, 0),
|
|
4, 4, TLV_DB_SCALE_ITEM(1200, 0, 0),
|
|
5, 5, TLV_DB_SCALE_ITEM(1500, 0, 0),
|
|
6, 6, TLV_DB_SCALE_ITEM(1800, 0, 0),
|
|
7, 7, TLV_DB_SCALE_ITEM(2100, 0, 0),
|
|
8, 8, TLV_DB_SCALE_ITEM(2400, 0, 0),
|
|
};
|
|
|
|
static unsigned int hpout_vol_tlv[] = {
|
|
TLV_DB_RANGE_HEAD(1),
|
|
0, 3, TLV_DB_SCALE_ITEM(-4800, 1200, 0),
|
|
};
|
|
|
|
static const char *const alc_func_txt[] = { "Off", "On" };
|
|
|
|
static const struct soc_enum alc_func =
|
|
SOC_ENUM_SINGLE(ES8336_ADC_ALC1_REG29, 6, 2, alc_func_txt);
|
|
|
|
static const char *const ng_type_txt[] = {
|
|
"Constant PGA Gain", "Mute ADC Output" };
|
|
|
|
static const struct soc_enum ng_type =
|
|
SOC_ENUM_SINGLE(ES8336_ADC_ALC6_REG2E, 6, 2, ng_type_txt);
|
|
|
|
static const char *const adcpol_txt[] = { "Normal", "Invert" };
|
|
|
|
static const struct soc_enum adcpol =
|
|
SOC_ENUM_SINGLE(ES8336_ADC_MUTE_REG26, 1, 2, adcpol_txt);
|
|
|
|
static const char *const dacpol_txt[] = {
|
|
"Normal", "R Invert", "L Invert", "L + R Invert" };
|
|
|
|
static const struct soc_enum dacpol =
|
|
SOC_ENUM_SINGLE(ES8336_DAC_SET1_REG30, 0, 4, dacpol_txt);
|
|
|
|
static const struct snd_kcontrol_new es8336_snd_controls[] = {
|
|
/* HP OUT VOLUME */
|
|
SOC_DOUBLE_TLV("HP Playback Volume", ES8336_CPHP_ICAL_VOL_REG18,
|
|
4, 0, 4, 1, hpout_vol_tlv),
|
|
/* HPMIXER VOLUME Control */
|
|
SOC_DOUBLE_TLV("HPMixer Gain", ES8336_HPMIX_VOL_REG16,
|
|
0, 4, 7, 0, hpmixer_gain_tlv),
|
|
|
|
/* DAC Digital controls */
|
|
SOC_DOUBLE_R_TLV("DAC Playback Volume", ES8336_DAC_VOLL_REG33,
|
|
ES8336_DAC_VOLR_REG34, 0, 0xC0, 1, dac_vol_tlv),
|
|
|
|
SOC_SINGLE("Enable DAC Soft Ramp", ES8336_DAC_SET1_REG30, 4, 1, 1),
|
|
SOC_SINGLE("DAC Soft Ramp Rate", ES8336_DAC_SET1_REG30, 2, 4, 0),
|
|
|
|
SOC_ENUM("Playback Polarity", dacpol),
|
|
SOC_SINGLE("DAC Notch Filter", ES8336_DAC_SET2_REG31, 6, 1, 0),
|
|
SOC_SINGLE("DAC Double Fs Mode", ES8336_DAC_SET2_REG31, 7, 1, 0),
|
|
SOC_SINGLE("DAC Volume Control-LeR", ES8336_DAC_SET2_REG31, 2, 1, 0),
|
|
SOC_SINGLE("DAC Stereo Enhancement", ES8336_DAC_SET3_REG32, 0, 7, 0),
|
|
|
|
/* +20dB D2SE PGA Control */
|
|
SOC_SINGLE_TLV("MIC Boost", ES8336_ADC_D2SEPGA_REG24,
|
|
0, 1, 0, mic_bst_tlv),
|
|
/* 0-+24dB Lineinput PGA Control */
|
|
SOC_SINGLE_TLV("Input PGA", ES8336_ADC_PGAGAIN_REG23,
|
|
4, 8, 0, linin_pga_tlv),
|
|
};
|
|
|
|
/* Analog Input MUX */
|
|
static const char * const es8336_analog_in_txt[] = {
|
|
"lin1-rin1",
|
|
"lin2-rin2",
|
|
"lin1-rin1 with 15db Boost",
|
|
"lin2-rin2 with 15db Boost"
|
|
};
|
|
|
|
static const unsigned int es8336_analog_in_values[] = { 0, 1, 2, 3 };
|
|
|
|
static const struct soc_enum es8336_analog_input_enum =
|
|
SOC_VALUE_ENUM_SINGLE(ES8336_ADC_PDN_LINSEL_REG22, 4, 3,
|
|
ARRAY_SIZE(es8336_analog_in_txt),
|
|
es8336_analog_in_txt,
|
|
es8336_analog_in_values);
|
|
|
|
static const struct snd_kcontrol_new es8336_analog_in_mux_controls =
|
|
SOC_DAPM_ENUM("Route", es8336_analog_input_enum);
|
|
|
|
/* Dmic MUX */
|
|
static const char * const es8336_dmic_txt[] = {
|
|
"dmic disable",
|
|
"dmic data at high level",
|
|
"dmic data at low level",
|
|
};
|
|
|
|
static const unsigned int es8336_dmic_values[] = { 0, 2, 3 };
|
|
|
|
static const struct soc_enum es8336_dmic_src_enum =
|
|
SOC_VALUE_ENUM_SINGLE(ES8336_ADC_DMIC_REG25, 0, 3,
|
|
ARRAY_SIZE(es8336_dmic_txt),
|
|
es8336_dmic_txt,
|
|
es8336_dmic_values);
|
|
|
|
static const struct snd_kcontrol_new es8336_dmic_src_controls =
|
|
SOC_DAPM_ENUM("Route", es8336_dmic_src_enum);
|
|
|
|
/* hp mixer mux */
|
|
static const char *const es8336_hpmux_texts[] = {
|
|
"lin1-rin1",
|
|
"lin2-rin2",
|
|
"lin-rin with Boost",
|
|
"lin-rin with Boost and PGA"
|
|
};
|
|
|
|
static const unsigned int es8336_hpmux_values[] = { 0, 1, 2, 3 };
|
|
|
|
static const struct soc_enum es8336_left_hpmux_enum =
|
|
SOC_VALUE_ENUM_SINGLE(ES8336_HPMIX_SEL_REG13, 4, 7,
|
|
ARRAY_SIZE(es8336_hpmux_texts),
|
|
es8336_hpmux_texts,
|
|
es8336_hpmux_values);
|
|
|
|
static const struct snd_kcontrol_new es8336_left_hpmux_controls =
|
|
SOC_DAPM_ENUM("Route", es8336_left_hpmux_enum);
|
|
|
|
static const struct soc_enum es8336_right_hpmux_enum =
|
|
SOC_VALUE_ENUM_SINGLE(ES8336_HPMIX_SEL_REG13, 0, 7,
|
|
ARRAY_SIZE(es8336_hpmux_texts),
|
|
es8336_hpmux_texts,
|
|
es8336_hpmux_values);
|
|
|
|
static const struct snd_kcontrol_new es8336_right_hpmux_controls =
|
|
SOC_DAPM_ENUM("Route", es8336_right_hpmux_enum);
|
|
|
|
/* headphone Output Mixer */
|
|
static const struct snd_kcontrol_new es8336_out_left_mix[] = {
|
|
SOC_DAPM_SINGLE("LLIN Switch", ES8336_HPMIX_SWITCH_REG14,
|
|
6, 1, 0),
|
|
SOC_DAPM_SINGLE("Left DAC Switch", ES8336_HPMIX_SWITCH_REG14,
|
|
7, 1, 0),
|
|
};
|
|
|
|
static const struct snd_kcontrol_new es8336_out_right_mix[] = {
|
|
SOC_DAPM_SINGLE("RLIN Switch", ES8336_HPMIX_SWITCH_REG14,
|
|
2, 1, 0),
|
|
SOC_DAPM_SINGLE("Right DAC Switch", ES8336_HPMIX_SWITCH_REG14,
|
|
3, 1, 0),
|
|
};
|
|
|
|
/* DAC data source mux */
|
|
static const char * const es8336_dacsrc_texts[] = {
|
|
"LDATA TO LDAC, RDATA TO RDAC",
|
|
"LDATA TO LDAC, LDATA TO RDAC",
|
|
"RDATA TO LDAC, RDATA TO RDAC",
|
|
"RDATA TO LDAC, LDATA TO RDAC",
|
|
};
|
|
|
|
static const unsigned int es8336_dacsrc_values[] = { 0, 1, 2, 3 };
|
|
|
|
static const struct soc_enum es8336_dacsrc_mux_enum =
|
|
SOC_VALUE_ENUM_SINGLE(ES8336_DAC_SET1_REG30, 6, 4,
|
|
ARRAY_SIZE(es8336_dacsrc_texts),
|
|
es8336_dacsrc_texts,
|
|
es8336_dacsrc_values);
|
|
static const struct snd_kcontrol_new es8336_dacsrc_mux_controls =
|
|
SOC_DAPM_ENUM("Route", es8336_dacsrc_mux_enum);
|
|
|
|
static const struct snd_soc_dapm_widget es8336_dapm_widgets[] = {
|
|
/* Input Lines */
|
|
SND_SOC_DAPM_INPUT("DMIC"),
|
|
SND_SOC_DAPM_INPUT("MIC1"),
|
|
SND_SOC_DAPM_INPUT("MIC2"),
|
|
|
|
SND_SOC_DAPM_MICBIAS("micbias", SND_SOC_NOPM,
|
|
0, 0),
|
|
/* Input MUX */
|
|
SND_SOC_DAPM_MUX("Differential Mux", SND_SOC_NOPM, 0, 0,
|
|
&es8336_analog_in_mux_controls),
|
|
|
|
SND_SOC_DAPM_PGA("Line input PGA", ES8336_ADC_PDN_LINSEL_REG22,
|
|
7, 1, NULL, 0),
|
|
|
|
/* ADCs */
|
|
SND_SOC_DAPM_ADC("Mono ADC", NULL, ES8336_ADC_PDN_LINSEL_REG22, 6, 1),
|
|
|
|
/* Dmic MUX */
|
|
SND_SOC_DAPM_MUX("Digital Mic Mux", SND_SOC_NOPM, 0, 0,
|
|
&es8336_dmic_src_controls),
|
|
|
|
/* Digital Interface */
|
|
SND_SOC_DAPM_AIF_OUT("I2S OUT", "I2S1 Capture", 1,
|
|
SND_SOC_NOPM, 0, 0),
|
|
|
|
SND_SOC_DAPM_AIF_IN("I2S IN", "I2S1 Playback", 0,
|
|
SND_SOC_NOPM, 0, 0),
|
|
|
|
/* DACs DATA SRC MUX */
|
|
SND_SOC_DAPM_MUX("DAC SRC Mux", SND_SOC_NOPM, 0, 0,
|
|
&es8336_dacsrc_mux_controls),
|
|
/* DACs */
|
|
SND_SOC_DAPM_DAC("Right DAC", NULL, ES8336_DAC_PDN_REG2F, 0, 1),
|
|
SND_SOC_DAPM_DAC("Left DAC", NULL, ES8336_DAC_PDN_REG2F, 4, 1),
|
|
|
|
/* Headphone Output Side */
|
|
/* hpmux for hp mixer */
|
|
SND_SOC_DAPM_MUX("Left Hp mux", SND_SOC_NOPM, 0, 0,
|
|
&es8336_left_hpmux_controls),
|
|
SND_SOC_DAPM_MUX("Right Hp mux", SND_SOC_NOPM, 0, 0,
|
|
&es8336_right_hpmux_controls),
|
|
/* Output mixer */
|
|
SND_SOC_DAPM_MIXER("Left Hp mixer", ES8336_HPMIX_PDN_REG15,
|
|
4, 1, &es8336_out_left_mix[0],
|
|
ARRAY_SIZE(es8336_out_left_mix)),
|
|
SND_SOC_DAPM_MIXER("Right Hp mixer", ES8336_HPMIX_PDN_REG15,
|
|
0, 1, &es8336_out_right_mix[0],
|
|
ARRAY_SIZE(es8336_out_right_mix)),
|
|
SND_SOC_DAPM_MIXER("Left Hp mixer2", SND_SOC_NOPM,
|
|
4, 1, &es8336_out_left_mix[0],
|
|
ARRAY_SIZE(es8336_out_left_mix)),
|
|
SND_SOC_DAPM_MIXER("Right Hp mixer2", SND_SOC_NOPM,
|
|
0, 1, &es8336_out_right_mix[0],
|
|
ARRAY_SIZE(es8336_out_right_mix)),
|
|
|
|
/* Output charge pump */
|
|
SND_SOC_DAPM_PGA("HPCP L", ES8336_CPHP_OUTEN_REG17,
|
|
6, 0, NULL, 0),
|
|
SND_SOC_DAPM_PGA("HPCP R", ES8336_CPHP_OUTEN_REG17,
|
|
2, 0, NULL, 0),
|
|
|
|
/* Output Driver */
|
|
SND_SOC_DAPM_PGA("HPVOL L", ES8336_CPHP_OUTEN_REG17,
|
|
5, 0, NULL, 0),
|
|
SND_SOC_DAPM_PGA("HPVOL R", ES8336_CPHP_OUTEN_REG17,
|
|
1, 0, NULL, 0),
|
|
/* Output Lines */
|
|
SND_SOC_DAPM_OUTPUT("HPOL"),
|
|
SND_SOC_DAPM_OUTPUT("HPOR"),
|
|
};
|
|
|
|
static const struct snd_soc_dapm_route es8336_dapm_routes[] = {
|
|
/*
|
|
* record route map
|
|
*/
|
|
{"MIC1", NULL, "micbias"},
|
|
{"MIC2", NULL, "micbias"},
|
|
{"DMIC", NULL, "micbias"},
|
|
|
|
{"Differential Mux", "lin1-rin1", "MIC1"},
|
|
{"Differential Mux", "lin2-rin2", "MIC2"},
|
|
{"Differential Mux", "lin1-rin1 with 15db Boost", "MIC1"},
|
|
{"Differential Mux", "lin2-rin2 with 15db Boost", "MIC2"},
|
|
{"Line input PGA", NULL, "Differential Mux"},
|
|
|
|
{"Mono ADC", NULL, "Line input PGA"},
|
|
|
|
{"Digital Mic Mux", "dmic disable", "Mono ADC"},
|
|
{"Digital Mic Mux", "dmic data at high level", "DMIC"},
|
|
{"Digital Mic Mux", "dmic data at low level", "DMIC"},
|
|
|
|
{"I2S OUT", NULL, "Digital Mic Mux"},
|
|
/*
|
|
* playback route map
|
|
*/
|
|
{"DAC SRC Mux", "LDATA TO LDAC, RDATA TO RDAC", "I2S IN"},
|
|
{"DAC SRC Mux", "LDATA TO LDAC, LDATA TO RDAC", "I2S IN"},
|
|
{"DAC SRC Mux", "RDATA TO LDAC, RDATA TO RDAC", "I2S IN"},
|
|
{"DAC SRC Mux", "RDATA TO LDAC, LDATA TO RDAC", "I2S IN"},
|
|
|
|
{"Left DAC", NULL, "DAC SRC Mux"},
|
|
{"Right DAC", NULL, "DAC SRC Mux"},
|
|
|
|
{"Left Hp mux", "lin1-rin1", "MIC1"},
|
|
{"Left Hp mux", "lin2-rin2", "MIC2"},
|
|
{"Left Hp mux", "lin-rin with Boost", "Differential Mux"},
|
|
{"Left Hp mux", "lin-rin with Boost and PGA", "Line input PGA"},
|
|
|
|
{"Right Hp mux", "lin1-rin1", "MIC1"},
|
|
{"Right Hp mux", "lin2-rin2", "MIC2"},
|
|
{"Right Hp mux", "lin-rin with Boost", "Differential Mux"},
|
|
{"Right Hp mux", "lin-rin with Boost and PGA", "Line input PGA"},
|
|
|
|
{"Left Hp mixer", "LLIN Switch", "Left Hp mux"},
|
|
{"Left Hp mixer", "Left DAC Switch", "Left DAC"},
|
|
|
|
{"Right Hp mixer", "RLIN Switch", "Right Hp mux"},
|
|
{"Right Hp mixer", "Right DAC Switch", "Right DAC"},
|
|
|
|
{"HPCP L", NULL, "Left Hp mixer"},
|
|
{"HPCP R", NULL, "Right Hp mixer"},
|
|
|
|
{"HPVOL L", NULL, "HPCP L"},
|
|
{"HPVOL R", NULL, "HPCP R"},
|
|
|
|
{"HPOL", NULL, "HPVOL L"},
|
|
{"HPOR", NULL, "HPVOL R"},
|
|
};
|
|
|
|
|
|
/* The set of rates we can generate from the above for each SYSCLK */
|
|
|
|
static unsigned int rates_12288[] = {
|
|
8000, 12000, 16000, 24000, 24000, 32000, 48000, 96000,
|
|
};
|
|
|
|
static struct snd_pcm_hw_constraint_list constraints_12288 = {
|
|
.count = ARRAY_SIZE(rates_12288),
|
|
.list = rates_12288,
|
|
};
|
|
|
|
static unsigned int rates_112896[] = {
|
|
8000, 11025, 22050, 44100,
|
|
};
|
|
|
|
static struct snd_pcm_hw_constraint_list constraints_112896 = {
|
|
.count = ARRAY_SIZE(rates_112896),
|
|
.list = rates_112896,
|
|
};
|
|
|
|
static unsigned int rates_12[] = {
|
|
8000, 11025, 12000, 16000, 22050, 24000, 32000, 44100, 48000,
|
|
48000, 88235, 96000,
|
|
};
|
|
|
|
static struct snd_pcm_hw_constraint_list constraints_12 = {
|
|
.count = ARRAY_SIZE(rates_12),
|
|
.list = rates_12,
|
|
};
|
|
|
|
/*
|
|
* Note that this should be called from init rather than from hw_params.
|
|
*/
|
|
static int es8336_set_dai_sysclk(struct snd_soc_dai *codec_dai,
|
|
int clk_id, unsigned int freq, int dir)
|
|
{
|
|
struct snd_soc_component *component = codec_dai->component;
|
|
struct es8336_priv *es8336 = snd_soc_component_get_drvdata(component);
|
|
|
|
switch (freq) {
|
|
case 11289600:
|
|
case 18432000:
|
|
case 22579200:
|
|
case 36864000:
|
|
es8336->sysclk_constraints = &constraints_112896;
|
|
es8336->sysclk = freq;
|
|
return 0;
|
|
case 12288000:
|
|
case 19200000:
|
|
case 16934400:
|
|
case 24576000:
|
|
case 33868800:
|
|
es8336->sysclk_constraints = &constraints_12288;
|
|
es8336->sysclk = freq;
|
|
return 0;
|
|
case 12000000:
|
|
case 24000000:
|
|
es8336->sysclk_constraints = &constraints_12;
|
|
es8336->sysclk = freq;
|
|
return 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int es8336_set_dai_fmt(struct snd_soc_dai *codec_dai,
|
|
unsigned int fmt)
|
|
{
|
|
struct snd_soc_component *component = codec_dai->component;
|
|
u8 iface = 0;
|
|
u8 adciface = 0;
|
|
u8 daciface = 0;
|
|
|
|
iface = snd_soc_component_read(component, ES8336_IFACE);
|
|
adciface = snd_soc_component_read(component, ES8336_ADC_IFACE);
|
|
daciface = snd_soc_component_read(component, ES8336_DAC_IFACE);
|
|
|
|
/* set master/slave audio interface */
|
|
switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
|
|
case SND_SOC_DAIFMT_CBM_CFM:
|
|
iface |= 0x80;
|
|
break;
|
|
case SND_SOC_DAIFMT_CBS_CFS:
|
|
iface &= 0x7F;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* interface format */
|
|
switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
|
|
case SND_SOC_DAIFMT_I2S:
|
|
adciface &= 0xFC;
|
|
daciface &= 0xFC;
|
|
break;
|
|
case SND_SOC_DAIFMT_RIGHT_J:
|
|
return -EINVAL;
|
|
case SND_SOC_DAIFMT_LEFT_J:
|
|
adciface &= 0xFC;
|
|
daciface &= 0xFC;
|
|
adciface |= 0x01;
|
|
daciface |= 0x01;
|
|
break;
|
|
case SND_SOC_DAIFMT_DSP_A:
|
|
adciface &= 0xDC;
|
|
daciface &= 0xDC;
|
|
adciface |= 0x03;
|
|
daciface |= 0x03;
|
|
break;
|
|
case SND_SOC_DAIFMT_DSP_B:
|
|
adciface &= 0xDC;
|
|
daciface &= 0xDC;
|
|
adciface |= 0x23;
|
|
daciface |= 0x23;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* clock inversion */
|
|
switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
|
|
case SND_SOC_DAIFMT_NB_NF:
|
|
iface &= 0xDF;
|
|
adciface &= 0xDF;
|
|
daciface &= 0xDF;
|
|
break;
|
|
case SND_SOC_DAIFMT_IB_IF:
|
|
iface |= 0x20;
|
|
adciface |= 0x20;
|
|
daciface |= 0x20;
|
|
break;
|
|
case SND_SOC_DAIFMT_IB_NF:
|
|
iface |= 0x20;
|
|
adciface &= 0xDF;
|
|
daciface &= 0xDF;
|
|
break;
|
|
case SND_SOC_DAIFMT_NB_IF:
|
|
iface &= 0xDF;
|
|
adciface |= 0x20;
|
|
daciface |= 0x20;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
snd_soc_component_write(component, ES8336_IFACE, iface);
|
|
snd_soc_component_write(component, ES8336_ADC_IFACE, adciface);
|
|
snd_soc_component_write(component, ES8336_DAC_IFACE, daciface);
|
|
return 0;
|
|
}
|
|
|
|
static int es8336_pcm_startup(struct snd_pcm_substream *substream,
|
|
struct snd_soc_dai *dai)
|
|
{
|
|
struct snd_soc_component *component = dai->component;
|
|
struct es8336_priv *es8336 = snd_soc_component_get_drvdata(component);
|
|
bool playback = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK);
|
|
|
|
snd_soc_component_write(component, ES8336_RESET_REG00, 0xC0);
|
|
snd_soc_component_write(component, ES8336_SYS_PDN_REG0D, 0x00);
|
|
/* es8336: both playback and capture need dac mclk */
|
|
snd_soc_component_update_bits(component, ES8336_CLKMGR_CLKSW_REG01,
|
|
ES8336_CLKMGR_MCLK_DIV_MASK |
|
|
ES8336_CLKMGR_DAC_MCLK_MASK,
|
|
ES8336_CLKMGR_MCLK_DIV_NML |
|
|
ES8336_CLKMGR_DAC_MCLK_EN);
|
|
es8336->pwr_count++;
|
|
|
|
if (playback) {
|
|
snd_soc_component_write(component, ES8336_SYS_LP1_REG0E, 0x3F);
|
|
snd_soc_component_write(component, ES8336_SYS_LP2_REG0F, 0x1F);
|
|
snd_soc_component_write(component, ES8336_HPMIX_SWITCH_REG14, 0x88);
|
|
snd_soc_component_write(component, ES8336_HPMIX_PDN_REG15, 0x00);
|
|
snd_soc_component_write(component, ES8336_HPMIX_VOL_REG16, 0xBB);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN2_REG1A, 0x10);
|
|
snd_soc_component_write(component, ES8336_CPHP_LDOCTL_REG1B, 0x30);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN1_REG19, 0x02);
|
|
snd_soc_component_write(component, ES8336_DAC_PDN_REG2F, 0x00);
|
|
snd_soc_component_write(component, ES8336_CPHP_OUTEN_REG17, 0x66);
|
|
snd_soc_component_update_bits(component, ES8336_CLKMGR_CLKSW_REG01,
|
|
ES8336_CLKMGR_DAC_MCLK_MASK |
|
|
ES8336_CLKMGR_DAC_ANALOG_MASK,
|
|
ES8336_CLKMGR_DAC_MCLK_EN |
|
|
ES8336_CLKMGR_DAC_ANALOG_EN);
|
|
msleep(50);
|
|
} else {
|
|
snd_soc_component_update_bits(component, ES8336_ADC_PDN_LINSEL_REG22, 0xC0, 0x00);
|
|
snd_soc_component_update_bits(component, ES8336_CLKMGR_CLKSW_REG01,
|
|
ES8336_CLKMGR_ADC_MCLK_MASK |
|
|
ES8336_CLKMGR_ADC_ANALOG_MASK,
|
|
ES8336_CLKMGR_ADC_MCLK_EN |
|
|
ES8336_CLKMGR_ADC_ANALOG_EN);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void es8336_pcm_shutdown(struct snd_pcm_substream *substream,
|
|
struct snd_soc_dai *dai)
|
|
{
|
|
struct snd_soc_component *component = dai->component;
|
|
struct es8336_priv *es8336 = snd_soc_component_get_drvdata(component);
|
|
bool playback = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK);
|
|
|
|
if (playback) {
|
|
snd_soc_component_write(component, ES8336_CPHP_OUTEN_REG17, 0x00);
|
|
snd_soc_component_write(component, ES8336_DAC_PDN_REG2F, 0x11);
|
|
snd_soc_component_write(component, ES8336_CPHP_LDOCTL_REG1B, 0x03);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN2_REG1A, 0x22);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN1_REG19, 0x06);
|
|
snd_soc_component_write(component, ES8336_HPMIX_SWITCH_REG14, 0x00);
|
|
snd_soc_component_write(component, ES8336_HPMIX_PDN_REG15, 0x33);
|
|
snd_soc_component_write(component, ES8336_HPMIX_VOL_REG16, 0x00);
|
|
snd_soc_component_write(component, ES8336_SYS_PDN_REG0D, 0x00);
|
|
snd_soc_component_write(component, ES8336_SYS_LP1_REG0E, 0xFF);
|
|
snd_soc_component_write(component, ES8336_SYS_LP2_REG0F, 0xFF);
|
|
snd_soc_component_update_bits(component, ES8336_CLKMGR_CLKSW_REG01,
|
|
ES8336_CLKMGR_DAC_ANALOG_MASK,
|
|
ES8336_CLKMGR_DAC_ANALOG_DIS);
|
|
} else {
|
|
snd_soc_component_update_bits(component, ES8336_ADC_PDN_LINSEL_REG22, 0xC0, 0xc0);
|
|
snd_soc_component_update_bits(component, ES8336_CLKMGR_CLKSW_REG01,
|
|
ES8336_CLKMGR_ADC_MCLK_MASK |
|
|
ES8336_CLKMGR_ADC_ANALOG_MASK,
|
|
ES8336_CLKMGR_ADC_MCLK_DIS |
|
|
ES8336_CLKMGR_ADC_ANALOG_DIS);
|
|
}
|
|
|
|
if (--es8336->pwr_count == 0) {
|
|
if (!es8336->hp_inserted)
|
|
snd_soc_component_write(component, ES8336_SYS_PDN_REG0D, 0x3F);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_CLKSW_REG01, 0xF3);
|
|
}
|
|
}
|
|
|
|
|
|
static int es8336_pcm_hw_params(struct snd_pcm_substream *substream,
|
|
struct snd_pcm_hw_params *params,
|
|
struct snd_soc_dai *dai)
|
|
{
|
|
struct snd_soc_component *component = dai->component;
|
|
int val = 0;
|
|
|
|
switch (params_format(params)) {
|
|
case SNDRV_PCM_FORMAT_S16_LE:
|
|
val = ES8336_DACWL_16;
|
|
break;
|
|
case SNDRV_PCM_FORMAT_S20_3LE:
|
|
val = ES8336_DACWL_20;
|
|
break;
|
|
case SNDRV_PCM_FORMAT_S24_LE:
|
|
val = ES8336_DACWL_24;
|
|
break;
|
|
case SNDRV_PCM_FORMAT_S32_LE:
|
|
val = ES8336_DACWL_32;
|
|
break;
|
|
default:
|
|
val = ES8336_DACWL_16;
|
|
break;
|
|
}
|
|
|
|
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
|
|
snd_soc_component_update_bits(component, ES8336_SDP_DACFMT_REG0B,
|
|
ES8336_DACWL_MASK, val);
|
|
else
|
|
snd_soc_component_update_bits(component, ES8336_SDP_ADCFMT_REG0A,
|
|
ES8336_ADCWL_MASK, val);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int es8336_mute(struct snd_soc_dai *dai, int mute, int direction)
|
|
{
|
|
struct snd_soc_component *component = dai->component;
|
|
struct es8336_priv *es8336 = snd_soc_component_get_drvdata(component);
|
|
|
|
es8336->muted = mute;
|
|
|
|
if (mute) {
|
|
es8336_enable_spk(es8336, false);
|
|
msleep(100);
|
|
snd_soc_component_write(component, ES8336_DAC_SET1_REG30, 0x20);
|
|
}
|
|
|
|
snd_soc_component_write(component, ES8336_DAC_SET1_REG30, 0x00);
|
|
msleep(130);
|
|
|
|
if (!es8336->hp_inserted)
|
|
es8336_enable_spk(es8336, true);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int es8336_set_bias_level(struct snd_soc_component *component,
|
|
enum snd_soc_bias_level level)
|
|
{
|
|
struct es8336_priv *es8336 = snd_soc_component_get_drvdata(component);
|
|
|
|
switch (level) {
|
|
case SND_SOC_BIAS_ON:
|
|
break;
|
|
|
|
case SND_SOC_BIAS_PREPARE:
|
|
break;
|
|
|
|
case SND_SOC_BIAS_STANDBY:
|
|
break;
|
|
|
|
case SND_SOC_BIAS_OFF:
|
|
snd_soc_component_write(component, ES8336_CPHP_OUTEN_REG17, 0x00);
|
|
snd_soc_component_write(component, ES8336_DAC_PDN_REG2F, 0x11);
|
|
snd_soc_component_write(component, ES8336_CPHP_LDOCTL_REG1B, 0x03);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN2_REG1A, 0x22);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN1_REG19, 0x06);
|
|
snd_soc_component_write(component, ES8336_HPMIX_SWITCH_REG14, 0x00);
|
|
snd_soc_component_write(component, ES8336_HPMIX_PDN_REG15, 0x33);
|
|
snd_soc_component_write(component, ES8336_HPMIX_VOL_REG16, 0x00);
|
|
snd_soc_component_update_bits(component, ES8336_ADC_PDN_LINSEL_REG22, 0xC0, 0xC0);
|
|
if (!es8336->hp_inserted)
|
|
snd_soc_component_write(component, ES8336_SYS_PDN_REG0D, 0x3F);
|
|
snd_soc_component_write(component, ES8336_SYS_LP1_REG0E, 0x3F);
|
|
snd_soc_component_write(component, ES8336_SYS_LP2_REG0F, 0x1F);
|
|
snd_soc_component_write(component, ES8336_RESET_REG00, 0x00);
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define es8336_RATES SNDRV_PCM_RATE_8000_96000
|
|
|
|
#define es8336_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE |\
|
|
SNDRV_PCM_FMTBIT_S24_LE)
|
|
|
|
static const struct snd_soc_dai_ops es8336_ops = {
|
|
.startup = es8336_pcm_startup,
|
|
.hw_params = es8336_pcm_hw_params,
|
|
.set_fmt = es8336_set_dai_fmt,
|
|
.set_sysclk = es8336_set_dai_sysclk,
|
|
.mute_stream = es8336_mute,
|
|
.shutdown = es8336_pcm_shutdown,
|
|
};
|
|
|
|
static struct snd_soc_dai_driver es8336_dai = {
|
|
.name = "es8336-hifi",
|
|
.playback = {
|
|
.stream_name = "Playback",
|
|
.channels_min = 1,
|
|
.channels_max = 2,
|
|
.rates = es8336_RATES,
|
|
.formats = es8336_FORMATS,
|
|
},
|
|
.capture = {
|
|
.stream_name = "Capture",
|
|
.channels_min = 1,
|
|
.channels_max = 2,
|
|
.rates = es8336_RATES,
|
|
.formats = es8336_FORMATS,
|
|
},
|
|
.ops = &es8336_ops,
|
|
.symmetric_rates = 1,
|
|
};
|
|
|
|
static int es8336_init_regs(struct snd_soc_component *component)
|
|
{
|
|
struct es8336_priv *es8336 = snd_soc_component_get_drvdata(component);
|
|
snd_soc_component_write(component, ES8336_RESET_REG00, 0x3f);
|
|
usleep_range(5000, 5500);
|
|
snd_soc_component_write(component, ES8336_RESET_REG00, 0x00);
|
|
snd_soc_component_write(component, ES8336_SYS_VMIDSEL_REG0C, 0xFF);
|
|
msleep(30);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_CLKSEL_REG02, 0x08);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_ADCOSR_REG03, 0x20);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_ADCDIV1_REG04, 0x11);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_ADCDIV2_REG05, 0x00);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_DACDIV1_REG06, 0x11);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_DACDIV2_REG07, 0x00);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_CPDIV_REG08, 0x00);
|
|
snd_soc_component_write(component, ES8336_SDP_MS_BCKDIV_REG09, 0x04);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_CLKSW_REG01, 0x7F);
|
|
snd_soc_component_write(component, ES8336_CAL_TYPE_REG1C, 0x0F);
|
|
snd_soc_component_write(component, ES8336_CAL_HPLIV_REG1E, 0x90);
|
|
snd_soc_component_write(component, ES8336_CAL_HPRIV_REG1F, 0x90);
|
|
snd_soc_component_write(component, ES8336_ADC_VOLUME_REG27, 0x00);
|
|
snd_soc_component_write(component, ES8336_ADC_PDN_LINSEL_REG22, es8336->mic_src);
|
|
snd_soc_component_write(component, ES8336_ADC_D2SEPGA_REG24, 0x00);
|
|
snd_soc_component_write(component, ES8336_ADC_DMIC_REG25, 0x08);
|
|
snd_soc_component_write(component, ES8336_DAC_SET2_REG31, 0x20);
|
|
snd_soc_component_write(component, ES8336_DAC_SET3_REG32, 0x00);
|
|
snd_soc_component_write(component, ES8336_DAC_VOLL_REG33, 0x00);
|
|
snd_soc_component_write(component, ES8336_DAC_VOLR_REG34, 0x00);
|
|
snd_soc_component_write(component, ES8336_SDP_ADCFMT_REG0A, 0x00);
|
|
snd_soc_component_write(component, ES8336_SDP_DACFMT_REG0B, 0x00);
|
|
snd_soc_component_write(component, ES8336_SYS_VMIDLOW_REG10, 0x11);
|
|
snd_soc_component_write(component, ES8336_SYS_VSEL_REG11, 0xFC);
|
|
snd_soc_component_write(component, ES8336_SYS_REF_REG12, 0x28);
|
|
snd_soc_component_write(component, ES8336_SYS_LP1_REG0E, 0x04);
|
|
snd_soc_component_write(component, ES8336_SYS_LP2_REG0F, 0x0C);
|
|
snd_soc_component_write(component, ES8336_DAC_PDN_REG2F, 0x11);
|
|
snd_soc_component_write(component, ES8336_HPMIX_SEL_REG13, 0x00);
|
|
snd_soc_component_write(component, ES8336_HPMIX_SWITCH_REG14, 0x88);
|
|
snd_soc_component_write(component, ES8336_HPMIX_PDN_REG15, 0x00);
|
|
snd_soc_component_write(component, ES8336_HPMIX_VOL_REG16, 0xBB);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN2_REG1A, 0x10);
|
|
snd_soc_component_write(component, ES8336_CPHP_LDOCTL_REG1B, 0x30);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN1_REG19, 0x02);
|
|
snd_soc_component_write(component, ES8336_CPHP_ICAL_VOL_REG18, 0x00);
|
|
snd_soc_component_write(component, ES8336_GPIO_SEL_REG4D, 0x02);
|
|
snd_soc_component_write(component, ES8336_GPIO_DEBUNCE_INT_REG4E, 0x02);
|
|
snd_soc_component_write(component, ES8336_TESTMODE_REG50, 0xA0);
|
|
snd_soc_component_write(component, ES8336_TEST1_REG51, 0x00);
|
|
snd_soc_component_write(component, ES8336_TEST2_REG52, 0x00);
|
|
snd_soc_component_write(component, ES8336_SYS_PDN_REG0D, 0x00);
|
|
snd_soc_component_write(component, ES8336_RESET_REG00, 0xC0);
|
|
msleep(50);
|
|
snd_soc_component_write(component, ES8336_ADC_PGAGAIN_REG23, 0x60);
|
|
snd_soc_component_write(component, ES8336_ADC_D2SEPGA_REG24, 0x01);
|
|
/* adc ds mode, HPF enable */
|
|
snd_soc_component_write(component, ES8336_ADC_DMIC_REG25, 0x08);
|
|
snd_soc_component_write(component, ES8336_ADC_ALC1_REG29, 0xcd);
|
|
snd_soc_component_write(component, ES8336_ADC_ALC2_REG2A, 0x08);
|
|
snd_soc_component_write(component, ES8336_ADC_ALC3_REG2B, 0xa0);
|
|
snd_soc_component_write(component, ES8336_ADC_ALC4_REG2C, 0x05);
|
|
snd_soc_component_write(component, ES8336_ADC_ALC5_REG2D, 0x06);
|
|
snd_soc_component_write(component, ES8336_ADC_ALC6_REG2E, 0x61);
|
|
return 0;
|
|
}
|
|
|
|
static int es8336_suspend(struct snd_soc_component *component)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int es8336_resume(struct snd_soc_component *component)
|
|
{
|
|
struct es8336_priv *es8336 = snd_soc_component_get_drvdata(component);
|
|
int ret;
|
|
|
|
es8336_reset(component); /* UPDATED BY DAVID,15-3-5 */
|
|
ret = snd_soc_component_read(component, ES8336_CLKMGR_ADCDIV2_REG05);
|
|
if (!ret) {
|
|
es8336_init_regs(component);
|
|
snd_soc_component_write(component, ES8336_GPIO_SEL_REG4D, 0x02);
|
|
/* max debance time, enable interrupt, low active */
|
|
snd_soc_component_write(component, ES8336_GPIO_DEBUNCE_INT_REG4E, 0xf3);
|
|
/* es8336_set_bias_level(component, SND_SOC_BIAS_OFF); */
|
|
snd_soc_component_write(component, ES8336_CPHP_OUTEN_REG17, 0x00);
|
|
snd_soc_component_write(component, ES8336_DAC_PDN_REG2F, 0x11);
|
|
snd_soc_component_write(component, ES8336_CPHP_LDOCTL_REG1B, 0x03);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN2_REG1A, 0x22);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN1_REG19, 0x06);
|
|
snd_soc_component_write(component, ES8336_HPMIX_SWITCH_REG14, 0x00);
|
|
snd_soc_component_write(component, ES8336_HPMIX_PDN_REG15, 0x33);
|
|
snd_soc_component_write(component, ES8336_HPMIX_VOL_REG16, 0x00);
|
|
if (!es8336->hp_inserted)
|
|
snd_soc_component_write(component, ES8336_SYS_PDN_REG0D, 0x3F);
|
|
snd_soc_component_write(component, ES8336_SYS_LP1_REG0E, 0xFF);
|
|
snd_soc_component_write(component, ES8336_SYS_LP2_REG0F, 0xFF);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_CLKSW_REG01, 0xF3);
|
|
snd_soc_component_update_bits(component, ES8336_ADC_PDN_LINSEL_REG22, 0xC0, 0xC0);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static irqreturn_t es8336_irq_handler(int irq, void *data)
|
|
{
|
|
struct es8336_priv *es8336 = data;
|
|
|
|
queue_delayed_work(system_power_efficient_wq, &es8336->work,
|
|
msecs_to_jiffies(es8336->debounce_time));
|
|
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
static void hp_work(struct work_struct *work)
|
|
{
|
|
struct es8336_priv *es8336;
|
|
|
|
es8336 = container_of(work, struct es8336_priv, work.work);
|
|
|
|
es8336->hp_inserted = gpiod_get_value(es8336->hp_det_gpio);
|
|
if (!es8336->muted) {
|
|
if (es8336->hp_inserted)
|
|
es8336_enable_spk(es8336, false);
|
|
else
|
|
es8336_enable_spk(es8336, true);
|
|
}
|
|
}
|
|
|
|
static int es8336_probe(struct snd_soc_component *component)
|
|
{
|
|
struct es8336_priv *es8336 = snd_soc_component_get_drvdata(component);
|
|
int ret = 0;
|
|
|
|
es8336_component = component;
|
|
ret = snd_soc_component_read(component, ES8336_CLKMGR_ADCDIV2_REG05);
|
|
if (!ret) {
|
|
es8336_reset(component); /* UPDATED BY DAVID,15-3-5 */
|
|
ret = snd_soc_component_read(component, ES8336_CLKMGR_ADCDIV2_REG05);
|
|
if (!ret) {
|
|
es8336_init_regs(component);
|
|
snd_soc_component_write(component, ES8336_GPIO_SEL_REG4D, 0x02);
|
|
/* max debance time, enable interrupt, low active */
|
|
snd_soc_component_write(component,
|
|
ES8336_GPIO_DEBUNCE_INT_REG4E, 0xf3);
|
|
|
|
/* es8336_set_bias_level(codec, SND_SOC_BIAS_OFF); */
|
|
snd_soc_component_write(component, ES8336_CPHP_OUTEN_REG17, 0x00);
|
|
snd_soc_component_write(component, ES8336_DAC_PDN_REG2F, 0x11);
|
|
snd_soc_component_write(component, ES8336_CPHP_LDOCTL_REG1B, 0x03);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN2_REG1A, 0x22);
|
|
snd_soc_component_write(component, ES8336_CPHP_PDN1_REG19, 0x06);
|
|
snd_soc_component_write(component, ES8336_HPMIX_SWITCH_REG14, 0x00);
|
|
snd_soc_component_write(component, ES8336_HPMIX_PDN_REG15, 0x33);
|
|
snd_soc_component_write(component, ES8336_HPMIX_VOL_REG16, 0x00);
|
|
if (!es8336->hp_inserted)
|
|
snd_soc_component_write(component, ES8336_SYS_PDN_REG0D,
|
|
0x3F);
|
|
snd_soc_component_write(component, ES8336_SYS_LP1_REG0E, 0xFF);
|
|
snd_soc_component_write(component, ES8336_SYS_LP2_REG0F, 0xFF);
|
|
snd_soc_component_write(component, ES8336_CLKMGR_CLKSW_REG01, 0xF3);
|
|
snd_soc_component_update_bits(component, ES8336_ADC_PDN_LINSEL_REG22, 0xC0, 0xC0);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void es8336_remove(struct snd_soc_component *component)
|
|
{
|
|
es8336_set_bias_level(component, SND_SOC_BIAS_OFF);
|
|
}
|
|
|
|
const struct regmap_config es8336_regmap_config = {
|
|
.reg_bits = 8,
|
|
.val_bits = 8,
|
|
.max_register = ES8336_TEST3_REG53,
|
|
.cache_type = REGCACHE_RBTREE,
|
|
.reg_defaults = es8336_reg_defaults,
|
|
.num_reg_defaults = ARRAY_SIZE(es8336_reg_defaults),
|
|
};
|
|
|
|
static const struct snd_soc_component_driver soc_component_dev_es8336 = {
|
|
.probe = es8336_probe,
|
|
.remove = es8336_remove,
|
|
.suspend = es8336_suspend,
|
|
.resume = es8336_resume,
|
|
.set_bias_level = es8336_set_bias_level,
|
|
|
|
.controls = es8336_snd_controls,
|
|
.num_controls = ARRAY_SIZE(es8336_snd_controls),
|
|
.dapm_widgets = es8336_dapm_widgets,
|
|
.num_dapm_widgets = ARRAY_SIZE(es8336_dapm_widgets),
|
|
.dapm_routes = es8336_dapm_routes,
|
|
.num_dapm_routes = ARRAY_SIZE(es8336_dapm_routes),
|
|
};
|
|
|
|
static int es8336_i2c_probe(struct i2c_client *i2c,
|
|
const struct i2c_device_id *id)
|
|
{
|
|
struct es8336_priv *es8336;
|
|
int ret = -1;
|
|
int hp_irq;
|
|
|
|
es8336 = devm_kzalloc(&i2c->dev, sizeof(*es8336), GFP_KERNEL);
|
|
if (!es8336)
|
|
return -ENOMEM;
|
|
|
|
es8336->debounce_time = 200;
|
|
es8336->pwr_count = 0;
|
|
es8336->hp_inserted = false;
|
|
es8336->muted = true;
|
|
|
|
es8336->regmap = devm_regmap_init_i2c(i2c, &es8336_regmap_config);
|
|
if (IS_ERR(es8336->regmap)) {
|
|
ret = PTR_ERR(es8336->regmap);
|
|
dev_err(&i2c->dev, "Failed to init regmap: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
i2c_set_clientdata(i2c, es8336);
|
|
|
|
es8336->spk_ctl_gpio = devm_gpiod_get_index_optional(&i2c->dev, "sel", 0,
|
|
GPIOD_OUT_HIGH);
|
|
ret = of_property_read_u8(i2c->dev.of_node, "mic-src", &es8336->mic_src);
|
|
if (ret != 0) {
|
|
dev_dbg(&i2c->dev, "mic1-src return %d", ret);
|
|
es8336->mic_src = 0x20;
|
|
}
|
|
dev_dbg(&i2c->dev, "mic1-src %x", es8336->mic_src);
|
|
|
|
if (IS_ERR_OR_NULL(es8336->spk_ctl_gpio))
|
|
dev_info(&i2c->dev, "Can not get spk_ctl_gpio\n");
|
|
else
|
|
es8336_enable_spk(es8336, false);
|
|
|
|
es8336->hp_det_gpio = devm_gpiod_get_index_optional(&i2c->dev, "det", 0,
|
|
GPIOD_IN);
|
|
|
|
if (IS_ERR_OR_NULL(es8336->hp_det_gpio)) {
|
|
dev_info(&i2c->dev, "Can not get hp_det_gpio\n");
|
|
} else {
|
|
INIT_DELAYED_WORK(&es8336->work, hp_work);
|
|
hp_irq = gpiod_to_irq(es8336->hp_det_gpio);
|
|
ret = devm_request_threaded_irq(&i2c->dev, hp_irq, NULL,
|
|
es8336_irq_handler,
|
|
IRQF_TRIGGER_FALLING |
|
|
IRQF_TRIGGER_RISING |
|
|
IRQF_ONESHOT,
|
|
"es8336_interrupt", es8336);
|
|
if (ret < 0) {
|
|
dev_err(&i2c->dev, "request_irq failed: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
schedule_delayed_work(&es8336->work,
|
|
msecs_to_jiffies(es8336->debounce_time));
|
|
}
|
|
|
|
ret = snd_soc_register_component(&i2c->dev,
|
|
&soc_component_dev_es8336,
|
|
&es8336_dai, 1);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int es8336_i2c_remove(struct i2c_client *client)
|
|
{
|
|
kfree(i2c_get_clientdata(client));
|
|
return 0;
|
|
}
|
|
|
|
static void es8336_i2c_shutdown(struct i2c_client *client)
|
|
{
|
|
struct es8336_priv *es8336 = i2c_get_clientdata(client);
|
|
|
|
if (es8336_component != NULL) {
|
|
es8336_enable_spk(es8336, false);
|
|
msleep(20);
|
|
es8336_set_bias_level(es8336_component, SND_SOC_BIAS_OFF);
|
|
}
|
|
}
|
|
|
|
static const struct i2c_device_id es8336_i2c_id[] = {
|
|
{"es8336", 0},
|
|
{"10ES8336:00", 0},
|
|
{"10ES8336", 0},
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(i2c, es8336_i2c_id);
|
|
|
|
static const struct of_device_id es8336_of_match[] = {
|
|
{ .compatible = "everest,es8336", },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(of, es8336_of_match);
|
|
|
|
static const struct acpi_device_id es8336_acpi_match[] = {
|
|
{ "ESSX8336", 0},
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(acpi, es8336_acpi_match);
|
|
|
|
static struct i2c_driver es8336_i2c_driver = {
|
|
.driver = {
|
|
.name = "es8336",
|
|
.of_match_table = es8336_of_match,
|
|
.acpi_match_table = es8336_acpi_match,
|
|
},
|
|
.probe = es8336_i2c_probe,
|
|
.remove = es8336_i2c_remove,
|
|
.shutdown = es8336_i2c_shutdown,
|
|
.id_table = es8336_i2c_id,
|
|
};
|
|
|
|
module_i2c_driver(es8336_i2c_driver);
|
|
MODULE_DESCRIPTION("ASoC es8336 driver");
|
|
MODULE_LICENSE("GPL");
|