/* * ac101.c * * (C) Copyright 2017-2018 * Seeed Technology Co., Ltd. * * PeterYang * * (C) Copyright 2014-2017 * Reuuimlla Technology Co., Ltd. * * huangxin * liushaohua * * X-Powers AC101 codec driver * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * */ /* #undef AC101_DEBG * use 'make DEBUG=1' to enable debugging */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "ac101_regs.h" #include "ac10x.h" /* * *** To sync channels *** * * 1. disable clock in codec hw_params() * 2. clear fifo in bcm2835 hw_params() * 3. clear fifo in bcm2385 prepare() * 4. enable RX in bcm2835 trigger() * 5. enable clock in machine trigger() */ /*Default initialize configuration*/ static bool speaker_double_used = 1; static int double_speaker_val = 0x1B; static int single_speaker_val = 0x19; static int headset_val = 0x3B; static int mainmic_val = 0x4; static int headsetmic_val = 0x4; static bool dmic_used = 0; static int adc_digital_val = 0xb0b0; static bool drc_used = false; #define AC101_RATES (SNDRV_PCM_RATE_8000_96000 & \ ~(SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_64000 | \ SNDRV_PCM_RATE_88200)) #define AC101_FORMATS (/*SNDRV_PCM_FMTBIT_S16_LE | \ SNDRV_PCM_FMTBIT_S24_LE |*/ \ SNDRV_PCM_FMTBIT_S32_LE | \ 0) static struct ac10x_priv* static_ac10x; int ac101_read(struct snd_soc_codec *codec, unsigned reg) { struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); int r, v = 0; if ((r = regmap_read(ac10x->regmap101, reg, &v)) < 0) { dev_err(codec->dev, "read reg %02X fail\n", reg); return r; } return v; } int ac101_write(struct snd_soc_codec *codec, unsigned reg, unsigned val) { struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); int v; v = regmap_write(ac10x->regmap101, reg, val); return v; } int ac101_update_bits(struct snd_soc_codec *codec, unsigned reg, unsigned mask, unsigned value ) { struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); int v; v = regmap_update_bits(ac10x->regmap101, reg, mask, value); return v; } #ifdef CONFIG_AC101_SWITCH_DETECT /******************************************************************************/ /********************************switch****************************************/ /******************************************************************************/ #define KEY_HEADSETHOOK 226 /* key define */ #define HEADSET_FILTER_CNT (10) /* * switch_hw_config:config the 53 codec register */ static void switch_hw_config(struct snd_soc_codec *codec) { int r; AC101_DBG(); /*HMIC/MMIC BIAS voltage level select:2.5v*/ ac101_update_bits(codec, OMIXER_BST1_CTRL, (0xf<state:%d\n", ac10x->state); input_report_switch(ac10x->inpdev, SW_HEADPHONE_INSERT, ac10x->state); input_sync(ac10x->inpdev); return; } /* * work_cb_clear_irq: clear audiocodec pending and Record the interrupt. */ static void work_cb_clear_irq(struct work_struct *work) { int reg_val = 0; struct ac10x_priv *ac10x = container_of(work, struct ac10x_priv, work_clear_irq); struct snd_soc_codec *codec = ac10x->codec; ac10x->irq_cntr++; reg_val = ac101_read(codec, HMIC_STS); if (BIT(HMIC_PULLOUT_PEND) & reg_val) { ac10x->pullout_cntr++; AC101_DBG("ac10x->pullout_cntr: %d\n", ac10x->pullout_cntr); } reg_val |= HMIC_PEND_ALL; ac101_write(codec, HMIC_STS, reg_val); reg_val = ac101_read(codec, HMIC_STS); if ((reg_val & HMIC_PEND_ALL) != 0){ reg_val |= HMIC_PEND_ALL; ac101_write(codec, HMIC_STS, reg_val); } if (cancel_work_sync(&ac10x->work_switch) != 0) { ac10x->irq_cntr--; } if (0 == schedule_work(&ac10x->work_switch)) { ac10x->irq_cntr--; AC101_DBG("[work_cb_clear_irq] add work struct failed!\n"); } } enum { HBIAS_LEVEL_1 = 0x02, HBIAS_LEVEL_2 = 0x0B, HBIAS_LEVEL_3 = 0x13, HBIAS_LEVEL_4 = 0x17, HBIAS_LEVEL_5 = 0x19, }; static int __ac101_get_hmic_data(struct snd_soc_codec *codec) { #ifdef AC101_DEBG static long counter; #endif int r, d; d = GET_HMIC_DATA(ac101_read(codec, HMIC_STS)); r = 0x1 << HMIC_DATA_PEND; ac101_write(codec, HMIC_STS, r); /* prevent i2c accessing too frequently */ usleep_range(1500, 3000); AC101_DBG("HMIC_DATA(%3ld): %02X\n", counter++, d); return d; } /* * work_cb_earphone_switch: judge the status of the headphone */ static void work_cb_earphone_switch(struct work_struct *work) { struct ac10x_priv *ac10x = container_of(work, struct ac10x_priv, work_switch); struct snd_soc_codec *codec = ac10x->codec; static int hook_flag1 = 0, hook_flag2 = 0; static int KEY_VOLUME_FLAG = 0; unsigned filter_buf = 0; int filt_index = 0; int t = 0; ac10x->irq_cntr--; /* read HMIC_DATA */ t = __ac101_get_hmic_data(codec); if ((t >= HBIAS_LEVEL_2) && (ac10x->mode == FOUR_HEADPHONE_PLUGIN)) { t = __ac101_get_hmic_data(codec); if (t >= HBIAS_LEVEL_5){ msleep(150); t = __ac101_get_hmic_data(codec); if (((t < HBIAS_LEVEL_2 && t >= HBIAS_LEVEL_1 - 1) || t >= HBIAS_LEVEL_5) && (ac10x->pullout_cntr == 0)) { input_report_key(ac10x->inpdev, KEY_HEADSETHOOK, 1); input_sync(ac10x->inpdev); AC101_DBG("KEY_HEADSETHOOK1\n"); if (hook_flag1 != hook_flag2) hook_flag1 = hook_flag2 = 0; hook_flag1++; } if (ac10x->pullout_cntr) ac10x->pullout_cntr--; } else if (t >= HBIAS_LEVEL_4) { msleep(80); t = __ac101_get_hmic_data(codec); if (t < HBIAS_LEVEL_5 && t >= HBIAS_LEVEL_4 && (ac10x->pullout_cntr == 0)) { KEY_VOLUME_FLAG = 1; input_report_key(ac10x->inpdev, KEY_VOLUMEUP, 1); input_sync(ac10x->inpdev); input_report_key(ac10x->inpdev, KEY_VOLUMEUP, 0); input_sync(ac10x->inpdev); AC101_DBG("HMIC_DATA: %d KEY_VOLUMEUP\n", t); } if (ac10x->pullout_cntr) ac10x->pullout_cntr--; } else if (t >= HBIAS_LEVEL_3){ msleep(80); t = __ac101_get_hmic_data(codec); if (t < HBIAS_LEVEL_4 && t >= HBIAS_LEVEL_3 && (ac10x->pullout_cntr == 0)){ KEY_VOLUME_FLAG = 1; input_report_key(ac10x->inpdev, KEY_VOLUMEDOWN, 1); input_sync(ac10x->inpdev); input_report_key(ac10x->inpdev, KEY_VOLUMEDOWN, 0); input_sync(ac10x->inpdev); AC101_DBG("KEY_VOLUMEDOWN\n"); } if (ac10x->pullout_cntr) ac10x->pullout_cntr--; } } else if ((t < HBIAS_LEVEL_2 && t >= HBIAS_LEVEL_1) && (ac10x->mode == FOUR_HEADPHONE_PLUGIN)) { t = __ac101_get_hmic_data(codec); if (t < HBIAS_LEVEL_2 && t >= HBIAS_LEVEL_1) { if (KEY_VOLUME_FLAG) { KEY_VOLUME_FLAG = 0; } if (hook_flag1 == (++hook_flag2)) { hook_flag1 = hook_flag2 = 0; input_report_key(ac10x->inpdev, KEY_HEADSETHOOK, 0); input_sync(ac10x->inpdev); AC101_DBG("KEY_HEADSETHOOK0\n"); } } } else { while (ac10x->irq_cntr == 0 && ac10x->irq != 0) { msleep(20); t = __ac101_get_hmic_data(codec); if (filt_index <= HEADSET_FILTER_CNT) { if (filt_index++ == 0) { filter_buf = t; } else if (filter_buf != t) { filt_index = 0; } continue; } filt_index = 0; if (filter_buf >= HBIAS_LEVEL_2) { ac10x->mode = THREE_HEADPHONE_PLUGIN; ac10x->state = 2; } else if (filter_buf >= HBIAS_LEVEL_1 - 1) { ac10x->mode = FOUR_HEADPHONE_PLUGIN; ac10x->state = 1; } else { ac10x->mode = HEADPHONE_IDLE; ac10x->state = 0; } switch_status_update(ac10x); ac10x->pullout_cntr = 0; break; } } } /* * audio_hmic_irq: the interrupt handlers */ static irqreturn_t audio_hmic_irq(int irq, void *para) { struct ac10x_priv *ac10x = (struct ac10x_priv *)para; if (ac10x == NULL) { return -EINVAL; } if (0 == schedule_work(&ac10x->work_clear_irq)){ AC101_DBG("[audio_hmic_irq] work already in queue_codec_irq, adding failed!\n"); } return IRQ_HANDLED; } static int ac101_switch_probe(struct ac10x_priv *ac10x) { struct i2c_client *i2c = ac10x->i2c101; long ret; ac10x->gpiod_irq = devm_gpiod_get_optional(&i2c->dev, "switch-irq", GPIOD_IN); if (IS_ERR(ac10x->gpiod_irq)) { ac10x->gpiod_irq = NULL; dev_err(&i2c->dev, "failed get switch-irq in device tree\n"); goto _err_irq; } gpiod_direction_input(ac10x->gpiod_irq); ac10x->irq = gpiod_to_irq(ac10x->gpiod_irq); if (IS_ERR_VALUE(ac10x->irq)) { pr_warn("[ac101] map gpio to irq failed, errno = %ld\n", ac10x->irq); ac10x->irq = 0; goto _err_irq; } /* request irq, set irq type to falling edge trigger */ ret = devm_request_irq(ac10x->codec->dev, ac10x->irq, audio_hmic_irq, IRQF_TRIGGER_FALLING, "SWTICH_EINT", ac10x); if (IS_ERR_VALUE(ret)) { pr_warn("[ac101] request virq %ld failed, errno = %ld\n", ac10x->irq, ret); goto _err_irq; } ac10x->mode = HEADPHONE_IDLE; ac10x->state = -1; /*use for judge the state of switch*/ INIT_WORK(&ac10x->work_switch, work_cb_earphone_switch); INIT_WORK(&ac10x->work_clear_irq, work_cb_clear_irq); /********************create input device************************/ ac10x->inpdev = devm_input_allocate_device(ac10x->codec->dev); if (!ac10x->inpdev) { AC101_DBG("input_allocate_device: not enough memory for input device\n"); ret = -ENOMEM; goto _err_input_allocate_device; } ac10x->inpdev->name = "seed-voicecard-headset"; ac10x->inpdev->phys = dev_name(ac10x->codec->dev); ac10x->inpdev->id.bustype = BUS_I2C; ac10x->inpdev->dev.parent = ac10x->codec->dev; input_set_drvdata(ac10x->inpdev, ac10x->codec); ac10x->inpdev->evbit[0] = BIT_MASK(EV_KEY) | BIT(EV_SW); set_bit(KEY_HEADSETHOOK, ac10x->inpdev->keybit); set_bit(KEY_VOLUMEUP, ac10x->inpdev->keybit); set_bit(KEY_VOLUMEDOWN, ac10x->inpdev->keybit); input_set_capability(ac10x->inpdev, EV_SW, SW_HEADPHONE_INSERT); ret = input_register_device(ac10x->inpdev); if (ret) { AC101_DBG("input_register_device: input_register_device failed\n"); goto _err_input_register_device; } /* the first headset state checking */ switch_hw_config(ac10x->codec); ac10x->irq_cntr = 1; schedule_work(&ac10x->work_switch); return 0; _err_input_register_device: _err_input_allocate_device: if (ac10x->irq) { devm_free_irq(&i2c->dev, ac10x->irq, ac10x); ac10x->irq = 0; } _err_irq: return ret; } /******************************************************************************/ /********************************switch****************************************/ /******************************************************************************/ #endif void drc_config(struct snd_soc_codec *codec) { int reg_val; reg_val = ac101_read(codec, 0xa3); reg_val &= ~(0x7ff<<0); reg_val |= 1<<0; ac101_write(codec, 0xa3, reg_val); ac101_write(codec, 0xa4, 0x2baf); reg_val = ac101_read(codec, 0xa5); reg_val &= ~(0x7ff<<0); reg_val |= 1<<0; ac101_write(codec, 0xa5, reg_val); ac101_write(codec, 0xa6, 0x2baf); reg_val = ac101_read(codec, 0xa7); reg_val &= ~(0x7ff<<0); ac101_write(codec, 0xa7, reg_val); ac101_write(codec, 0xa8, 0x44a); reg_val = ac101_read(codec, 0xa9); reg_val &= ~(0x7ff<<0); ac101_write(codec, 0xa9, reg_val); ac101_write(codec, 0xaa, 0x1e06); reg_val = ac101_read(codec, 0xab); reg_val &= ~(0x7ff<<0); reg_val |= (0x352<<0); ac101_write(codec, 0xab, reg_val); ac101_write(codec, 0xac, 0x6910); reg_val = ac101_read(codec, 0xad); reg_val &= ~(0x7ff<<0); reg_val |= (0x77a<<0); ac101_write(codec, 0xad, reg_val); ac101_write(codec, 0xae, 0xaaaa); reg_val = ac101_read(codec, 0xaf); reg_val &= ~(0x7ff<<0); reg_val |= (0x2de<<0); ac101_write(codec, 0xaf, reg_val); ac101_write(codec, 0xb0, 0xc982); ac101_write(codec, 0x16, 0x9f9f); } void drc_enable(struct snd_soc_codec *codec,bool on) { int reg_val; if (on) { ac101_write(codec, 0xb5, 0xA080); reg_val = ac101_read(codec, MOD_CLK_ENA); reg_val |= (0x1<<6); ac101_write(codec, MOD_CLK_ENA, reg_val); reg_val = ac101_read(codec, MOD_RST_CTRL); reg_val |= (0x1<<6); ac101_write(codec, MOD_RST_CTRL, reg_val); reg_val = ac101_read(codec, 0xa0); reg_val |= (0x7<<0); ac101_write(codec, 0xa0, reg_val); } else { ac101_write(codec, 0xb5, 0x0); reg_val = ac101_read(codec, MOD_CLK_ENA); reg_val &= ~(0x1<<6); ac101_write(codec, MOD_CLK_ENA, reg_val); reg_val = ac101_read(codec, MOD_RST_CTRL); reg_val &= ~(0x1<<6); ac101_write(codec, MOD_RST_CTRL, reg_val); reg_val = ac101_read(codec, 0xa0); reg_val &= ~(0x7<<0); ac101_write(codec, 0xa0, reg_val); } } void set_configuration(struct snd_soc_codec *codec) { if (speaker_double_used) { ac101_update_bits(codec, SPKOUT_CTRL, (0x1f<dac_mutex); switch (event) { case SND_SOC_DAPM_PRE_PMU: AC101_DBG(); if (ac10x->dac_enable == 0){ /*enable dac module clk*/ ac101_update_bits(codec, MOD_CLK_ENA, (0x1<dac_enable++; break; case SND_SOC_DAPM_POST_PMD: if (ac10x->dac_enable != 0){ ac10x->dac_enable = 0; ac101_update_bits(codec, DAC_DIG_CTRL, (0x1<dac_mutex); return 0; } static int ac101_headphone_event(struct snd_soc_codec* codec, int event) { switch (event) { case SND_SOC_DAPM_POST_PMU: /*open*/ AC101_DBG("post:open\n"); ac101_update_bits(codec, OMIXER_DACA_CTRL, (0xf<codec, SYSCLK_CTRL); return (reg_val & (0x1<aif1_clken == 0){ ret = ac101_update_bits(codec, SYSCLK_CTRL, (0x1<aif1_clken++; } } break; case SND_SOC_DAPM_POST_PMD: if (ac10x->aif1_clken != 0) { /* disable aif1clk & sysclk */ ret = ac101_update_bits(codec, SYSCLK_CTRL, (0x1<aif1_clken = 0; } break; } } AC101_DBG("event=%d pre_up/%d post_down/%d\n", event, SND_SOC_DAPM_PRE_PMU, SND_SOC_DAPM_POST_PMD); return ret; } /** * snd_ac101_get_volsw - single mixer get callback * @kcontrol: mixer control * @ucontrol: control element information * * Callback to get the value of a single mixer control, or a double mixer * control that spans 2 registers. * * Returns 0 for success. */ static int snd_ac101_get_volsw(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol ){ struct soc_mixer_control *mc = (struct soc_mixer_control *)kcontrol->private_value; unsigned int val, mask = (1 << fls(mc->max)) - 1; unsigned int invert = mc->invert; int ret; if ((ret = ac101_read(static_ac10x->codec, mc->reg)) < 0) return ret; val = (ret >> mc->shift) & mask; ucontrol->value.integer.value[0] = val - mc->min; if (invert) { ucontrol->value.integer.value[0] = mc->max - ucontrol->value.integer.value[0]; } if (snd_soc_volsw_is_stereo(mc)) { val = (ret >> mc->rshift) & mask; ucontrol->value.integer.value[1] = val - mc->min; if (invert) { ucontrol->value.integer.value[1] = mc->max - ucontrol->value.integer.value[1]; } } return 0; } /** * snd_ac101_put_volsw - single mixer put callback * @kcontrol: mixer control * @ucontrol: control element information * * Callback to set the value of a single mixer control, or a double mixer * control that spans 2 registers. * * Returns 0 for success. */ static int snd_ac101_put_volsw(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol ){ struct soc_mixer_control *mc = (struct soc_mixer_control *)kcontrol->private_value; unsigned int sign_bit = mc->sign_bit; unsigned int val, mask = (1 << fls(mc->max)) - 1; unsigned int invert = mc->invert; int ret; if (sign_bit) mask = BIT(sign_bit + 1) - 1; val = ((ucontrol->value.integer.value[0] + mc->min) & mask); if (invert) { val = mc->max - val; } ret = ac101_update_bits(static_ac10x->codec, mc->reg, mask << mc->shift, val << mc->shift); if (! snd_soc_volsw_is_stereo(mc)) { return ret; } val = ((ucontrol->value.integer.value[1] + mc->min) & mask); if (invert) { val = mc->max - val; } ret = ac101_update_bits(static_ac10x->codec, mc->reg, mask << mc->rshift, val << mc->rshift); return ret; } static const DECLARE_TLV_DB_SCALE(dac_vol_tlv, -11925, 75, 0); static const DECLARE_TLV_DB_SCALE(dac_mix_vol_tlv, -600, 600, 0); static const DECLARE_TLV_DB_SCALE(dig_vol_tlv, -7308, 116, 0); static const DECLARE_TLV_DB_SCALE(speaker_vol_tlv, -4800, 150, 0); static const DECLARE_TLV_DB_SCALE(headphone_vol_tlv, -6300, 100, 0); static struct snd_kcontrol_new ac101_controls[] = { /*DAC*/ SOC_DOUBLE_TLV("DAC volume", DAC_VOL_CTRL, DAC_VOL_L, DAC_VOL_R, 0xff, 0, dac_vol_tlv), SOC_DOUBLE_TLV("DAC mixer gain", DAC_MXR_GAIN, DACL_MXR_GAIN, DACR_MXR_GAIN, 0xf, 0, dac_mix_vol_tlv), SOC_SINGLE_TLV("digital volume", DAC_DBG_CTRL, DVC, 0x3f, 1, dig_vol_tlv), SOC_SINGLE_TLV("Speaker Playback Volume", SPKOUT_CTRL, SPK_VOL, 0x1f, 0, speaker_vol_tlv), SOC_SINGLE_TLV("Headphone Playback Volume", HPOUT_CTRL, HP_VOL, 0x3f, 0, headphone_vol_tlv), }; /* PLL divisors */ struct pll_div { unsigned int pll_in; unsigned int pll_out; int m; int n_i; int n_f; }; struct aif1_fs { unsigned samp_rate; int bclk_div; int srbit; #define _SERIES_24_576K 0 #define _SERIES_22_579K 1 int series; }; struct kv_map { int val; int bit; }; /* * Note : pll code from original tdm/i2s driver. * freq_out = freq_in * N/(M*(2k+1)) , k=1,N=N_i+N_f,N_f=factor*0.2; * N_i[0,1023], N_f_factor[0,7], m[1,64]=REG_VAL[1-63,0] */ static const struct pll_div codec_pll_div[] = { {128000, _FREQ_22_579K, 1, 529, 1}, {192000, _FREQ_22_579K, 1, 352, 4}, {256000, _FREQ_22_579K, 1, 264, 3}, {384000, _FREQ_22_579K, 1, 176, 2}, /*((176+2*0.2)*6000000)/(38*(2*1+1))*/ {1411200, _FREQ_22_579K, 1, 48, 0}, {2822400, _FREQ_22_579K, 1, 24, 0}, /* accurate, 11025 * 256 */ {5644800, _FREQ_22_579K, 1, 12, 0}, /* accurate, 22050 * 256 */ {6000000, _FREQ_22_579K, 38, 429, 0}, /*((429+0*0.2)*6000000)/(38*(2*1+1))*/ {11289600, _FREQ_22_579K, 1, 6, 0}, /* accurate, 44100 * 256 */ {13000000, _FREQ_22_579K, 19, 99, 0}, {19200000, _FREQ_22_579K, 25, 88, 1}, {24000000, _FREQ_22_579K, 63, 177, 4}, /* 22577778 Hz */ {128000, _FREQ_24_576K, 1, 576, 0}, {192000, _FREQ_24_576K, 1, 384, 0}, {256000, _FREQ_24_576K, 1, 288, 0}, {384000, _FREQ_24_576K, 1, 192, 0}, {2048000, _FREQ_24_576K, 1, 36, 0}, /* accurate, 8000 * 256 */ {3072000, _FREQ_24_576K, 1, 24, 0}, /* accurate, 12000 * 256 */ {4096000, _FREQ_24_576K, 1, 18, 0}, /* accurate, 16000 * 256 */ {6000000, _FREQ_24_576K, 25, 307, 1}, {6144000, _FREQ_24_576K, 4, 48, 0}, /* accurate, 24000 * 256 */ {12288000, _FREQ_24_576K, 8, 48, 0}, /* accurate, 48000 * 256 */ {13000000, _FREQ_24_576K, 42, 238, 1}, {19200000, _FREQ_24_576K, 25, 96, 0}, {24000000, _FREQ_24_576K, 25, 76, 4}, /* accurate */ {_FREQ_22_579K, _FREQ_22_579K, 8, 24, 0}, /* accurate, 88200 * 256 */ {_FREQ_24_576K, _FREQ_24_576K, 8, 24, 0}, /* accurate, 96000 * 256 */ }; static const struct aif1_fs codec_aif1_fs[] = { {8000, 12, 0}, {11025, 8, 1, _SERIES_22_579K}, {12000, 8, 2}, {16000, 6, 3}, {22050, 4, 4, _SERIES_22_579K}, {24000, 4, 5}, /* {32000, 3, 6}, dividing by 3 is not support */ {44100, 2, 7, _SERIES_22_579K}, {48000, 2, 8}, {96000, 1, 9}, }; static const struct kv_map codec_aif1_lrck[] = { {16, 0}, {32, 1}, {64, 2}, {128, 3}, {256, 4}, }; static const struct kv_map codec_aif1_wsize[] = { {8, 0}, {16, 1}, {20, 2}, {24, 3}, {32, 3}, }; static const unsigned ac101_bclkdivs[] = { 1, 2, 4, 6, 8, 12, 16, 24, 32, 48, 64, 96, 128, 192, 0, 0, }; static int ac101_aif_play(struct ac10x_priv* ac10x) { struct snd_soc_codec * codec = ac10x->codec; late_enable_dac(codec, SND_SOC_DAPM_PRE_PMU); ac101_headphone_event(codec, SND_SOC_DAPM_POST_PMU); if (drc_used) { drc_enable(codec, 1); } /* Enable Left & Right Speaker */ ac101_update_bits(codec, SPKOUT_CTRL, (0x1 << LSPK_EN) | (0x1 << RSPK_EN), (0x1 << LSPK_EN) | (0x1 << RSPK_EN)); if (ac10x->gpiod_spk_amp_gate) { gpiod_set_value(ac10x->gpiod_spk_amp_gate, 1); } return 0; } static void ac10x_work_aif_play(struct work_struct *work) { struct ac10x_priv *ac10x = container_of(work, struct ac10x_priv, dlywork.work); ac101_aif_play(ac10x); return; } int ac101_aif_mute(struct snd_soc_dai *codec_dai, int mute) { struct snd_soc_codec *codec = codec_dai->codec; struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); AC101_DBG("mute=%d\n", mute); ac101_write(codec, DAC_VOL_CTRL, mute? 0: 0xA0A0); if (!mute) { #if _MASTER_MULTI_CODEC != _MASTER_AC101 /* enable global clock */ ac10x->aif1_clken = 0; ac101_aif1clk(codec, SND_SOC_DAPM_PRE_PMU, 0); ac101_aif_play(ac10x); #else schedule_delayed_work(&ac10x->dlywork, msecs_to_jiffies(50)); #endif } else { #if _MASTER_MULTI_CODEC == _MASTER_AC101 cancel_delayed_work_sync(&ac10x->dlywork); #endif if (ac10x->gpiod_spk_amp_gate) { gpiod_set_value(ac10x->gpiod_spk_amp_gate, 0); } /* Disable Left & Right Speaker */ ac101_update_bits(codec, SPKOUT_CTRL, (0x1 << LSPK_EN) | (0x1 << RSPK_EN), (0x0 << LSPK_EN) | (0x0 << RSPK_EN)); if (drc_used) { drc_enable(codec, 0); } ac101_headphone_event(codec, SND_SOC_DAPM_PRE_PMD); late_enable_dac(codec, SND_SOC_DAPM_POST_PMD); #if _MASTER_MULTI_CODEC != _MASTER_AC101 ac10x->aif1_clken = 1; ac101_aif1clk(codec, SND_SOC_DAPM_POST_PMD, 0); #endif } return 0; } void ac101_aif_shutdown(struct snd_pcm_substream *substream, struct snd_soc_dai *codec_dai) { struct snd_soc_codec *codec = codec_dai->codec; struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); AC101_DBG("stream = %s, play: %d, capt: %d, active: %d\n", snd_pcm_stream_str(substream), codec_dai->stream[SNDRV_PCM_STREAM_PLAYBACK].active, codec_dai->stream[SNDRV_PCM_STREAM_CAPTURE].active, snd_soc_dai_active(codec_dai)); if (!snd_soc_dai_active(codec_dai)) { ac10x->aif1_clken = 1; ac101_aif1clk(codec, SND_SOC_DAPM_POST_PMD, 0); } else { ac101_aif1clk(codec, SND_SOC_DAPM_PRE_PMU, 0); } } static int ac101_set_pll(struct snd_soc_dai *codec_dai, int pll_id, int source, unsigned int freq_in, unsigned int freq_out) { struct snd_soc_codec *codec = codec_dai->codec; int i, m, n_i, n_f; AC101_DBG("pll_id:%d\n", pll_id); /* clear volatile reserved bits*/ ac101_update_bits(codec, SYSCLK_CTRL, 0xFF & ~(0x1 << SYSCLK_ENA), 0x0); /* select aif1 clk srouce from mclk1 */ ac101_update_bits(codec, SYSCLK_CTRL, (0x3< _FREQ_24_576K)) { return -EINVAL; } else if ((freq_in == _FREQ_24_576K) || (freq_in == _FREQ_22_579K)) { if (pll_id == AC101_MCLK1) { /*select aif1 clk source from mclk1*/ ac101_update_bits(codec, SYSCLK_CTRL, (0x3<codec; struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); int reg_val, freq_out; unsigned channels; AC101_DBG("+++\n"); if (_MASTER_MULTI_CODEC == _MASTER_AC101 && ac101_sysclk_started()) { /* not configure hw_param twice if stream is playback, tell the caller it's started */ if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { return 1; } } /* get channels count & slot size */ channels = params_channels(params); switch (params_format(params)) { case SNDRV_PCM_FORMAT_S24_LE: case SNDRV_PCM_FORMAT_S32_LE: aif1_slot_size = 32; break; case SNDRV_PCM_FORMAT_S16_LE: default: aif1_slot_size = 16; break; } /* set LRCK/BCLK ratio */ aif1_lrck_div = aif1_slot_size * channels; for (i = 0; i < ARRAY_SIZE(codec_aif1_lrck); i++) { if (codec_aif1_lrck[i].val == aif1_lrck_div) { break; } } ac101_update_bits(codec, AIF_CLK_CTRL, (0x7<stream[SNDRV_PCM_STREAM_CAPTURE].active && dmic_used && codec_aif1_fs[i].samp_rate == 44100) { ac101_update_bits(codec, AIF_SR_CTRL, (0xf< 2) reg_val = 2; ac101_update_bits(codec, AIF1_ADCDAT_CTRL, 0x3 << AIF1_SLOT_SIZ, reg_val << AIF1_SLOT_SIZ); /* setting pll if it's master mode */ reg_val = ac101_read(codec, AIF_CLK_CTRL); if ((reg_val & (0x1 << AIF1_MSTR_MOD)) == 0) { unsigned bclkdiv; ac101_set_pll(codec_dai, AC101_MCLK1, 0, ac10x->sysclk, freq_out); bclkdiv = freq_out / (aif1_lrck_div * params_rate(params)); for (i = 0; i < ARRAY_SIZE(ac101_bclkdivs) - 1; i++) { if (ac101_bclkdivs[i] >= bclkdiv) { break; } } ac101_update_bits(codec, AIF_CLK_CTRL, (0xf<codec; AC101_DBG(); /* * master or slave selection * 0 = Master mode * 1 = Slave mode */ reg_val = ac101_read(codec, AIF_CLK_CTRL); reg_val &= ~(0x1<codec; AC101_DBG("\n\n\n"); if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) { } return 0; } #if _MASTER_MULTI_CODEC == _MASTER_AC101 static int ac101_set_clock(int y_start_n_stop, struct snd_pcm_substream *substream, int cmd, struct snd_soc_dai *dai) { int r; if (y_start_n_stop) { /* enable global clock */ r = ac101_aif1clk(static_ac10x->codec, SND_SOC_DAPM_PRE_PMU, 1); } else { /* disable global clock */ static_ac10x->aif1_clken = 1; r = ac101_aif1clk(static_ac10x->codec, SND_SOC_DAPM_POST_PMD, 0); } return r; } #endif int ac101_trigger(struct snd_pcm_substream *substream, int cmd, struct snd_soc_dai *dai) { struct snd_soc_codec *codec = dai->codec; struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); int ret = 0; unsigned long flags; AC101_DBG("stream=%s cmd=%d\n", snd_pcm_stream_str(substream), cmd); switch (cmd) { case SNDRV_PCM_TRIGGER_START: case SNDRV_PCM_TRIGGER_RESUME: case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: #if _MASTER_MULTI_CODEC == _MASTER_AC101 spin_lock_irqsave(&ac10x->lock, flags); if (ac10x->aif1_clken == 0){ /* * enable aif1clk, it' here due to reduce time between 'AC108 Sysclk Enable' and 'AC101 Sysclk Enable' * Or else the two AC108 chips lost the sync. */ ret = 0; ret = ret || ac101_update_bits(codec, MOD_CLK_ENA, (0x1<lock, flags); ac101_set_clock(1, substream, cmd, dai); #endif break; case SNDRV_PCM_TRIGGER_STOP: case SNDRV_PCM_TRIGGER_SUSPEND: case SNDRV_PCM_TRIGGER_PAUSE_PUSH: ac101_set_clock(0, NULL, 0, NULL); break; default: ret = -EINVAL; } AC101_DBG("stream=%s cmd=%d;finished %d\n", snd_pcm_stream_str(substream), cmd, ret); return ret; } #if 0 static int ac101_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 ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); AC101_DBG("id=%d freq=%d, dir=%d\n", clk_id, freq, dir); ac10x->sysclk = freq; return 0; } static const struct snd_soc_dai_ops ac101_aif1_dai_ops = { //.startup = ac101_audio_startup, //.shutdown = ac101_aif_shutdown, //.set_sysclk = ac101_set_dai_sysclk, //.set_pll = ac101_set_pll, //.set_fmt = ac101_set_dai_fmt, //.hw_params = ac101_hw_params, //.trigger = ac101_trigger, //.digital_mute = ac101_aif_mute, }; static struct snd_soc_dai_driver ac101_dai[] = { { .name = "ac10x-aif1", .id = AIF1_CLK, .playback = { .stream_name = "Playback", .channels_min = 1, .channels_max = 8, .rates = AC101_RATES, .formats = AC101_FORMATS, }, #if 0 .capture = { .stream_name = "Capture", .channels_min = 1, .channels_max = 8, .rates = AC101_RATES, .formats = AC101_FORMATS, }, #endif .ops = &ac101_aif1_dai_ops, } }; #endif static void codec_resume_work(struct work_struct *work) { struct ac10x_priv *ac10x = container_of(work, struct ac10x_priv, codec_resume); struct snd_soc_codec *codec = ac10x->codec; AC101_DBG("+++\n"); set_configuration(codec); if (drc_used) { drc_config(codec); } /*enable this bit to prevent leakage from ldoin*/ ac101_update_bits(codec, ADDA_TUNE3, (0x1<bias_level = level; return 0; } int ac101_codec_probe(struct snd_soc_codec *codec) { int ret = 0; struct ac10x_priv *ac10x; ac10x = dev_get_drvdata(codec->dev); if (ac10x == NULL) { AC101_DBG("not set client data!\n"); return -ENOMEM; } ac10x->codec = codec; INIT_DELAYED_WORK(&ac10x->dlywork, ac10x_work_aif_play); INIT_WORK(&ac10x->codec_resume, codec_resume_work); ac10x->dac_enable = 0; ac10x->aif1_clken = 0; mutex_init(&ac10x->dac_mutex); #if _MASTER_MULTI_CODEC == _MASTER_AC101 seeed_voice_card_register_set_clock(SNDRV_PCM_STREAM_PLAYBACK, ac101_set_clock); #endif set_configuration(ac10x->codec); /*enable this bit to prevent leakage from ldoin*/ ac101_update_bits(codec, ADDA_TUNE3, (0x1<get = snd_ac101_get_volsw; skn->put = snd_ac101_put_volsw; } ret = snd_soc_add_codec_controls(codec, ac101_controls, ARRAY_SIZE(ac101_controls)); if (ret) { pr_err("[ac10x] Failed to register audio mode control, " "will continue without it.\n"); } #ifdef CONFIG_AC101_SWITCH_DETECT ret = ac101_switch_probe(ac10x); if (ret) { // not care the switch return value } #endif return 0; } /* power down chip */ int ac101_codec_remove(struct snd_soc_codec *codec) { #ifdef CONFIG_AC101_SWITCH_DETECT struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); if (ac10x->irq) { devm_free_irq(codec->dev, ac10x->irq, ac10x); ac10x->irq = 0; } if (cancel_work_sync(&ac10x->work_switch) != 0) { } if (cancel_work_sync(&ac10x->work_clear_irq) != 0) { } if (ac10x->inpdev) { input_unregister_device(ac10x->inpdev); ac10x->inpdev = NULL; } #endif return 0; } int ac101_codec_suspend(struct snd_soc_codec *codec) { struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); AC101_DBG("[codec]:suspend\n"); regcache_cache_only(ac10x->regmap101, true); return 0; } int ac101_codec_resume(struct snd_soc_codec *codec) { struct ac10x_priv *ac10x = snd_soc_codec_get_drvdata(codec); int ret; AC101_DBG("[codec]:resume"); /* Sync reg_cache with the hardware */ regcache_cache_only(ac10x->regmap101, false); ret = regcache_sync(ac10x->regmap101); if (ret != 0) { dev_err(codec->dev, "Failed to sync register cache: %d\n", ret); regcache_cache_only(ac10x->regmap101, true); return ret; } #ifdef CONFIG_AC101_SWITCH_DETECT ac10x->mode = HEADPHONE_IDLE; ac10x->state = -1; #endif ac101_set_bias_level(codec, SND_SOC_BIAS_STANDBY); schedule_work(&ac10x->codec_resume); return 0; } /***************************************************************************/ static ssize_t ac101_debug_store(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { struct ac10x_priv *ac10x = dev_get_drvdata(dev); int val = 0, flag = 0; u16 value_w, value_r; u8 reg, num, i=0; val = simple_strtol(buf, NULL, 16); flag = (val >> 24) & 0xF; if (flag) { reg = (val >> 16) & 0xFF; value_w = val & 0xFFFF; ac101_write(ac10x->codec, reg, value_w); printk("write 0x%x to reg:0x%x\n", value_w, reg); } else { reg = (val >> 8) & 0xFF; num = val & 0xff; printk("\n"); printk("read:start add:0x%x,count:0x%x\n", reg, num); regcache_cache_bypass(ac10x->regmap101, true); do { value_r = ac101_read(ac10x->codec, reg); printk("0x%x: 0x%04x ", reg++, value_r); if (++i % 4 == 0 || i == num) printk("\n"); } while (i < num); regcache_cache_bypass(ac10x->regmap101, false); } return count; } static ssize_t ac101_debug_show(struct device *dev, struct device_attribute *attr, char *buf) { printk("echo flag|reg|val > ac10x\n"); printk("eg read star addres=0x06,count 0x10:echo 0610 >ac10x\n"); printk("eg write value:0x13fe to address:0x06 :echo 10613fe > ac10x\n"); return 0; } static DEVICE_ATTR(ac10x, 0644, ac101_debug_show, ac101_debug_store); static struct attribute *audio_debug_attrs[] = { &dev_attr_ac10x.attr, NULL, }; static struct attribute_group audio_debug_attr_group = { .name = "ac101_debug", .attrs = audio_debug_attrs, }; /***************************************************************************/ /************************************************************/ static bool ac101_volatile_reg(struct device *dev, unsigned int reg) { switch (reg) { case PLL_CTRL2: case HMIC_STS: return true; } return false; } static const struct regmap_config ac101_regmap = { .reg_bits = 8, .val_bits = 16, .reg_stride = 1, .max_register = 0xB5, .cache_type = REGCACHE_FLAT, .volatile_reg = ac101_volatile_reg, }; /* Sync reg_cache from the hardware */ int ac10x_fill_regcache(struct device* dev, struct regmap* map) { int r, i, n; int v; n = regmap_get_max_register(map); for (i = 0; i < n; i++) { regcache_cache_bypass(map, true); r = regmap_read(map, i, &v); if (r) { dev_err(dev, "failed to read register %d\n", i); continue; } regcache_cache_bypass(map, false); regcache_cache_only(map, true); r = regmap_write(map, i, v); regcache_cache_only(map, false); } regcache_cache_bypass(map, false); regcache_cache_only(map, false); return 0; } int ac101_probe(struct i2c_client *i2c, const struct i2c_device_id *id) { struct ac10x_priv *ac10x = i2c_get_clientdata(i2c); int ret = 0; unsigned v = 0; AC101_DBG(); static_ac10x = ac10x; ac10x->regmap101 = devm_regmap_init_i2c(i2c, &ac101_regmap); if (IS_ERR(ac10x->regmap101)) { ret = PTR_ERR(ac10x->regmap101); dev_err(&i2c->dev, "Fail to initialize I/O: %d\n", ret); return ret; } /* Chip reset */ regcache_cache_only(ac10x->regmap101, false); ret = regmap_write(ac10x->regmap101, CHIP_AUDIO_RST, 0); msleep(50); /* sync regcache for FLAT type */ ac10x_fill_regcache(&i2c->dev, ac10x->regmap101); ret = regmap_read(ac10x->regmap101, CHIP_AUDIO_RST, &v); if (ret < 0) { dev_err(&i2c->dev, "failed to read vendor ID: %d\n", ret); return ret; } if (v != AC101_CHIP_ID) { dev_err(&i2c->dev, "chip is not AC101 (%X)\n", v); dev_err(&i2c->dev, "Expected %X\n", AC101_CHIP_ID); return -ENODEV; } ret = sysfs_create_group(&i2c->dev.kobj, &audio_debug_attr_group); if (ret) { pr_err("failed to create attr group\n"); } ac10x->gpiod_spk_amp_gate = devm_gpiod_get_optional(&i2c->dev, "spk-amp-switch", GPIOD_OUT_LOW); if (IS_ERR(ac10x->gpiod_spk_amp_gate)) { ac10x->gpiod_spk_amp_gate = NULL; dev_err(&i2c->dev, "failed get spk-amp-switch in device tree\n"); } return 0; } void ac101_shutdown(struct i2c_client *i2c) { struct ac10x_priv *ac10x = i2c_get_clientdata(i2c); struct snd_soc_codec *codec = ac10x->codec; int reg_val; if (codec == NULL) { pr_err(": no sound card.\n"); return; } /*set headphone volume to 0*/ reg_val = ac101_read(codec, HPOUT_CTRL); reg_val &= ~(0x3f<dev.kobj, &audio_debug_attr_group); return 0; } MODULE_DESCRIPTION("ASoC ac10x driver"); MODULE_AUTHOR("huangxin,liushaohua"); MODULE_AUTHOR("PeterYang");