Files
kernel_drivers_examples/debug/codec/es8316.c
2017-03-06 15:09:42 +00:00

1424 lines
42 KiB
C

#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.h>
#include <linux/interrupt.h>
#include <linux/irq.h>
#include <linux/regmap.h>
#include "es8316.h"
#if 1
#define DBG(x...) printk(x)
#else
#define DBG(x...) do { } while (0)
#endif
#define alsa_dbg DBG
#define dmic_used 1
#define amic_used 0
#define INVALID_GPIO -1
#define ES8316_CODEC_SET_SPK 1
#define ES8316_CODEC_SET_HP 2
int HP_IRQ = 0;
int hp_irq_flag = 0;
int es8316_init_reg = 0;
#define GPIO_LOW 0
#define GPIO_HIGH 1
#ifndef es8316_DEF_VOL
#define es8316_DEF_VOL 0x1e
#endif
struct snd_soc_codec *es8316_codec;
static int es8316_init_regs(struct snd_soc_codec *codec);
static int es8316_set_bias_level(struct snd_soc_codec *codec,
enum snd_soc_bias_level level);
static const struct reg_default es8316_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},
};
static int es8316_readable(struct snd_soc_codec *codec, unsigned int reg)
{
if (reg <= 90)
return 1;
else
return 0;
}
static int es8316_volatile(struct snd_soc_codec *codec, unsigned int reg)
{
if (reg <= 90)
return 1;
else
return 0;
}
struct es8316_chip {
/* alway include these two member */
struct device *dev;
struct i2c_client *client;
struct regmap *regmap;
unsigned int dmic_amic;
unsigned int sysclk;
struct snd_pcm_hw_constraint_list *sysclk_constraints;
int spk_ctl_gpio;
int hp_ctl_gpio;
int hp_det_gpio;
bool spk_gpio_level;
bool hp_gpio_level;
bool hp_det_level;
int pwr_count;
};
/*
* es8316_reset
* write value 0xff to reg0x00, the chip will be in reset mode
* then, writer 0x00 to reg0x00, unreset the chip
*/
static int es8316_reset(struct snd_soc_codec *codec)
{
snd_soc_write(codec, ES8316_RESET_REG00, 0x3F);
usleep_range(5000, 5500);
return snd_soc_write(codec, ES8316_RESET_REG00, 0x03);
}
static int es8316_set_gpio(struct es8316_chip *chip, int gpio, bool level)
{
if (!chip) {
DBG("%s : es8316_chip is NULL\n", __func__);
return 0;
}
DBG("%s : set %s %s ctl gpio %s\n", __func__,
gpio & ES8316_CODEC_SET_SPK ? "spk" : "",
gpio & ES8316_CODEC_SET_HP ? "hp" : "",
level ? "HIGH" : "LOW");
if ((gpio & ES8316_CODEC_SET_SPK)
&& chip && chip->spk_ctl_gpio != INVALID_GPIO) {
DBG("%d,set_value%s\n", chip->spk_ctl_gpio, level ?
"HIGH" : "LOW");
gpio_set_value(chip->spk_ctl_gpio, level);
}
return 0;
}
static char mute_flag = 1;
static irqreturn_t hp_det_irq_handler(int irq, void *dev_id)
{
int ret;
unsigned int type;
struct es8316_chip *chip = (struct es8316_chip *)dev_id;
disable_irq_nosync(irq);
type = gpio_get_value(chip->hp_det_gpio) ?
IRQ_TYPE_EDGE_FALLING : IRQ_TYPE_EDGE_RISING;
ret = irq_set_irq_type(irq, type);
if (ret < 0) {
DBG("%s: irq_set_irq_type(%d, %d) failed\n",
__func__, irq, type);
return -1;
}
if (chip->hp_det_level != gpio_get_value(chip->hp_det_gpio))
hp_irq_flag = 0;
else
hp_irq_flag = 1;
if (mute_flag == 0) {
if (chip->hp_det_level == gpio_get_value(chip->hp_det_gpio))
{
es8316_set_gpio(chip, ES8316_CODEC_SET_SPK, !chip->spk_gpio_level);
printk("%s, %d, %d\n", __FUNCTION__, __LINE__, !chip->spk_gpio_level);
}
else
{
es8316_set_gpio(chip, ES8316_CODEC_SET_SPK, chip->spk_gpio_level);
printk("%s, %d, %d\n", __FUNCTION__, __LINE__, chip->spk_gpio_level);
}
}
enable_irq(irq);
return IRQ_HANDLED;
}
/*
* es8316S Controls
*/
/*#define DECLARE_TLV_DB_SCALE(name, min, step, mute) */
/*static const DECLARE_TLV_DB_SCALE(hpout_vol_tlv, -4800, 1200, 0);*/
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 const DECLARE_TLV_DB_SCALE(linin_pga_tlv, 0, 300, 0);*/
/* {0, +3, +6, +9, +12, +15, +18, +21, +24,+27,+30,+33} dB */
static unsigned int linin_pga_tlv[] = {
TLV_DB_RANGE_HEAD(12),
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(ES8316_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(ES8316_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(ES8316_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(ES8316_DAC_SET1_REG30, 0, 4, dacpol_txt);
static const struct snd_kcontrol_new es8316_snd_controls[] = {
/* HP OUT VOLUME */
SOC_DOUBLE_TLV("HP Playback Volume", ES8316_CPHP_ICAL_VOL_REG18,
4, 0, 0, 1, hpout_vol_tlv),
/* HPMIXER VOLUME Control */
SOC_DOUBLE_TLV("HPMixer Gain", ES8316_HPMIX_VOL_REG16,
0, 4, 7, 0, hpmixer_gain_tlv),
/* DAC Digital controls */
SOC_DOUBLE_R_TLV("DAC Playback Volume", ES8316_DAC_VOLL_REG33,
ES8316_DAC_VOLR_REG34, 0, 0xC0, 1, dac_vol_tlv),
SOC_SINGLE("Enable DAC Soft Ramp", ES8316_DAC_SET1_REG30, 4, 1, 1),
SOC_SINGLE("DAC Soft Ramp Rate", ES8316_DAC_SET1_REG30, 2, 4, 0),
SOC_ENUM("Playback Polarity", dacpol),
SOC_SINGLE("DAC Notch Filter", ES8316_DAC_SET2_REG31, 6, 1, 0),
SOC_SINGLE("DAC Double Fs Mode", ES8316_DAC_SET2_REG31, 7, 1, 0),
SOC_SINGLE("DAC Volume Control-LeR", ES8316_DAC_SET2_REG31, 2, 1, 0),
SOC_SINGLE("DAC Stereo Enhancement", ES8316_DAC_SET3_REG32, 0, 7, 0),
/* +20dB D2SE PGA Control */
SOC_SINGLE_TLV("MIC Boost", ES8316_ADC_D2SEPGA_REG24,
0, 1, 0, mic_bst_tlv),
/* 0-+24dB Lineinput PGA Control */
SOC_SINGLE_TLV("Input PGA", ES8316_ADC_PGAGAIN_REG23,
4, 8, 0, linin_pga_tlv),
/* ADC Digital Control */
SOC_SINGLE_TLV("ADC Capture Volume", ES8316_ADC_VOLUME_REG27,
0, 0xC0, 1, adc_vol_tlv),
SOC_SINGLE("ADC Soft Ramp", ES8316_ADC_MUTE_REG26, 4, 1, 0),
SOC_ENUM("Capture Polarity", adcpol),
SOC_SINGLE("ADC Double FS Mode", ES8316_ADC_DMIC_REG25, 4, 1, 0),
/* ADC ALC Control */
SOC_SINGLE("ALC Capture Target Volume",
ES8316_ADC_ALC3_REG2B, 4, 10, 0),
SOC_SINGLE("ALC Capture Max PGA", ES8316_ADC_ALC1_REG29, 0, 28, 0),
SOC_SINGLE("ALC Capture Min PGA", ES8316_ADC_ALC2_REG2A, 0, 28, 0),
SOC_ENUM("ALC Capture Function", alc_func),
SOC_SINGLE("ALC Capture Hold Time", ES8316_ADC_ALC3_REG2B, 0, 10, 0),
SOC_SINGLE("ALC Capture Decay Time", ES8316_ADC_ALC4_REG2C, 4, 10, 0),
SOC_SINGLE("ALC Capture Attack Time", ES8316_ADC_ALC4_REG2C, 0, 10, 0),
SOC_SINGLE("ALC Capture NG Threshold", ES8316_ADC_ALC6_REG2E, 0, 31, 0),
SOC_ENUM("ALC Capture NG Type", ng_type),
SOC_SINGLE("ALC Capture NG Switch", ES8316_ADC_ALC6_REG2E, 5, 1, 0),
};
/* Analog Input MUX */
static const char * const es8316_analog_in_txt[] = {
"lin1-rin1",
"lin2-rin2",
"lin1-rin1 with 20db Boost",
"lin2-rin2 with 20db Boost"
};
static const unsigned int es8316_analog_in_values[] = {
0,/*1,*/
1,
2,
3
};
static const struct soc_enum es8316_analog_input_enum =
SOC_VALUE_ENUM_SINGLE(ES8316_ADC_PDN_LINSEL_REG22, 4, 3,
ARRAY_SIZE(es8316_analog_in_txt),
es8316_analog_in_txt,
es8316_analog_in_values);
static const struct snd_kcontrol_new es8316_analog_in_mux_controls =
SOC_DAPM_ENUM("Route", es8316_analog_input_enum);
/* Dmic MUX */
static const char * const es8316_dmic_txt[] = {
"dmic disable",
"dmic data at high level",
"dmic data at low level",
};
static const unsigned int es8316_dmic_values[] = {
0,/*1,*/
1,
2
};
static const struct soc_enum es8316_dmic_src_enum =
SOC_VALUE_ENUM_SINGLE(ES8316_ADC_DMIC_REG25, 0, 3,
ARRAY_SIZE(es8316_dmic_txt),
es8316_dmic_txt,
es8316_dmic_values);
static const struct snd_kcontrol_new es8316_dmic_src_controls =
SOC_DAPM_ENUM("Route", es8316_dmic_src_enum);
/* hp mixer mux */
static const char *const es8316_hpmux_texts[] = {
"lin1-rin1",
"lin2-rin2",
"lin-rin with Boost",
"lin-rin with Boost and PGA"
};
static const unsigned int es8316_hpmux_values[] = {0, 1, 2, 3};
static const struct soc_enum es8316_left_hpmux_enum =
SOC_VALUE_ENUM_SINGLE(ES8316_HPMIX_SEL_REG13, 4, 7,
ARRAY_SIZE(es8316_hpmux_texts),
es8316_hpmux_texts,
es8316_hpmux_values);
static const struct snd_kcontrol_new es8316_left_hpmux_controls =
SOC_DAPM_VALUE_ENUM("Route", es8316_left_hpmux_enum);
static const struct soc_enum es8316_right_hpmux_enum =
SOC_VALUE_ENUM_SINGLE(ES8316_HPMIX_SEL_REG13, 0, 7,
ARRAY_SIZE(es8316_hpmux_texts),
es8316_hpmux_texts,
es8316_hpmux_values);
static const struct snd_kcontrol_new es8316_right_hpmux_controls =
SOC_DAPM_VALUE_ENUM("Route", es8316_right_hpmux_enum);
/* headphone Output Mixer */
static const struct snd_kcontrol_new es8316_out_left_mix[] = {
SOC_DAPM_SINGLE("LLIN Switch", ES8316_HPMIX_SWITCH_REG14,
6, 1, 0),
SOC_DAPM_SINGLE("Left DAC Switch", ES8316_HPMIX_SWITCH_REG14,
7, 1, 0),
};
static const struct snd_kcontrol_new es8316_out_right_mix[] = {
SOC_DAPM_SINGLE("RLIN Switch", ES8316_HPMIX_SWITCH_REG14,
2, 1, 0),
SOC_DAPM_SINGLE("Right DAC Switch", ES8316_HPMIX_SWITCH_REG14,
3, 1, 0),
};
/* DAC data source mux */
static const char * const es8316_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 es8316_dacsrc_values[] = {
0, 1, 2, 3};
static const struct soc_enum es8316_dacsrc_mux_enum =
SOC_VALUE_ENUM_SINGLE(ES8316_DAC_SET1_REG30, 6, 4,
ARRAY_SIZE(es8316_dacsrc_texts),
es8316_dacsrc_texts,
es8316_dacsrc_values);
static const struct snd_kcontrol_new es8316_dacsrc_mux_controls =
SOC_DAPM_VALUE_ENUM("Route", es8316_dacsrc_mux_enum);
static const struct snd_soc_dapm_widget es8316_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,
&es8316_analog_in_mux_controls),
SND_SOC_DAPM_PGA("Line input PGA", ES8316_ADC_PDN_LINSEL_REG22,
7, 1, NULL, 0),
/* ADCs */
SND_SOC_DAPM_ADC("Mono ADC", NULL, ES8316_ADC_PDN_LINSEL_REG22, 6, 1),
/* Dmic MUX */
SND_SOC_DAPM_MUX("Digital Mic Mux", SND_SOC_NOPM, 0, 0,
&es8316_dmic_src_controls),
/* Digital Interface */
SND_SOC_DAPM_AIF_OUT("I2S OUT", "I2S1 Capture", 1,
ES8316_SDP_ADCFMT_REG0A, 6, 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,
&es8316_dacsrc_mux_controls),
/* DACs */
SND_SOC_DAPM_DAC("Right DAC", NULL, ES8316_DAC_PDN_REG2F, 0, 1),
SND_SOC_DAPM_DAC("Left DAC", NULL, ES8316_DAC_PDN_REG2F, 4, 1),
/* Headphone Output Side */
/* hpmux for hp mixer */
SND_SOC_DAPM_MUX("Left Hp mux", SND_SOC_NOPM, 0, 0,
&es8316_left_hpmux_controls),
SND_SOC_DAPM_MUX("Right Hp mux", SND_SOC_NOPM, 0, 0,
&es8316_right_hpmux_controls),
/* Output mixer */
SND_SOC_DAPM_MIXER("Left Hp mixer", ES8316_HPMIX_PDN_REG15,
4, 1, &es8316_out_left_mix[0],
ARRAY_SIZE(es8316_out_left_mix)),
SND_SOC_DAPM_MIXER("Right Hp mixer", ES8316_HPMIX_PDN_REG15,
0, 1, &es8316_out_right_mix[0],
ARRAY_SIZE(es8316_out_right_mix)),
SND_SOC_DAPM_MIXER("Left Hp mixer", SND_SOC_NOPM,
4, 1, &es8316_out_left_mix[0],
ARRAY_SIZE(es8316_out_left_mix)),
SND_SOC_DAPM_MIXER("Right Hp mixer", SND_SOC_NOPM,
0, 1, &es8316_out_right_mix[0],
ARRAY_SIZE(es8316_out_right_mix)),
/* Ouput charge pump */
SND_SOC_DAPM_PGA("HPCP L", SND_SOC_NOPM,
0, 0, NULL, 0),
SND_SOC_DAPM_PGA("HPCP R", SND_SOC_NOPM,
0, 0, NULL, 0),
SND_SOC_DAPM_PGA("HPCP L", ES8316_CPHP_OUTEN_REG17,
6, 0, NULL, 0),
SND_SOC_DAPM_PGA("HPCP R", ES8316_CPHP_OUTEN_REG17,
2, 0, NULL, 0),
/* Ouput Driver */
SND_SOC_DAPM_PGA("HPVOL L", SND_SOC_NOPM,
0, 0, NULL, 0),
SND_SOC_DAPM_PGA("HPVOL R", SND_SOC_NOPM,
0, 0, NULL, 0),
/* Ouput Driver */
SND_SOC_DAPM_PGA("HPVOL L", ES8316_CPHP_OUTEN_REG17,
5, 0, NULL, 0),
SND_SOC_DAPM_PGA("HPVOL R", ES8316_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 es8316_dapm_routes[] = {
/*
* record route map
*/
{"MIC1", NULL, "micbias"},
{"MIC2", NULL, "micbias"},
{"DMIC", NULL, "micbias"},
{"Differential Mux", "lin1-rin1", "MIC1"},
{"Differential Mux", "lin2-rin2", "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"},
};
struct _coeff_div {
u32 mclk; /*mclk frequency*/
u32 rate; /*sample rate*/
u8 div; /*adcclk and dacclk divider*/
u8 lrck_h; /*adclrck divider and daclrck divider*/
u8 lrck_l;
u8 sr; /*sclk divider*/
u8 osr; /*adc osr*/
};
/* codec hifi mclk clock divider coefficients */
static const struct _coeff_div coeff_div[] = {
/* 8k */
{12288000, 8000 , 6 , 0x06, 0x00, 21, 32},
{11289600, 8000 , 6 , 0x05, 0x83, 20, 29},
{18432000, 8000 , 9 , 0x09, 0x00, 27, 32},
{16934400, 8000 , 8 , 0x08, 0x44, 25, 33},
{12000000, 8000 , 7 , 0x05, 0xdc, 21, 25},
{19200000, 8000 , 12, 0x09, 0x60, 27, 25},
/* 11.025k */
{11289600, 11025, 4 , 0x04, 0x00, 16, 32},
{16934400, 11025, 6 , 0x06, 0x00, 21, 32},
{12000000, 11025, 4 , 0x04, 0x40, 17, 34},
/* 16k */
{12288000, 16000, 3 , 0x03, 0x00, 12, 32},
{18432000, 16000, 5 , 0x04, 0x80, 18, 25},
{12000000, 16000, 3 , 0x02, 0xee, 12, 31},
{19200000, 16000, 6 , 0x04, 0xb0, 18, 25},
/* 22.05k */
{11289600, 22050, 2 , 0x02, 0x00, 8 , 32},
{16934400, 22050, 3 , 0x03, 0x00, 12, 32},
{12000000, 22050, 2 , 0x02, 0x20, 8 , 34},
/* 32k */
{12288000, 32000, 1 , 0x01, 0x80, 6 , 48},
{18432000, 32000, 2 , 0x02, 0x40, 9 , 32},
{12000000, 32000, 1 , 0x01, 0x77, 6 , 31},
{19200000, 32000, 3 , 0x02, 0x58, 10, 25},
/* 44.1k */
{11289600, 44100, 1 , 0x01, 0x00, 4 , 32},
{16934400, 44100, 1 , 0x01, 0x80, 6 , 32},
{12000000, 44100, 1 , 0x01, 0x10, 4 , 34},
/* 48k */
{12288000, 48000, 1 , 0x01, 0x00, 4 , 32},
{18432000, 48000, 1 , 0x01, 0x80, 6 , 32},
{12000000, 48000, 1 , 0x00, 0xfa, 4 , 31},
{19200000, 48000, 2 , 0x01, 0x90, 6, 25},
/* 88.2k */
{11289600, 88200, 1 , 0x00, 0x80, 2 , 32},
{16934400, 88200, 1 , 0x00, 0xc0, 3 , 48},
{12000000, 88200, 1 , 0x00, 0x88, 2 , 34},
/* 96k */
{12288000, 96000, 1 , 0x00, 0x80, 2 , 32},
{18432000, 96000, 1 , 0x00, 0xc0, 3 , 48},
{12000000, 96000, 1 , 0x00, 0x7d, 1 , 31},
{19200000, 96000, 1 , 0x00, 0xc8, 3 , 25},
};
static inline int get_coeff(int mclk, int rate)
{
int i;
for (i = 0; i < ARRAY_SIZE(coeff_div); i++) {
if (coeff_div[i].rate == rate && coeff_div[i].mclk == mclk)
return i;
}
return -EINVAL;
}
/* 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 es8316_set_dai_sysclk(struct snd_soc_dai *codec_dai,
int clk_id, unsigned int freq, int dir)
{
struct snd_soc_codec *codec = codec_dai->codec;
struct es8316_chip *chip = snd_soc_codec_get_drvdata(codec);
switch (freq) {
case 11289600:
case 18432000:
case 22579200:
case 36864000:
chip->sysclk_constraints = &constraints_112896;
chip->sysclk = freq;
return 0;
case 12288000:
case 19200000:
case 16934400:
case 24576000:
case 33868800:
chip->sysclk_constraints = &constraints_12288;
chip->sysclk = freq;
return 0;
case 12000000:
case 24000000:
chip->sysclk_constraints = &constraints_12;
chip->sysclk = freq;
return 0;
}
return -EINVAL;
}
static int es8316_set_dai_fmt(struct snd_soc_dai *codec_dai,
unsigned int fmt)
{
struct snd_soc_codec *codec = codec_dai->codec;
u8 iface = 0;
u8 adciface = 0;
u8 daciface = 0;
alsa_dbg("%s----%d, fmt[%02x]\n", __func__, __LINE__, fmt);
iface = snd_soc_read(codec, ES8316_IFACE);
adciface = snd_soc_read(codec, ES8316_ADC_IFACE);
daciface = snd_soc_read(codec, ES8316_DAC_IFACE);
/* set master/slave audio interface */
switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
case SND_SOC_DAIFMT_CBM_CFM:
alsa_dbg("es8316 in master mode");
iface |= 0x80;
break;
case SND_SOC_DAIFMT_CBS_CFS:
alsa_dbg("es8316 in slave mode");
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_write(codec, ES8316_IFACE, iface);
snd_soc_write(codec, ES8316_ADC_IFACE, adciface);
snd_soc_write(codec, ES8316_DAC_IFACE, daciface);
return 0;
}
static int es8316_pcm_startup(struct snd_pcm_substream *substream,
struct snd_soc_dai *dai)
{
struct snd_soc_codec *codec = dai->codec;
struct es8316_chip *chip = snd_soc_codec_get_drvdata(codec);
bool playback = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK);
DBG("Enter::%s----%d chip->sysclk=%d\n",
__func__, __LINE__, chip->sysclk);
snd_soc_write(codec, ES8316_RESET_REG00, 0xC0);
snd_soc_write(codec, ES8316_SYS_PDN_REG0D, 0x00);
/* es8316: both playback and capture need dac mclk */
snd_soc_update_bits(codec, ES8316_CLKMGR_CLKSW_REG01,
ES8316_CLKMGR_MCLK_DIV_MASK |
ES8316_CLKMGR_DAC_MCLK_MASK,
ES8316_CLKMGR_MCLK_DIV_NML |
ES8316_CLKMGR_DAC_MCLK_EN);
chip->pwr_count++;
if (playback) {
snd_soc_write(codec, ES8316_SYS_LP1_REG0E, 0x3F);
snd_soc_write(codec, ES8316_SYS_LP2_REG0F, 0x1F);
snd_soc_write(codec, ES8316_HPMIX_SWITCH_REG14, 0x88);
snd_soc_write(codec, ES8316_HPMIX_PDN_REG15, 0x00);
snd_soc_write(codec, ES8316_HPMIX_VOL_REG16, 0xBB);
snd_soc_write(codec, ES8316_CPHP_PDN2_REG1A, 0x10);
snd_soc_write(codec, ES8316_CPHP_LDOCTL_REG1B, 0x30);
snd_soc_write(codec, ES8316_CPHP_PDN1_REG19, 0x02);
snd_soc_write(codec, ES8316_DAC_PDN_REG2F, 0x00);
snd_soc_write(codec, ES8316_CPHP_OUTEN_REG17, 0x66);
snd_soc_update_bits(codec, ES8316_CLKMGR_CLKSW_REG01,
ES8316_CLKMGR_DAC_MCLK_MASK |
ES8316_CLKMGR_DAC_ANALOG_MASK,
ES8316_CLKMGR_DAC_MCLK_EN |
ES8316_CLKMGR_DAC_ANALOG_EN);
msleep(50);
DBG("%s playback\n", __func__);
} else {
snd_soc_update_bits(codec,
ES8316_ADC_PDN_LINSEL_REG22, 0xC0, 0x20);
snd_soc_update_bits(codec, ES8316_CLKMGR_CLKSW_REG01,
ES8316_CLKMGR_ADC_MCLK_MASK |
ES8316_CLKMGR_ADC_ANALOG_MASK,
ES8316_CLKMGR_ADC_MCLK_EN |
ES8316_CLKMGR_ADC_ANALOG_EN);
DBG("%s capture\n", __func__);
}
/* The set of sample rates that can be supported depends on the
* MCLK supplied to the CODEC - enforce this.
*/
if (!chip->sysclk) {
dev_err(codec->dev,
"No MCLK configured, call set_sysclk() on init\n");
return -EINVAL;
}
snd_pcm_hw_constraint_list(substream->runtime, 0,
SNDRV_PCM_HW_PARAM_RATE,
chip->sysclk_constraints);
return 0;
}
static void es8316_pcm_shutdown(struct snd_pcm_substream *substream,
struct snd_soc_dai *dai)
{
struct snd_soc_pcm_runtime *rtd = substream->private_data;
struct snd_soc_codec *codec = rtd->codec;
struct es8316_chip *chip = snd_soc_codec_get_drvdata(codec);
bool playback = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK);
if (playback) {
snd_soc_write(codec, ES8316_CPHP_OUTEN_REG17, 0x00);
snd_soc_write(codec, ES8316_DAC_PDN_REG2F, 0x11);
snd_soc_write(codec, ES8316_CPHP_LDOCTL_REG1B, 0x03);
snd_soc_write(codec, ES8316_CPHP_PDN2_REG1A, 0x22);
snd_soc_write(codec, ES8316_CPHP_PDN1_REG19, 0x06);
snd_soc_write(codec, ES8316_HPMIX_SWITCH_REG14, 0x00);
snd_soc_write(codec, ES8316_HPMIX_PDN_REG15, 0x33);
snd_soc_write(codec, ES8316_HPMIX_VOL_REG16, 0x00);
snd_soc_write(codec, ES8316_SYS_PDN_REG0D, 0x00);
snd_soc_write(codec, ES8316_SYS_LP1_REG0E, 0xFF);
snd_soc_write(codec, ES8316_SYS_LP2_REG0F, 0xFF);
snd_soc_update_bits(codec, ES8316_CLKMGR_CLKSW_REG01,
ES8316_CLKMGR_DAC_ANALOG_MASK,
ES8316_CLKMGR_DAC_ANALOG_DIS);
DBG("%s playback\n", __func__);
} else {
snd_soc_write(codec, ES8316_ADC_PDN_LINSEL_REG22, 0xc0);
snd_soc_update_bits(codec, ES8316_CLKMGR_CLKSW_REG01,
ES8316_CLKMGR_ADC_MCLK_MASK |
ES8316_CLKMGR_ADC_ANALOG_MASK,
ES8316_CLKMGR_ADC_MCLK_DIS |
ES8316_CLKMGR_ADC_ANALOG_DIS);
DBG("%s capture\n", __func__);
}
if (--chip->pwr_count) {
DBG("%s pwr count: %d\n", __func__, chip->pwr_count);
} else {
snd_soc_write(codec, ES8316_SYS_PDN_REG0D, 0x3F);
snd_soc_write(codec, ES8316_CLKMGR_CLKSW_REG01, 0xF3);
DBG("%s pwr count close ES8316_SYS_PDN_REG0D\n", __func__);
}
}
static int es8316_pcm_hw_params(struct snd_pcm_substream *substream,
struct snd_pcm_hw_params *params,
struct snd_soc_dai *dai)
{
struct snd_soc_pcm_runtime *rtd = substream->private_data;
struct snd_soc_codec *codec = rtd->codec;
struct es8316_chip *chip = snd_soc_codec_get_drvdata(codec);
int retv;
u16 osrate = snd_soc_read(codec,
ES8316_CLKMGR_ADCOSR_REG03) & 0xc0;
u16 mclkdiv = snd_soc_read(codec,
ES8316_CLKMGR_CLKSW_REG01) & 0x7f;
u16 srate = snd_soc_read(codec,
ES8316_SDP_MS_BCKDIV_REG09) & 0xE0;
u16 adciface = snd_soc_read(codec,
ES8316_SDP_ADCFMT_REG0A) & 0xE3;
u16 daciface = snd_soc_read(codec,
ES8316_SDP_DACFMT_REG0B) & 0xE3;
u16 adcdiv = snd_soc_read(codec,
ES8316_CLKMGR_ADCDIV1_REG04);
u16 adclrckdiv_l = snd_soc_read(codec,
ES8316_CLKMGR_ADCDIV2_REG05) & 0x00;
u16 dacdiv = snd_soc_read(codec, ES8316_CLKMGR_DACDIV1_REG06);
u16 daclrckdiv_l = snd_soc_read(codec,
ES8316_CLKMGR_DACDIV2_REG07) & 0x00;
int coeff;
u16 adclrckdiv_h = adcdiv & 0xf0;
u16 daclrckdiv_h = dacdiv & 0xf0;
adcdiv &= 0x0f;
dacdiv &= 0x0f;
coeff = get_coeff(chip->sysclk, params_rate(params));
if (coeff < 0) {
coeff = get_coeff(chip->sysclk / 2, params_rate(params));
mclkdiv |= 0x80;
}
if (coeff < 0) {
dev_err(codec->dev,
"Unable to configure sample rate %dHz with %dHz MCLK\n",
params_rate(params), chip->sysclk);
return coeff;
}
/* bit size */
switch (params_format(params)) {
case SNDRV_PCM_FORMAT_S16_LE:
adciface |= 0x000C;
daciface |= 0x000C;
break;
case SNDRV_PCM_FORMAT_S20_3LE:
adciface |= 0x0004;
daciface |= 0x0004;
break;
case SNDRV_PCM_FORMAT_S24_LE:
break;
case SNDRV_PCM_FORMAT_S32_LE:
adciface |= 0x0010;
daciface |= 0x0010;
break;
}
/* set iface & srate*/
snd_soc_update_bits(codec, ES8316_SDP_DACFMT_REG0B, 0xe3, daciface);
snd_soc_update_bits(codec, ES8316_SDP_ADCFMT_REG0A, 0xe3, adciface);
snd_soc_update_bits(codec, ES8316_CLKMGR_CLKSW_REG01, 0x80, mclkdiv);
if (coeff >= 0) {
osrate = coeff_div[coeff].osr;
osrate &= 0x3f;
srate |= coeff_div[coeff].sr;
srate &= 0x1f;
adcdiv |= (coeff_div[coeff].div << 4);
adclrckdiv_h |= coeff_div[coeff].lrck_h;
adcdiv &= 0xf0;
adclrckdiv_h &= 0x0f;
adcdiv |= adclrckdiv_h;
adclrckdiv_l = coeff_div[coeff].lrck_l;
dacdiv |= (coeff_div[coeff].div << 4);
daclrckdiv_h |= coeff_div[coeff].lrck_h;
dacdiv &= 0xf0;
daclrckdiv_h &= 0x0f;
dacdiv |= daclrckdiv_h;
daclrckdiv_l = coeff_div[coeff].lrck_l;
}
retv = snd_soc_read(codec, ES8316_GPIO_FLAG);
return 0;
}
static int es8316_mute(struct snd_soc_dai *dai, int mute)
{
/* 根据DAI来获取chip */
struct snd_soc_codec *codec = dai->codec;
struct es8316_chip *chip = snd_soc_codec_get_drvdata(codec);
mute_flag = mute;
if (mute) {
es8316_set_gpio(chip, ES8316_CODEC_SET_SPK, chip->spk_gpio_level);
printk("%s, %d, %d\n", __FUNCTION__, __LINE__, !chip->spk_gpio_level);
msleep(100);
snd_soc_write(codec, ES8316_DAC_SET1_REG30, 0x20);
} else if (dai->playback_active) {
snd_soc_write(codec, ES8316_DAC_SET1_REG30, 0x00);
msleep(130);
if (hp_irq_flag == 0)
{
es8316_set_gpio(chip, ES8316_CODEC_SET_SPK, chip->spk_gpio_level);
printk("%s, %d, %d\n", __FUNCTION__, __LINE__, chip->spk_gpio_level);
}
msleep(150);
}
return 0;
}
static int es8316_set_bias_level(struct snd_soc_codec *codec,
enum snd_soc_bias_level level)
{
switch (level) {
case SND_SOC_BIAS_ON:
dev_dbg(codec->dev, "%s on\n", __func__);
break;
case SND_SOC_BIAS_PREPARE:
dev_dbg(codec->dev, "%s prepare\n", __func__);
break;
case SND_SOC_BIAS_STANDBY:
dev_dbg(codec->dev, "%s standby\n", __func__);
break;
case SND_SOC_BIAS_OFF:
dev_dbg(codec->dev, "%s off\n", __func__);
snd_soc_write(codec, ES8316_CPHP_OUTEN_REG17, 0x00);
snd_soc_write(codec, ES8316_DAC_PDN_REG2F, 0x11);
snd_soc_write(codec, ES8316_CPHP_LDOCTL_REG1B, 0x03);
snd_soc_write(codec, ES8316_CPHP_PDN2_REG1A, 0x22);
snd_soc_write(codec, ES8316_CPHP_PDN1_REG19, 0x06);
snd_soc_write(codec, ES8316_HPMIX_SWITCH_REG14, 0x00);
snd_soc_write(codec, ES8316_HPMIX_PDN_REG15, 0x33);
snd_soc_write(codec, ES8316_HPMIX_VOL_REG16, 0x00);
snd_soc_write(codec, ES8316_ADC_PDN_LINSEL_REG22, 0xC0);
snd_soc_write(codec, ES8316_SYS_PDN_REG0D, 0x3F);
snd_soc_write(codec, ES8316_SYS_LP1_REG0E, 0x3F);
snd_soc_write(codec, ES8316_SYS_LP2_REG0F, 0x1F);
snd_soc_write(codec, ES8316_RESET_REG00, 0x00);
break;
}
codec->dapm.bias_level = level;
return 0;
}
#define es8316_RATES SNDRV_PCM_RATE_8000_96000
#define es8316_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE |\
SNDRV_PCM_FMTBIT_S24_LE)
static struct snd_soc_dai_ops es8316_ops = {
.startup = es8316_pcm_startup,
.hw_params = es8316_pcm_hw_params,
.set_fmt = es8316_set_dai_fmt,
.set_sysclk = es8316_set_dai_sysclk,
.digital_mute = es8316_mute,
.shutdown = es8316_pcm_shutdown,
};
static struct snd_soc_dai_driver es8316_dai = {
.name = "ES8316 HiFi",
.playback = {
.stream_name = "Playback",
.channels_min = 1,
.channels_max = 2,
.rates = es8316_RATES,
.formats = es8316_FORMATS,
},
.capture = {
.stream_name = "Capture",
.channels_min = 1,
.channels_max = 2,
.rates = es8316_RATES,
.formats = es8316_FORMATS,
},
.ops = &es8316_ops,
.symmetric_rates = 1,
};
static int es8316_init_regs(struct snd_soc_codec *codec)
{
dev_dbg(codec->dev, "%s\n", __func__);
snd_soc_write(codec, ES8316_RESET_REG00, 0x3f);
usleep_range(5000, 5500);
snd_soc_write(codec, ES8316_RESET_REG00, 0x00);
snd_soc_write(codec, ES8316_SYS_VMIDSEL_REG0C, 0xFF);
msleep(30);
snd_soc_write(codec, ES8316_CLKMGR_CLKSEL_REG02, 0x08);
snd_soc_write(codec, ES8316_CLKMGR_ADCOSR_REG03, 0x20);
snd_soc_write(codec, ES8316_CLKMGR_ADCDIV1_REG04, 0x11);
snd_soc_write(codec, ES8316_CLKMGR_ADCDIV2_REG05, 0x00);
snd_soc_write(codec, ES8316_CLKMGR_DACDIV1_REG06, 0x11);
snd_soc_write(codec, ES8316_CLKMGR_DACDIV2_REG07, 0x00);
snd_soc_write(codec, ES8316_CLKMGR_CPDIV_REG08, 0x00);
snd_soc_write(codec, ES8316_SDP_MS_BCKDIV_REG09, 0x04);
snd_soc_write(codec, ES8316_CLKMGR_CLKSW_REG01, 0x7F);
snd_soc_write(codec, ES8316_CAL_TYPE_REG1C, 0x0F);
snd_soc_write(codec, ES8316_CAL_HPLIV_REG1E, 0x90);
snd_soc_write(codec, ES8316_CAL_HPRIV_REG1F, 0x90);
snd_soc_write(codec, ES8316_ADC_VOLUME_REG27, 0x00);
snd_soc_write(codec, ES8316_ADC_PDN_LINSEL_REG22, 0xc0);
snd_soc_write(codec, ES8316_ADC_D2SEPGA_REG24, 0x00);
snd_soc_write(codec, ES8316_ADC_DMIC_REG25, 0x08);
snd_soc_write(codec, ES8316_DAC_SET2_REG31, 0x20);
snd_soc_write(codec, ES8316_DAC_SET3_REG32, 0x00);
snd_soc_write(codec, ES8316_DAC_VOLL_REG33, 0x00);
snd_soc_write(codec, ES8316_DAC_VOLR_REG34, 0x00);
snd_soc_write(codec, ES8316_SDP_ADCFMT_REG0A, 0x00);
snd_soc_write(codec, ES8316_SDP_DACFMT_REG0B, 0x00);
snd_soc_write(codec, ES8316_SYS_VMIDLOW_REG10, 0x11);
snd_soc_write(codec, ES8316_SYS_VSEL_REG11, 0xFC);
snd_soc_write(codec, ES8316_SYS_REF_REG12, 0x28);
snd_soc_write(codec, ES8316_SYS_LP1_REG0E, 0x04);
snd_soc_write(codec, ES8316_SYS_LP2_REG0F, 0x0C);
snd_soc_write(codec, ES8316_DAC_PDN_REG2F, 0x11);
snd_soc_write(codec, ES8316_HPMIX_SEL_REG13, 0x00);
snd_soc_write(codec, ES8316_HPMIX_SWITCH_REG14, 0x88);
snd_soc_write(codec, ES8316_HPMIX_PDN_REG15, 0x00);
snd_soc_write(codec, ES8316_HPMIX_VOL_REG16, 0xBB);
snd_soc_write(codec, ES8316_CPHP_PDN2_REG1A, 0x10);
snd_soc_write(codec, ES8316_CPHP_LDOCTL_REG1B, 0x30);
snd_soc_write(codec, ES8316_CPHP_PDN1_REG19, 0x02);
snd_soc_write(codec, ES8316_CPHP_ICAL_VOL_REG18, 0x00);
snd_soc_write(codec, ES8316_GPIO_SEL_REG4D, 0x00);
snd_soc_write(codec, ES8316_GPIO_DEBUNCE_INT_REG4E, 0x02);
snd_soc_write(codec, ES8316_TESTMODE_REG50, 0xA0);
snd_soc_write(codec, ES8316_TEST1_REG51, 0x00);
snd_soc_write(codec, ES8316_TEST2_REG52, 0x00);
snd_soc_write(codec, ES8316_SYS_PDN_REG0D, 0x00);
snd_soc_write(codec, ES8316_RESET_REG00, 0xC0);
msleep(50);
snd_soc_write(codec, ES8316_ADC_PGAGAIN_REG23, 0x60);
snd_soc_write(codec, ES8316_ADC_D2SEPGA_REG24, 0x01);
/* adc ds mode, HPF enable */
snd_soc_write(codec, ES8316_ADC_DMIC_REG25, 0x08);
snd_soc_write(codec, ES8316_ADC_ALC1_REG29, 0xcd);
snd_soc_write(codec, ES8316_ADC_ALC2_REG2A, 0x08);
snd_soc_write(codec, ES8316_ADC_ALC3_REG2B, 0xa0);
snd_soc_write(codec, ES8316_ADC_ALC4_REG2C, 0x05);
snd_soc_write(codec, ES8316_ADC_ALC5_REG2D, 0x06);
snd_soc_write(codec, ES8316_ADC_ALC6_REG2E, 0x61);
return 0;
}
static int es8316_suspend(struct snd_soc_codec *codec)
{
return 0;
}
static int es8316_resume(struct snd_soc_codec *codec)
{
struct es8316_chip *chip = snd_soc_codec_get_drvdata(codec);
int retv;
pr_info("%s: %d\n", __func__, __LINE__);
retv = es8316_reset(codec); /* UPDATED BY DAVID,15-3-5 */
retv = snd_soc_read(codec, ES8316_CLKMGR_ADCDIV2_REG05);
if (retv == 0) {
es8316_init_regs(codec);
if (chip->dmic_amic == dmic_used) {
/* set gpio2 to DMIC CLK */
snd_soc_write(codec, ES8316_GPIO_SEL_REG4D, 0x02);
} else {
/* set gpio2 to GM SHORT */
snd_soc_write(codec, ES8316_GPIO_SEL_REG4D, 0x00);
}
/* max debance time, enable interrupt, low active */
snd_soc_write(codec, ES8316_GPIO_DEBUNCE_INT_REG4E, 0xf3);
/* es8316_set_bias_level(codec, SND_SOC_BIAS_OFF); */
snd_soc_write(codec, ES8316_CPHP_OUTEN_REG17, 0x00);
snd_soc_write(codec, ES8316_DAC_PDN_REG2F, 0x11);
snd_soc_write(codec, ES8316_CPHP_LDOCTL_REG1B, 0x03);
snd_soc_write(codec, ES8316_CPHP_PDN2_REG1A, 0x22);
snd_soc_write(codec, ES8316_CPHP_PDN1_REG19, 0x06);
snd_soc_write(codec, ES8316_HPMIX_SWITCH_REG14, 0x00);
snd_soc_write(codec, ES8316_HPMIX_PDN_REG15, 0x33);
snd_soc_write(codec, ES8316_HPMIX_VOL_REG16, 0x00);
snd_soc_write(codec, ES8316_SYS_PDN_REG0D, 0x3F);
snd_soc_write(codec, ES8316_SYS_LP1_REG0E, 0xFF);
snd_soc_write(codec, ES8316_SYS_LP2_REG0F, 0xFF);
snd_soc_write(codec, ES8316_CLKMGR_CLKSW_REG01, 0xF3);
snd_soc_write(codec, ES8316_ADC_PDN_LINSEL_REG22, 0xc0);
es8316_init_reg = 1;
}
return 0;
}
static int es8316_probe(struct snd_soc_codec *codec)
{
int ret = 0, retv;
struct es8316_chip *chip = snd_soc_codec_get_drvdata(codec);
DBG("---%s--start--\n", __func__);
ret = snd_soc_codec_set_cache_io(codec, 8, 8, SND_SOC_I2C);
if (ret < 0) {
dev_err(codec->dev, "fail to reset audio (%d)\n", ret);
goto err;
}
retv = snd_soc_read(codec, ES8316_CLKMGR_ADCDIV2_REG05);
if (retv == 0) {
retv = es8316_reset(codec); /* UPDATED BY DAVID,15-3-5 */
retv = snd_soc_read(codec, ES8316_CLKMGR_ADCDIV2_REG05);
pr_err("%s: %d, retv=%x\n", __func__, __LINE__, retv);
if (retv == 0) {
es8316_init_regs(codec);
if (chip->dmic_amic == dmic_used) {
/* set gpio2 to DMIC CLK */
snd_soc_write(codec,
ES8316_GPIO_SEL_REG4D, 0x02);
} else {
/* set gpio2 to GM SHORT */
snd_soc_write(codec,
ES8316_GPIO_SEL_REG4D, 0x00);
}
/* max debance time, enable interrupt, low active */
snd_soc_write(codec,
ES8316_GPIO_DEBUNCE_INT_REG4E, 0xf3);
/* es8316_set_bias_level(codec, SND_SOC_BIAS_OFF); */
snd_soc_write(codec, ES8316_CPHP_OUTEN_REG17, 0x00);
snd_soc_write(codec, ES8316_DAC_PDN_REG2F, 0x11);
snd_soc_write(codec, ES8316_CPHP_LDOCTL_REG1B, 0x03);
snd_soc_write(codec, ES8316_CPHP_PDN2_REG1A, 0x22);
snd_soc_write(codec, ES8316_CPHP_PDN1_REG19, 0x06);
snd_soc_write(codec, ES8316_HPMIX_SWITCH_REG14, 0x00);
snd_soc_write(codec, ES8316_HPMIX_PDN_REG15, 0x33);
snd_soc_write(codec, ES8316_HPMIX_VOL_REG16, 0x00);
snd_soc_write(codec, ES8316_SYS_PDN_REG0D, 0x3F);
snd_soc_write(codec, ES8316_SYS_LP1_REG0E, 0xFF);
snd_soc_write(codec, ES8316_SYS_LP2_REG0F, 0xFF);
snd_soc_write(codec, ES8316_CLKMGR_CLKSW_REG01, 0xF3);
snd_soc_write(codec,
ES8316_ADC_PDN_LINSEL_REG22, 0xc0);
es8316_init_reg = 1;
}
}
codec->dapm.idle_bias_off = 0;
#if defined(HS_IRQ)
det_initalize();
#elif defined(HS_TIMER)
hsdet_init();
#endif
err:
return ret;
}
static int es8316_remove(struct snd_soc_codec *codec)
{
es8316_set_bias_level(codec, SND_SOC_BIAS_OFF);
return 0;
}
static struct snd_soc_codec_driver soc_codec_dev_es8316 = {
.probe = es8316_probe,
.remove = es8316_remove,
.suspend = es8316_suspend,
.resume = es8316_resume,
.set_bias_level = es8316_set_bias_level,
.reg_cache_size = ARRAY_SIZE(es8316_reg_defaults),
.reg_word_size = sizeof(u8),
.reg_cache_default = es8316_reg_defaults,
.volatile_register = es8316_volatile,
.readable_register = es8316_readable,
.controls = es8316_snd_controls,
.num_controls = ARRAY_SIZE(es8316_snd_controls),
.dapm_widgets = es8316_dapm_widgets,
.num_dapm_widgets = ARRAY_SIZE(es8316_dapm_widgets),
.dapm_routes = es8316_dapm_routes,
.num_dapm_routes = ARRAY_SIZE(es8316_dapm_routes),
};
static void es8316_i2c_shutdown(struct i2c_client *client)
{
struct snd_soc_codec *codec;
struct es8316_chip *chip = container_of(client, struct es8316_chip, client);
if (!es8316_codec)
goto err;
es8316_set_gpio(chip, ES8316_CODEC_SET_SPK, !chip->spk_gpio_level);
printk("%s, %d, %d\n", __FUNCTION__, __LINE__, !chip->spk_gpio_level);
mdelay(150);
codec = es8316_codec;
snd_soc_write(codec, ES8316_CPHP_ICAL_VOL_REG18, 0x33);
snd_soc_write(codec, ES8316_CPHP_OUTEN_REG17, 0x00);
snd_soc_write(codec, ES8316_CPHP_LDOCTL_REG1B, 0x03);
snd_soc_write(codec, ES8316_CPHP_PDN2_REG1A, 0x22);
snd_soc_write(codec, ES8316_CPHP_PDN1_REG19, 0x06);
snd_soc_write(codec, ES8316_HPMIX_SWITCH_REG14, 0x00);
snd_soc_write(codec, ES8316_HPMIX_PDN_REG15, 0x33);
snd_soc_write(codec, ES8316_HPMIX_VOL_REG16, 0x00);
snd_soc_write(codec, ES8316_ADC_PDN_LINSEL_REG22, 0xC0);
snd_soc_write(codec, ES8316_DAC_PDN_REG2F, 0x11);
snd_soc_write(codec, ES8316_SYS_PDN_REG0D, 0x3F);
snd_soc_write(codec, ES8316_CLKMGR_CLKSW_REG01, 0x03);
snd_soc_write(codec, ES8316_RESET_REG00, 0x7F);
err:
return;
}
int gpio_setup(struct es8316_chip *chip, struct i2c_client *client)
{
int ret;
enum of_gpio_flags flags;
struct device_node *np = client->dev.of_node;
chip->spk_ctl_gpio = of_get_named_gpio_flags(np,
"spk-con-gpio", 0, &flags);
if (chip->spk_ctl_gpio < 0)
{
DBG("%s() Can not read property spk codec-en-gpio\n", __func__);
chip->spk_ctl_gpio = INVALID_GPIO;
}
else
{
chip->spk_gpio_level = (flags & OF_GPIO_ACTIVE_LOW) ? 0 : 1;
ret = gpio_request(chip->spk_ctl_gpio, NULL);
gpio_direction_output(chip->spk_ctl_gpio,
!chip->spk_gpio_level);
}
chip->hp_ctl_gpio = of_get_named_gpio_flags(np,
"hp-con-gpio", 0, &flags);
if (chip->hp_ctl_gpio < 0)
{
DBG("%s() Can not read property hp codec-en-gpio\n", __func__);
chip->hp_ctl_gpio = INVALID_GPIO;
}
else
{
chip->hp_gpio_level = (flags & OF_GPIO_ACTIVE_LOW) ? 0 : 1;
ret = gpio_request(chip->hp_ctl_gpio, NULL);
gpio_direction_output(chip->hp_ctl_gpio, !chip->hp_gpio_level);
}
chip->hp_det_gpio = of_get_named_gpio_flags(np,
"hp-det-gpio", 0, &flags);
if (chip->hp_det_gpio < 0)
{
DBG("%s() Can not read property hp_det gpio\n", __func__);
chip->hp_det_gpio = INVALID_GPIO;
}
else
{
chip->hp_det_level = (flags & OF_GPIO_ACTIVE_LOW) ? 0 : 1;
ret = gpio_request(chip->hp_det_gpio, NULL);
if (ret != 0)
{
DBG("%s request HP_DET error", __func__);
return ret;
}
gpio_direction_input(chip->hp_det_gpio);
}
}
static int es8316_i2c_probe(struct i2c_client *client,
const struct i2c_device_id *id)
{
struct es8316_chip *chip;
int ret = -1;
unsigned long irq_flag = 0;
int hp_irq = 0;
int val = 0;
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
/* Do i2c functionality check */
if (!i2c_check_functionality(adapter, I2C_FUNC_I2C)) {
dev_warn(&adapter->dev, "I2C-Adapter doesn't support I2C_FUNC_I2C\n");
return -EIO;
}
/* alloc memory for chip data */
chip = devm_kzalloc(&client->dev, sizeof(struct es8316_chip), GFP_KERNEL);
if (chip == NULL)
{
printk("ERROR: No memory\n");
return -ENOMEM;
}
/* 设置chip */
chip->client = client;
chip->dev = &client->dev;
chip->dmic_amic = amic_used; /*if internal mic is amic*/
chip->pwr_count = 0;
/* set client data */
i2c_set_clientdata(client, chip);
/* gpio setup */
ret = gpio_setup(chip, client);
if (ret < 0)
{
printk("parse gpios error\n");
}
/* irq setup */
ret = devm_request_threaded_irq(chip->dev, gpio_to_irq(chip->hp_det_gpio), NULL, hp_det_irq_handler, IRQF_TRIGGER_LOW | IRQF_ONESHOT, "ES8316", chip);
if(ret == 0)
printk("%s:register ISR (irq=%d)\n", __FUNCTION__, gpio_to_irq(chip->hp_det_gpio));
/* register codec */
ret = snd_soc_register_codec(&client->dev,
&soc_codec_dev_es8316,
&es8316_dai, 1);
return ret;
}
static int es8316_i2c_remove(struct i2c_client *client)
{
snd_soc_unregister_codec(&client->dev);
kfree(i2c_get_clientdata(client));
return 0;
}
static const unsigned short normal_i2c[] = {0x10, I2C_CLIENT_END};
static const struct i2c_device_id es8316_i2c_id[] = {
{"es8316", 0},
{"10ES8316:00", 0},
{"10ES8316", 0},
{ }
};
MODULE_DEVICE_TABLE(i2c, es8316_i2c_id);
static struct i2c_driver es8316_i2c_driver = {
.driver = {
.name = "es8316",
.owner = THIS_MODULE,
},
.shutdown = es8316_i2c_shutdown,
.probe = es8316_i2c_probe,
.remove = es8316_i2c_remove,
.id_table = es8316_i2c_id,
.class = I2C_CLASS_HWMON,
.address_list = normal_i2c,
};
static int es8316_init(void)
{
return i2c_add_driver(&es8316_i2c_driver);
}
module_init(es8316_init);
static void es8316_exit(void)
{
return i2c_del_driver(&es8316_i2c_driver);
}
module_exit(es8316_exit);
MODULE_DESCRIPTION("ASoC es8316 driver");
MODULE_AUTHOR("M_O_Bz@163.com");
MODULE_LICENSE("GPL");