Lines Matching +full:adc +full:- +full:channel +full:- +full:clk +full:- +full:src

1 // SPDX-License-Identifier: GPL-2.0-only
8 * Co-author: Meng-Huang Kuo <mhkuo@nuvoton.com>
17 #include <linux/clk.h>
34 #define NUVOTON_CODEC_DAI "nau8825-hifi"
82 /* ratio for input clk freq */
107 { 64, 2 }, /* OSR 64, SRC 1/4 */
108 { 256, 0 }, /* OSR 256, SRC 1 */
109 { 128, 1 }, /* OSR 128, SRC 1/2 */
111 { 32, 3 }, /* OSR 32, SRC 1/8 */
115 { 32, 3 }, /* OSR 32, SRC 1/8 */
116 { 64, 2 }, /* OSR 64, SRC 1/4 */
117 { 128, 1 }, /* OSR 128, SRC 1/2 */
118 { 256, 0 }, /* OSR 256, SRC 1 */
239 * nau8825_sema_acquire - acquire the semaphore of nau88l25
249 * this function returns -ETIME. If the sleep is interrupted by a signal,
250 * this function will return -EINTR. It returns 0 if the semaphore was
262 ret = down_timeout(&nau8825->xtalk_sem, timeout); in nau8825_sema_acquire()
264 dev_warn(nau8825->dev, "Acquire semaphore timeout\n"); in nau8825_sema_acquire()
266 ret = down_trylock(&nau8825->xtalk_sem); in nau8825_sema_acquire()
268 dev_warn(nau8825->dev, "Acquire semaphore fail\n"); in nau8825_sema_acquire()
275 * nau8825_sema_release - release the semaphore of nau88l25
283 up(&nau8825->xtalk_sem); in nau8825_sema_release()
287 * nau8825_sema_reset - reset the semaphore for nau88l25
295 nau8825->xtalk_sem.count = 1; in nau8825_sema_reset()
299 * nau8825_hpvol_ramp - Ramp up the headphone volume change gradually to target level.
306 * The headphone volume is from 0dB to minimum -54dB and -1dB per step.
327 /* only handle volume from 0dB to minimum -54dB */ in nau8825_hpvol_ramp()
335 value = to - volume + from; in nau8825_hpvol_ramp()
336 regmap_update_bits(nau8825->regmap, NAU8825_REG_HSVOL_CTRL, in nau8825_hpvol_ramp()
345 regmap_update_bits(nau8825->regmap, NAU8825_REG_HSVOL_CTRL, in nau8825_hpvol_ramp()
351 * nau8825_intlog10_dec3 - Computes log10 of a value
353 * dvb-math. The source code locates as the following.
354 * Linux/drivers/media/dvb-core/dvb_math.c
365 msb = fls(value) - 1; in nau8825_intlog10_dec3()
373 * 0x00231f56 -> 0x8C7D5800 in nau8825_intlog10_dec3()
374 * the result is y * 2^31 -> "significand" in nau8825_intlog10_dec3()
380 significand = value << (31 - msb); in nau8825_intlog10_dec3()
392 * (error / 0x800000) * (logtable_next - logtable_current) in nau8825_intlog10_dec3()
398 ((logtable[(logentry + 1) & 0xff] - in nau8825_intlog10_dec3()
413 * nau8825_xtalk_sidetone - computes cross talk suppression sidetone gain.
435 gain = (sig_org - sig_cros) * 20 + GAIN_AUGMENT; in nau8825_xtalk_sidetone()
437 gain = (sig_cros - sig_org) * 20 + GAIN_AUGMENT; in nau8825_xtalk_sidetone()
438 sidetone = SIDETONE_BASE - gain * 2; in nau8825_xtalk_sidetone()
451 return -EINVAL; in nau8825_xtalk_baktab_index_by_reg()
458 if (nau8825->xtalk_baktab_initialized) in nau8825_xtalk_backup()
463 regmap_read(nau8825->regmap, nau8825_xtalk_baktab[i].reg, in nau8825_xtalk_backup()
466 nau8825->xtalk_baktab_initialized = true; in nau8825_xtalk_backup()
473 if (!nau8825->xtalk_baktab_initialized) in nau8825_xtalk_restore()
490 regmap_write(nau8825->regmap, nau8825_xtalk_baktab[i].reg, in nau8825_xtalk_restore()
494 nau8825->xtalk_baktab_initialized = false; in nau8825_xtalk_restore()
500 regmap_update_bits(nau8825->regmap, NAU8825_REG_ENA_CTRL, in nau8825_xtalk_prepare_dac()
509 regmap_update_bits(nau8825->regmap, NAU8825_REG_CHARGE_PUMP, in nau8825_xtalk_prepare_dac()
513 regmap_update_bits(nau8825->regmap, NAU8825_REG_RDAC, in nau8825_xtalk_prepare_dac()
518 regmap_update_bits(nau8825->regmap, NAU8825_REG_POWER_UP_CONTROL, in nau8825_xtalk_prepare_dac()
523 regmap_update_bits(nau8825->regmap, NAU8825_REG_POWER_UP_CONTROL, in nau8825_xtalk_prepare_dac()
527 regmap_update_bits(nau8825->regmap, NAU8825_REG_HSD_CTRL, in nau8825_xtalk_prepare_dac()
530 regmap_update_bits(nau8825->regmap, NAU8825_REG_BOOST, in nau8825_xtalk_prepare_dac()
533 regmap_update_bits(nau8825->regmap, NAU8825_REG_CLASSG_CTRL, in nau8825_xtalk_prepare_dac()
540 /* Power up left ADC and raise 5dB than Vmid for Vref */ in nau8825_xtalk_prepare_adc()
541 regmap_update_bits(nau8825->regmap, NAU8825_REG_ANALOG_ADC_2, in nau8825_xtalk_prepare_adc()
549 regmap_write(nau8825->regmap, NAU8825_REG_FLL1, 0x0); in nau8825_xtalk_clock()
550 regmap_write(nau8825->regmap, NAU8825_REG_FLL2, 0x3126); in nau8825_xtalk_clock()
551 regmap_write(nau8825->regmap, NAU8825_REG_FLL3, 0x0008); in nau8825_xtalk_clock()
552 regmap_write(nau8825->regmap, NAU8825_REG_FLL4, 0x0010); in nau8825_xtalk_clock()
553 regmap_write(nau8825->regmap, NAU8825_REG_FLL5, 0x0); in nau8825_xtalk_clock()
554 regmap_write(nau8825->regmap, NAU8825_REG_FLL6, 0x6000); in nau8825_xtalk_clock()
556 regmap_update_bits(nau8825->regmap, NAU8825_REG_CLK_DIVIDER, in nau8825_xtalk_clock()
558 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL6, NAU8825_DCO_EN, in nau8825_xtalk_clock()
563 regmap_update_bits(nau8825->regmap, NAU8825_REG_CLK_DIVIDER, in nau8825_xtalk_clock()
565 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL1, in nau8825_xtalk_clock()
576 regmap_update_bits(nau8825->regmap, NAU8825_REG_I2S_PCM_CTRL2, in nau8825_xtalk_prepare()
584 if (index != -EINVAL) { in nau8825_xtalk_prepare()
592 /* Config channel path and digital gain */ in nau8825_xtalk_prepare()
593 regmap_update_bits(nau8825->regmap, NAU8825_REG_DACL_CTRL, in nau8825_xtalk_prepare()
596 regmap_update_bits(nau8825->regmap, NAU8825_REG_DACR_CTRL, in nau8825_xtalk_prepare()
602 regmap_update_bits(nau8825->regmap, NAU8825_REG_IMM_MODE_CTRL, in nau8825_xtalk_prepare()
608 regmap_update_bits(nau8825->regmap, in nau8825_xtalk_prepare()
611 regmap_update_bits(nau8825->regmap, NAU8825_REG_CHARGE_PUMP, in nau8825_xtalk_prepare()
618 regmap_update_bits(nau8825->regmap, NAU8825_REG_BOOST, in nau8825_xtalk_clean_dac()
621 regmap_update_bits(nau8825->regmap, NAU8825_REG_HSD_CTRL, in nau8825_xtalk_clean_dac()
625 regmap_update_bits(nau8825->regmap, NAU8825_REG_CHARGE_PUMP, in nau8825_xtalk_clean_dac()
629 regmap_update_bits(nau8825->regmap, NAU8825_REG_BIAS_ADJ, in nau8825_xtalk_clean_dac()
633 regmap_update_bits(nau8825->regmap, NAU8825_REG_POWER_UP_CONTROL, in nau8825_xtalk_clean_dac()
635 regmap_update_bits(nau8825->regmap, NAU8825_REG_POWER_UP_CONTROL, in nau8825_xtalk_clean_dac()
639 regmap_update_bits(nau8825->regmap, NAU8825_REG_RDAC, in nau8825_xtalk_clean_dac()
642 regmap_update_bits(nau8825->regmap, NAU8825_REG_CHARGE_PUMP, in nau8825_xtalk_clean_dac()
645 regmap_update_bits(nau8825->regmap, NAU8825_REG_ENA_CTRL, in nau8825_xtalk_clean_dac()
648 if (!nau8825->irq) in nau8825_xtalk_clean_dac()
649 regmap_update_bits(nau8825->regmap, in nau8825_xtalk_clean_dac()
655 /* Power down left ADC and restore voltage to Vmid */ in nau8825_xtalk_clean_adc()
656 regmap_update_bits(nau8825->regmap, NAU8825_REG_ANALOG_ADC_2, in nau8825_xtalk_clean_adc()
667 regmap_write(nau8825->regmap, NAU8825_REG_IMM_MODE_CTRL, 0); in nau8825_xtalk_clean()
669 regmap_update_bits(nau8825->regmap, NAU8825_REG_INTERRUPT_MASK, in nau8825_xtalk_clean()
672 regmap_update_bits(nau8825->regmap, NAU8825_REG_I2S_PCM_CTRL2, in nau8825_xtalk_clean()
681 /* Apply ADC volume for better cross talk performance */ in nau8825_xtalk_imm_start()
682 regmap_update_bits(nau8825->regmap, NAU8825_REG_ADC_DGAIN_CTRL, in nau8825_xtalk_imm_start()
684 /* Disables JKTIP(HPL) DAC channel for right to left measurement. in nau8825_xtalk_imm_start()
687 regmap_update_bits(nau8825->regmap, NAU8825_REG_BIAS_ADJ, in nau8825_xtalk_imm_start()
690 switch (nau8825->xtalk_state) { in nau8825_xtalk_imm_start()
693 regmap_update_bits(nau8825->regmap, NAU8825_REG_BIAS_ADJ, in nau8825_xtalk_imm_start()
699 regmap_update_bits(nau8825->regmap, NAU8825_REG_BIAS_ADJ, in nau8825_xtalk_imm_start()
708 regmap_update_bits(nau8825->regmap, NAU8825_REG_IMM_MODE_CTRL, in nau8825_xtalk_imm_start()
715 regmap_update_bits(nau8825->regmap, in nau8825_xtalk_imm_stop()
722 * sending a 23Hz -24dBV sine wave into the headset output DAC and through
725 * an ADC which converts the measurement to a binary code. With two separated
743 switch (nau8825->xtalk_state) { in nau8825_xtalk_measure()
745 /* In prepare state, set up clock, intrruption, DAC path, ADC in nau8825_xtalk_measure()
751 nau8825->xtalk_state = NAU8825_XTALK_HPR_R2L; in nau8825_xtalk_measure()
758 regmap_read(nau8825->regmap, NAU8825_REG_IMM_RMS_L, in nau8825_xtalk_measure()
759 &nau8825->imp_rms[NAU8825_XTALK_HPR_R2L]); in nau8825_xtalk_measure()
760 dev_dbg(nau8825->dev, "HPR_R2L imm: %x\n", in nau8825_xtalk_measure()
761 nau8825->imp_rms[NAU8825_XTALK_HPR_R2L]); in nau8825_xtalk_measure()
762 /* Disable then re-enable IMM mode to update */ in nau8825_xtalk_measure()
765 nau8825->xtalk_state = NAU8825_XTALK_HPL_R2L; in nau8825_xtalk_measure()
775 regmap_read(nau8825->regmap, NAU8825_REG_IMM_RMS_L, in nau8825_xtalk_measure()
776 &nau8825->imp_rms[NAU8825_XTALK_HPL_R2L]); in nau8825_xtalk_measure()
777 dev_dbg(nau8825->dev, "HPL_R2L imm: %x\n", in nau8825_xtalk_measure()
778 nau8825->imp_rms[NAU8825_XTALK_HPL_R2L]); in nau8825_xtalk_measure()
781 nau8825->xtalk_state = NAU8825_XTALK_IMM; in nau8825_xtalk_measure()
785 * signal level vlues are ready. The side tone gain is deter- in nau8825_xtalk_measure()
790 nau8825->imp_rms[NAU8825_XTALK_HPR_R2L], in nau8825_xtalk_measure()
791 nau8825->imp_rms[NAU8825_XTALK_HPL_R2L]); in nau8825_xtalk_measure()
792 dev_dbg(nau8825->dev, "cross talk sidetone: %x\n", sidetone); in nau8825_xtalk_measure()
793 regmap_write(nau8825->regmap, NAU8825_REG_DAC_DGAIN_CTRL, in nau8825_xtalk_measure()
796 nau8825->xtalk_state = NAU8825_XTALK_DONE; in nau8825_xtalk_measure()
812 if (nau8825->xtalk_state == NAU8825_XTALK_IMM) in nau8825_xtalk_work()
821 if (nau8825->xtalk_state == NAU8825_XTALK_DONE) { in nau8825_xtalk_work()
822 snd_soc_jack_report(nau8825->jack, nau8825->xtalk_event, in nau8825_xtalk_work()
823 nau8825->xtalk_event_mask); in nau8825_xtalk_work()
825 nau8825->xtalk_protect = false; in nau8825_xtalk_work()
835 if (nau8825->xtalk_enable && nau8825->xtalk_state != in nau8825_xtalk_cancel()
837 cancel_work_sync(&nau8825->xtalk_work); in nau8825_xtalk_cancel()
842 nau8825->xtalk_state = NAU8825_XTALK_DONE; in nau8825_xtalk_cancel()
843 nau8825->xtalk_protect = false; in nau8825_xtalk_cancel()
917 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); in nau8825_adc_event()
923 regmap_update_bits(nau8825->regmap, NAU8825_REG_ENA_CTRL, in nau8825_adc_event()
927 if (!nau8825->irq) in nau8825_adc_event()
928 regmap_update_bits(nau8825->regmap, in nau8825_adc_event()
932 return -EINVAL; in nau8825_adc_event()
941 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); in nau8825_pump_event()
948 regmap_update_bits(nau8825->regmap, NAU8825_REG_CHARGE_PUMP, in nau8825_pump_event()
952 regmap_update_bits(nau8825->regmap, NAU8825_REG_CHARGE_PUMP, in nau8825_pump_event()
956 return -EINVAL; in nau8825_pump_event()
965 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); in nau8825_output_dac_event()
971 regmap_update_bits(nau8825->regmap, NAU8825_REG_BIAS_ADJ, in nau8825_output_dac_event()
975 regmap_update_bits(nau8825->regmap, NAU8825_REG_BIAS_ADJ, in nau8825_output_dac_event()
979 return -EINVAL; in nau8825_output_dac_event()
988 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); in system_clock_control()
990 struct regmap *regmap = nau8825->regmap; in system_clock_control()
993 dev_dbg(nau8825->dev, "system clock control : POWER OFF\n"); in system_clock_control()
1014 struct soc_bytes_ext *params = (void *)kcontrol->private_value; in nau8825_biq_coeff_get()
1016 if (!component->regmap) in nau8825_biq_coeff_get()
1017 return -EINVAL; in nau8825_biq_coeff_get()
1019 regmap_raw_read(component->regmap, NAU8825_REG_BIQ_COF1, in nau8825_biq_coeff_get()
1020 ucontrol->value.bytes.data, params->max); in nau8825_biq_coeff_get()
1028 struct soc_bytes_ext *params = (void *)kcontrol->private_value; in nau8825_biq_coeff_put()
1031 if (!component->regmap) in nau8825_biq_coeff_put()
1032 return -EINVAL; in nau8825_biq_coeff_put()
1034 data = kmemdup(ucontrol->value.bytes.data, in nau8825_biq_coeff_put()
1035 params->max, GFP_KERNEL | GFP_DMA); in nau8825_biq_coeff_put()
1037 return -ENOMEM; in nau8825_biq_coeff_put()
1039 regmap_update_bits(component->regmap, NAU8825_REG_BIQ_CTRL, in nau8825_biq_coeff_put()
1041 regmap_raw_write(component->regmap, NAU8825_REG_BIQ_COF1, in nau8825_biq_coeff_put()
1042 data, params->max); in nau8825_biq_coeff_put()
1043 regmap_update_bits(component->regmap, NAU8825_REG_BIQ_CTRL, in nau8825_biq_coeff_put()
1051 "ADC", "DAC"
1074 static const DECLARE_TLV_DB_MINMAX_MUTE(adc_vol_tlv, -10300, 2400);
1075 static const DECLARE_TLV_DB_MINMAX_MUTE(sidetone_vol_tlv, -4200, 0);
1076 static const DECLARE_TLV_DB_MINMAX(dac_vol_tlv, -5400, 0);
1077 static const DECLARE_TLV_DB_MINMAX(fepga_gain_tlv, -100, 3600);
1078 static const DECLARE_TLV_DB_MINMAX_MUTE(crosstalk_vol_tlv, -9600, 2400);
1092 SOC_ENUM("ADC Decimation Rate", nau8825_adc_decimation_enum),
1133 SND_SOC_DAPM_ADC_E("ADC", NULL, SND_SOC_NOPM, 0, 0,
1136 SND_SOC_DAPM_SUPPLY("ADC Clock", NAU8825_REG_ENA_CTRL, 7, 0, NULL, 0),
1137 SND_SOC_DAPM_SUPPLY("ADC Power", NAU8825_REG_ANALOG_ADC_2, 6, 0, NULL,
1140 /* ADC for button press detection. A dapm supply widget is used to
1190 /* HPOL/R are ungrounded by disabling 16 Ohm pull-downs on playback */
1210 {"ADC", NULL, "Frontend PGA"},
1211 {"ADC", NULL, "ADC Clock"},
1212 {"ADC", NULL, "ADC Power"},
1213 {"AIFTX", NULL, "ADC"},
1256 regmap_read(nau8825->regmap, in nau8825_get_osr()
1263 regmap_read(nau8825->regmap, in nau8825_get_osr()
1275 struct snd_soc_component *component = dai->component; in nau8825_dai_startup()
1279 osr = nau8825_get_osr(nau8825, substream->stream); in nau8825_dai_startup()
1280 if (!osr || !osr->osr) in nau8825_dai_startup()
1281 return -EINVAL; in nau8825_dai_startup()
1283 return snd_pcm_hw_constraint_minmax(substream->runtime, in nau8825_dai_startup()
1285 0, CLK_DA_AD_MAX / osr->osr); in nau8825_dai_startup()
1292 struct snd_soc_component *component = dai->component; in nau8825_hw_params()
1296 int err = -EINVAL; in nau8825_hw_params()
1301 * DAC or ADC clock frequency is defined as Over Sampling Rate (OSR) in nau8825_hw_params()
1306 osr = nau8825_get_osr(nau8825, substream->stream); in nau8825_hw_params()
1307 if (!osr || !osr->osr) in nau8825_hw_params()
1309 if (params_rate(params) * osr->osr > CLK_DA_AD_MAX) in nau8825_hw_params()
1311 if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) in nau8825_hw_params()
1312 regmap_update_bits(nau8825->regmap, NAU8825_REG_CLK_DIVIDER, in nau8825_hw_params()
1314 osr->clk_src << NAU8825_CLK_DAC_SRC_SFT); in nau8825_hw_params()
1316 regmap_update_bits(nau8825->regmap, NAU8825_REG_CLK_DIVIDER, in nau8825_hw_params()
1318 osr->clk_src << NAU8825_CLK_ADC_SRC_SFT); in nau8825_hw_params()
1321 regmap_read(nau8825->regmap, NAU8825_REG_I2S_PCM_CTRL2, &ctrl_val); in nau8825_hw_params()
1333 regmap_update_bits(nau8825->regmap, NAU8825_REG_I2S_PCM_CTRL2, in nau8825_hw_params()
1355 regmap_update_bits(nau8825->regmap, NAU8825_REG_I2S_PCM_CTRL1, in nau8825_hw_params()
1368 struct snd_soc_component *component = codec_dai->component; in nau8825_set_dai_fmt()
1379 return -EINVAL; in nau8825_set_dai_fmt()
1389 return -EINVAL; in nau8825_set_dai_fmt()
1410 return -EINVAL; in nau8825_set_dai_fmt()
1415 regmap_update_bits(nau8825->regmap, NAU8825_REG_I2S_PCM_CTRL1, in nau8825_set_dai_fmt()
1419 regmap_update_bits(nau8825->regmap, NAU8825_REG_I2S_PCM_CTRL2, in nau8825_set_dai_fmt()
1429 * nau8825_set_tdm_slot - configure DAI TDM.
1437 * The limitation is DAC and ADC need shift 4 slots at 8 slots mode.
1442 struct snd_soc_component *component = dai->component; in nau8825_set_tdm_slot()
1447 dev_err(nau8825->dev, "Only support 4 or 8 slots!\n"); in nau8825_set_tdm_slot()
1448 return -EINVAL; in nau8825_set_tdm_slot()
1451 /* The driver is limited to 1-channel for ADC, and 2-channel for DAC on TDM mode */ in nau8825_set_tdm_slot()
1454 dev_err(nau8825->dev, in nau8825_set_tdm_slot()
1455 "The limitation is 1-channel for ADC, and 2-channel for DAC on TDM mode.\n"); in nau8825_set_tdm_slot()
1456 return -EINVAL; in nau8825_set_tdm_slot()
1463 dev_err(nau8825->dev, in nau8825_set_tdm_slot()
1464 "Slot assignment of DAC and ADC need to set same interval.\n"); in nau8825_set_tdm_slot()
1465 return -EINVAL; in nau8825_set_tdm_slot()
1470 regmap_update_bits(nau8825->regmap, NAU8825_REG_I2S_PCM_CTRL2, in nau8825_set_tdm_slot()
1472 regmap_read(nau8825->regmap, NAU8825_REG_I2S_PCM_CTRL1, &value); in nau8825_set_tdm_slot()
1509 return -EINVAL; in nau8825_set_tdm_slot()
1512 ctrl_val |= adc_s - 1; in nau8825_set_tdm_slot()
1514 regmap_update_bits(nau8825->regmap, NAU8825_REG_TDM_CTRL, in nau8825_set_tdm_slot()
1518 regmap_update_bits(nau8825->regmap, NAU8825_REG_LEFT_TIME_SLOT, in nau8825_set_tdm_slot()
1536 .name = "nau8825-hifi",
1547 .channels_max = 2, /* Only 1 channel of data */
1555 * nau8825_enable_jack_detect - Specify a jack for event reporting
1568 struct regmap *regmap = nau8825->regmap; in nau8825_enable_jack_detect()
1570 nau8825->jack = jack; in nau8825_enable_jack_detect()
1572 if (!nau8825->jack) { in nau8825_enable_jack_detect()
1621 /* Reset the intrruption status from rightmost bit if the corres- in nau8825_int_status_clear_all()
1635 struct snd_soc_dapm_context *dapm = nau8825->dapm; in nau8825_eject_jack()
1636 struct regmap *regmap = nau8825->regmap; in nau8825_eject_jack()
1654 /* Enable the insertion interruption, disable the ejection inter- in nau8825_eject_jack()
1655 * ruption, and then bypass de-bounce circuit. in nau8825_eject_jack()
1668 /* Disable ADC needed for interruptions at audo mode */ in nau8825_eject_jack()
1679 struct regmap *regmap = nau8825->regmap; in nau8825_setup_auto_irq()
1690 /* Enable ADC needed for interruptions */ in nau8825_setup_auto_irq()
1704 /* Not bypass de-bounce circuit */ in nau8825_setup_auto_irq()
1738 struct regmap *regmap = nau8825->regmap; in nau8825_jack_insert()
1739 struct snd_soc_dapm_context *dapm = nau8825->dapm; in nau8825_jack_insert()
1747 nau8825->high_imped = true; in nau8825_jack_insert()
1749 nau8825->high_imped = false; in nau8825_jack_insert()
1757 dev_dbg(nau8825->dev, "OMTP (micgnd1) mic connected\n"); in nau8825_jack_insert()
1777 dev_dbg(nau8825->dev, "CTIA (micgnd2) mic connected\n"); in nau8825_jack_insert()
1798 dev_err(nau8825->dev, "detection error; disable mic function\n"); in nau8825_jack_insert()
1816 struct regmap *regmap = nau8825->regmap; in nau8825_interrupt()
1820 dev_err(nau8825->dev, "failed to read irq status\n"); in nau8825_interrupt()
1837 * lower 8 bits - for long pressed buttons in nau8825_interrupt()
1839 nau8825->button_pressed = nau8825_button_decode( in nau8825_interrupt()
1842 event |= nau8825->button_pressed; in nau8825_interrupt()
1851 if (nau8825->xtalk_enable && !nau8825->high_imped) { in nau8825_interrupt()
1855 if (!nau8825->xtalk_protect) { in nau8825_interrupt()
1856 /* Raise protection for cross talk de- in nau8825_interrupt()
1858 * The driver has to cancel the pro- in nau8825_interrupt()
1863 nau8825->xtalk_protect = true; in nau8825_interrupt()
1866 nau8825->xtalk_protect = false; in nau8825_interrupt()
1869 if (nau8825->xtalk_protect) { in nau8825_interrupt()
1870 nau8825->xtalk_state = in nau8825_interrupt()
1872 schedule_work(&nau8825->xtalk_work); in nau8825_interrupt()
1879 if (nau8825->xtalk_protect) { in nau8825_interrupt()
1881 nau8825->xtalk_protect = false; in nau8825_interrupt()
1885 dev_warn(nau8825->dev, "Headset completion IRQ fired but no headset connected\n"); in nau8825_interrupt()
1895 if (nau8825->xtalk_state == NAU8825_XTALK_PREPARE) { in nau8825_interrupt()
1896 nau8825->xtalk_event = event; in nau8825_interrupt()
1897 nau8825->xtalk_event_mask = event_mask; in nau8825_interrupt()
1901 if (nau8825->xtalk_enable && nau8825->xtalk_protect) in nau8825_interrupt()
1902 schedule_work(&nau8825->xtalk_work); in nau8825_interrupt()
1936 if (event_mask && nau8825->xtalk_state == NAU8825_XTALK_DONE) in nau8825_interrupt()
1937 snd_soc_jack_report(nau8825->jack, event, event_mask); in nau8825_interrupt()
1944 struct regmap *regmap = nau8825->regmap; in nau8825_setup_buttons()
1948 nau8825->sar_voltage << NAU8825_SAR_TRACKING_GAIN_SFT); in nau8825_setup_buttons()
1951 nau8825->sar_compare_time << NAU8825_SAR_COMPARE_TIME_SFT); in nau8825_setup_buttons()
1954 nau8825->sar_sampling_time << NAU8825_SAR_SAMPLING_TIME_SFT); in nau8825_setup_buttons()
1958 (nau8825->sar_threshold_num - 1) << NAU8825_KEYDET_LEVELS_NR_SFT); in nau8825_setup_buttons()
1961 nau8825->sar_hysteresis << NAU8825_KEYDET_HYSTERESIS_SFT); in nau8825_setup_buttons()
1964 nau8825->key_debounce << NAU8825_KEYDET_SHORTKEY_DEBOUNCE_SFT); in nau8825_setup_buttons()
1967 (nau8825->sar_threshold[0] << 8) | nau8825->sar_threshold[1]); in nau8825_setup_buttons()
1969 (nau8825->sar_threshold[2] << 8) | nau8825->sar_threshold[3]); in nau8825_setup_buttons()
1971 (nau8825->sar_threshold[4] << 8) | nau8825->sar_threshold[5]); in nau8825_setup_buttons()
1973 (nau8825->sar_threshold[6] << 8) | nau8825->sar_threshold[7]); in nau8825_setup_buttons()
1983 struct regmap *regmap = nau8825->regmap; in nau8825_init_regs()
1988 regmap_update_bits(nau8825->regmap, NAU8825_REG_BIAS_ADJ, in nau8825_init_regs()
1990 regmap_update_bits(nau8825->regmap, NAU8825_REG_BOOST, in nau8825_init_regs()
1996 nau8825->vref_impedance << NAU8825_BIAS_VMID_SEL_SFT); in nau8825_init_regs()
2006 nau8825->jkdet_enable ? 0 : NAU8825_JKDET_OUTPUT_EN); in nau8825_init_regs()
2009 nau8825->jkdet_pull_enable ? 0 : NAU8825_JKDET_PULL_EN); in nau8825_init_regs()
2012 nau8825->jkdet_pull_up ? NAU8825_JKDET_PULL_UP : 0); in nau8825_init_regs()
2015 /* jkdet_polarity - 1 is for active-low */ in nau8825_init_regs()
2016 nau8825->jkdet_polarity ? 0 : NAU8825_JACK_POLARITY); in nau8825_init_regs()
2020 nau8825->jack_insert_debounce << NAU8825_JACK_INSERT_DEBOUNCE_SFT); in nau8825_init_regs()
2023 nau8825->jack_eject_debounce << NAU8825_JACK_EJECT_DEBOUNCE_SFT); in nau8825_init_regs()
2029 /* Mask unneeded IRQs: 1 - disable, 0 - enable */ in nau8825_init_regs()
2033 NAU8825_MICBIAS_VOLTAGE_MASK, nau8825->micbias_voltage); in nau8825_init_regs()
2035 if (nau8825->sar_threshold_num) in nau8825_init_regs()
2054 regmap_update_bits(nau8825->regmap, NAU8825_REG_BIAS_ADJ, in nau8825_init_regs()
2075 /* Config L/R channel */ in nau8825_init_regs()
2076 regmap_update_bits(nau8825->regmap, NAU8825_REG_DACL_CTRL, in nau8825_init_regs()
2078 regmap_update_bits(nau8825->regmap, NAU8825_REG_DACR_CTRL, in nau8825_init_regs()
2086 nau8825->adcout_ds << NAU8825_ADCOUT_DS_SFT); in nau8825_init_regs()
2108 nau8825->dapm = dapm; in nau8825_component_probe()
2122 * nau8825_calc_fll_param - Calculate FLL parameters.
2138 * freq_in by 1, 2, 4, or 8 using FLL pre-scalar. in nau8825_calc_fll_param()
2147 return -EINVAL; in nau8825_calc_fll_param()
2148 fll_param->clk_ref_div = fll_pre_scalar[i].val; in nau8825_calc_fll_param()
2156 return -EINVAL; in nau8825_calc_fll_param()
2157 fll_param->ratio = fll_ratio[i].val; in nau8825_calc_fll_param()
2160 * FDCO must be within the 90MHz - 124MHz or the FFL cannot be in nau8825_calc_fll_param()
2175 return -EINVAL; in nau8825_calc_fll_param()
2176 fll_param->mclk_src = mclk_src_scaling[fvco_sel].val; in nau8825_calc_fll_param()
2178 /* Calculate the FLL 10-bit integer input and the FLL 16-bit fractional in nau8825_calc_fll_param()
2181 fvco = div_u64(fvco_max << 16, fref * fll_param->ratio); in nau8825_calc_fll_param()
2182 fll_param->fll_int = (fvco >> 16) & 0x3FF; in nau8825_calc_fll_param()
2183 fll_param->fll_frac = fvco & 0xFFFF; in nau8825_calc_fll_param()
2190 regmap_update_bits(nau8825->regmap, NAU8825_REG_CLK_DIVIDER, in nau8825_fll_apply()
2192 NAU8825_CLK_SRC_MCLK | fll_param->mclk_src); in nau8825_fll_apply()
2194 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL1, in nau8825_fll_apply()
2196 fll_param->ratio | (0x6 << NAU8825_ICTRL_LATCH_SFT)); in nau8825_fll_apply()
2197 /* FLL 16-bit fractional input */ in nau8825_fll_apply()
2198 regmap_write(nau8825->regmap, NAU8825_REG_FLL2, fll_param->fll_frac); in nau8825_fll_apply()
2199 /* FLL 10-bit integer input */ in nau8825_fll_apply()
2200 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL3, in nau8825_fll_apply()
2201 NAU8825_FLL_INTEGER_MASK, fll_param->fll_int); in nau8825_fll_apply()
2202 /* FLL pre-scaler */ in nau8825_fll_apply()
2203 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL4, in nau8825_fll_apply()
2205 fll_param->clk_ref_div << NAU8825_FLL_REF_DIV_SFT); in nau8825_fll_apply()
2207 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL5, in nau8825_fll_apply()
2209 /* Disable free-running mode */ in nau8825_fll_apply()
2210 regmap_update_bits(nau8825->regmap, in nau8825_fll_apply()
2212 if (fll_param->fll_frac) { in nau8825_fll_apply()
2214 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL5, in nau8825_fll_apply()
2219 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL6, in nau8825_fll_apply()
2224 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL5, in nau8825_fll_apply()
2227 regmap_update_bits(nau8825->regmap, NAU8825_REG_FLL6, in nau8825_fll_apply()
2243 dev_err(component->dev, "Unsupported input clock %d\n", freq_in); in nau8825_set_pll()
2246 dev_dbg(component->dev, "mclk_src=%x ratio=%x fll_frac=%x fll_int=%x clk_ref_div=%x\n", in nau8825_set_pll()
2252 regmap_update_bits(nau8825->regmap, NAU8825_REG_CLK_DIVIDER, in nau8825_set_pll()
2261 nau8825->mclk = devm_clk_get(nau8825->dev, "mclk"); in nau8825_mclk_prepare()
2262 if (IS_ERR(nau8825->mclk)) { in nau8825_mclk_prepare()
2263 dev_info(nau8825->dev, "No 'mclk' clock found, assume MCLK is managed externally"); in nau8825_mclk_prepare()
2267 if (!nau8825->mclk_freq) { in nau8825_mclk_prepare()
2268 ret = clk_prepare_enable(nau8825->mclk); in nau8825_mclk_prepare()
2270 dev_err(nau8825->dev, "Unable to prepare codec mclk\n"); in nau8825_mclk_prepare()
2275 if (nau8825->mclk_freq != freq) { in nau8825_mclk_prepare()
2276 freq = clk_round_rate(nau8825->mclk, freq); in nau8825_mclk_prepare()
2277 ret = clk_set_rate(nau8825->mclk, freq); in nau8825_mclk_prepare()
2279 dev_err(nau8825->dev, "Unable to set mclk rate\n"); in nau8825_mclk_prepare()
2282 nau8825->mclk_freq = freq; in nau8825_mclk_prepare()
2302 struct regmap *regmap = nau8825->regmap; in nau8825_configure_sysclk()
2309 if (nau8825->mclk_freq) { in nau8825_configure_sysclk()
2310 clk_disable_unprepare(nau8825->mclk); in nau8825_configure_sysclk()
2311 nau8825->mclk_freq = 0; in nau8825_configure_sysclk()
2317 * interrupt handler. In order to avoid the playback inter- in nau8825_configure_sysclk()
2335 if (nau8825_is_jack_inserted(nau8825->regmap)) { in nau8825_configure_sysclk()
2355 dev_warn(nau8825->dev, "Disable clock for power saving when no headset connected\n"); in nau8825_configure_sysclk()
2357 if (nau8825->mclk_freq) { in nau8825_configure_sysclk()
2358 clk_disable_unprepare(nau8825->mclk); in nau8825_configure_sysclk()
2359 nau8825->mclk_freq = 0; in nau8825_configure_sysclk()
2365 * interrupt handler. In order to avoid the playback inter- in nau8825_configure_sysclk()
2387 * interrupt handler. In order to avoid the playback inter- in nau8825_configure_sysclk()
2405 if (nau8825->mclk_freq) { in nau8825_configure_sysclk()
2406 clk_disable_unprepare(nau8825->mclk); in nau8825_configure_sysclk()
2407 nau8825->mclk_freq = 0; in nau8825_configure_sysclk()
2413 * interrupt handler. In order to avoid the playback inter- in nau8825_configure_sysclk()
2431 if (nau8825->mclk_freq) { in nau8825_configure_sysclk()
2432 clk_disable_unprepare(nau8825->mclk); in nau8825_configure_sysclk()
2433 nau8825->mclk_freq = 0; in nau8825_configure_sysclk()
2438 dev_err(nau8825->dev, "Invalid clock id (%d)\n", clk_id); in nau8825_configure_sysclk()
2439 return -EINVAL; in nau8825_configure_sysclk()
2442 dev_dbg(nau8825->dev, "Sysclk is %dHz and clock id is %d\n", freq, in nau8825_configure_sysclk()
2457 struct regmap *regmap = nau8825->regmap; in nau8825_resume_setup()
2466 * bypass de-bounce circuit. in nau8825_resume_setup()
2495 if (nau8825->mclk_freq) { in nau8825_set_bias_level()
2496 ret = clk_prepare_enable(nau8825->mclk); in nau8825_set_bias_level()
2498 dev_err(nau8825->dev, "Unable to prepare component mclk\n"); in nau8825_set_bias_level()
2510 regmap_update_bits(nau8825->regmap, NAU8825_REG_MIC_BIAS, in nau8825_set_bias_level()
2513 regmap_update_bits(nau8825->regmap, in nau8825_set_bias_level()
2520 regmap_write(nau8825->regmap, in nau8825_set_bias_level()
2522 /* Disable ADC needed for interruptions at audo mode */ in nau8825_set_bias_level()
2523 regmap_update_bits(nau8825->regmap, NAU8825_REG_ENA_CTRL, in nau8825_set_bias_level()
2525 if (nau8825->mclk_freq) in nau8825_set_bias_level()
2526 clk_disable_unprepare(nau8825->mclk); in nau8825_set_bias_level()
2536 disable_irq(nau8825->irq); in nau8825_suspend()
2539 snd_soc_dapm_disable_pin(nau8825->dapm, "SAR"); in nau8825_suspend()
2540 snd_soc_dapm_disable_pin(nau8825->dapm, "MICBIAS"); in nau8825_suspend()
2541 snd_soc_dapm_sync(nau8825->dapm); in nau8825_suspend()
2542 regcache_cache_only(nau8825->regmap, true); in nau8825_suspend()
2543 regcache_mark_dirty(nau8825->regmap); in nau8825_suspend()
2553 regcache_cache_only(nau8825->regmap, false); in nau8825_resume()
2554 regcache_sync(nau8825->regmap); in nau8825_resume()
2555 nau8825->xtalk_protect = true; in nau8825_resume()
2558 nau8825->xtalk_protect = false; in nau8825_resume()
2559 enable_irq(nau8825->irq); in nau8825_resume()
2600 struct device *dev = nau8825->dev; in nau8825_print_device_properties()
2602 dev_dbg(dev, "jkdet-enable: %d\n", nau8825->jkdet_enable); in nau8825_print_device_properties()
2603 dev_dbg(dev, "jkdet-pull-enable: %d\n", nau8825->jkdet_pull_enable); in nau8825_print_device_properties()
2604 dev_dbg(dev, "jkdet-pull-up: %d\n", nau8825->jkdet_pull_up); in nau8825_print_device_properties()
2605 dev_dbg(dev, "jkdet-polarity: %d\n", nau8825->jkdet_polarity); in nau8825_print_device_properties()
2606 dev_dbg(dev, "micbias-voltage: %d\n", nau8825->micbias_voltage); in nau8825_print_device_properties()
2607 dev_dbg(dev, "vref-impedance: %d\n", nau8825->vref_impedance); in nau8825_print_device_properties()
2609 dev_dbg(dev, "sar-threshold-num: %d\n", nau8825->sar_threshold_num); in nau8825_print_device_properties()
2610 for (i = 0; i < nau8825->sar_threshold_num; i++) in nau8825_print_device_properties()
2611 dev_dbg(dev, "sar-threshold[%d]=%d\n", i, in nau8825_print_device_properties()
2612 nau8825->sar_threshold[i]); in nau8825_print_device_properties()
2614 dev_dbg(dev, "sar-hysteresis: %d\n", nau8825->sar_hysteresis); in nau8825_print_device_properties()
2615 dev_dbg(dev, "sar-voltage: %d\n", nau8825->sar_voltage); in nau8825_print_device_properties()
2616 dev_dbg(dev, "sar-compare-time: %d\n", nau8825->sar_compare_time); in nau8825_print_device_properties()
2617 dev_dbg(dev, "sar-sampling-time: %d\n", nau8825->sar_sampling_time); in nau8825_print_device_properties()
2618 dev_dbg(dev, "short-key-debounce: %d\n", nau8825->key_debounce); in nau8825_print_device_properties()
2619 dev_dbg(dev, "jack-insert-debounce: %d\n", in nau8825_print_device_properties()
2620 nau8825->jack_insert_debounce); in nau8825_print_device_properties()
2621 dev_dbg(dev, "jack-eject-debounce: %d\n", in nau8825_print_device_properties()
2622 nau8825->jack_eject_debounce); in nau8825_print_device_properties()
2623 dev_dbg(dev, "crosstalk-enable: %d\n", in nau8825_print_device_properties()
2624 nau8825->xtalk_enable); in nau8825_print_device_properties()
2625 dev_dbg(dev, "adcout-drive-strong: %d\n", nau8825->adcout_ds); in nau8825_print_device_properties()
2632 nau8825->jkdet_enable = device_property_read_bool(dev, in nau8825_read_device_properties()
2633 "nuvoton,jkdet-enable"); in nau8825_read_device_properties()
2634 nau8825->jkdet_pull_enable = device_property_read_bool(dev, in nau8825_read_device_properties()
2635 "nuvoton,jkdet-pull-enable"); in nau8825_read_device_properties()
2636 nau8825->jkdet_pull_up = device_property_read_bool(dev, in nau8825_read_device_properties()
2637 "nuvoton,jkdet-pull-up"); in nau8825_read_device_properties()
2638 ret = device_property_read_u32(dev, "nuvoton,jkdet-polarity", in nau8825_read_device_properties()
2639 &nau8825->jkdet_polarity); in nau8825_read_device_properties()
2641 nau8825->jkdet_polarity = 1; in nau8825_read_device_properties()
2642 ret = device_property_read_u32(dev, "nuvoton,micbias-voltage", in nau8825_read_device_properties()
2643 &nau8825->micbias_voltage); in nau8825_read_device_properties()
2645 nau8825->micbias_voltage = 6; in nau8825_read_device_properties()
2646 ret = device_property_read_u32(dev, "nuvoton,vref-impedance", in nau8825_read_device_properties()
2647 &nau8825->vref_impedance); in nau8825_read_device_properties()
2649 nau8825->vref_impedance = 2; in nau8825_read_device_properties()
2650 ret = device_property_read_u32(dev, "nuvoton,sar-threshold-num", in nau8825_read_device_properties()
2651 &nau8825->sar_threshold_num); in nau8825_read_device_properties()
2653 nau8825->sar_threshold_num = 4; in nau8825_read_device_properties()
2654 ret = device_property_read_u32_array(dev, "nuvoton,sar-threshold", in nau8825_read_device_properties()
2655 nau8825->sar_threshold, nau8825->sar_threshold_num); in nau8825_read_device_properties()
2657 nau8825->sar_threshold[0] = 0x08; in nau8825_read_device_properties()
2658 nau8825->sar_threshold[1] = 0x12; in nau8825_read_device_properties()
2659 nau8825->sar_threshold[2] = 0x26; in nau8825_read_device_properties()
2660 nau8825->sar_threshold[3] = 0x73; in nau8825_read_device_properties()
2662 ret = device_property_read_u32(dev, "nuvoton,sar-hysteresis", in nau8825_read_device_properties()
2663 &nau8825->sar_hysteresis); in nau8825_read_device_properties()
2665 nau8825->sar_hysteresis = 0; in nau8825_read_device_properties()
2666 ret = device_property_read_u32(dev, "nuvoton,sar-voltage", in nau8825_read_device_properties()
2667 &nau8825->sar_voltage); in nau8825_read_device_properties()
2669 nau8825->sar_voltage = 6; in nau8825_read_device_properties()
2670 ret = device_property_read_u32(dev, "nuvoton,sar-compare-time", in nau8825_read_device_properties()
2671 &nau8825->sar_compare_time); in nau8825_read_device_properties()
2673 nau8825->sar_compare_time = 1; in nau8825_read_device_properties()
2674 ret = device_property_read_u32(dev, "nuvoton,sar-sampling-time", in nau8825_read_device_properties()
2675 &nau8825->sar_sampling_time); in nau8825_read_device_properties()
2677 nau8825->sar_sampling_time = 1; in nau8825_read_device_properties()
2678 ret = device_property_read_u32(dev, "nuvoton,short-key-debounce", in nau8825_read_device_properties()
2679 &nau8825->key_debounce); in nau8825_read_device_properties()
2681 nau8825->key_debounce = 3; in nau8825_read_device_properties()
2682 ret = device_property_read_u32(dev, "nuvoton,jack-insert-debounce", in nau8825_read_device_properties()
2683 &nau8825->jack_insert_debounce); in nau8825_read_device_properties()
2685 nau8825->jack_insert_debounce = 7; in nau8825_read_device_properties()
2686 ret = device_property_read_u32(dev, "nuvoton,jack-eject-debounce", in nau8825_read_device_properties()
2687 &nau8825->jack_eject_debounce); in nau8825_read_device_properties()
2689 nau8825->jack_eject_debounce = 0; in nau8825_read_device_properties()
2690 nau8825->xtalk_enable = device_property_read_bool(dev, in nau8825_read_device_properties()
2691 "nuvoton,crosstalk-enable"); in nau8825_read_device_properties()
2692 nau8825->adcout_ds = device_property_read_bool(dev, "nuvoton,adcout-drive-strong"); in nau8825_read_device_properties()
2694 nau8825->mclk = devm_clk_get(dev, "mclk"); in nau8825_read_device_properties()
2695 if (PTR_ERR(nau8825->mclk) == -EPROBE_DEFER) { in nau8825_read_device_properties()
2696 return -EPROBE_DEFER; in nau8825_read_device_properties()
2697 } else if (PTR_ERR(nau8825->mclk) == -ENOENT) { in nau8825_read_device_properties()
2699 nau8825->mclk = NULL; in nau8825_read_device_properties()
2701 } else if (IS_ERR(nau8825->mclk)) { in nau8825_read_device_properties()
2702 return -EINVAL; in nau8825_read_device_properties()
2712 ret = devm_request_threaded_irq(nau8825->dev, nau8825->irq, NULL, in nau8825_setup_irq()
2717 dev_err(nau8825->dev, "Cannot request irq %d (%d)\n", in nau8825_setup_irq()
2718 nau8825->irq, ret); in nau8825_setup_irq()
2727 struct device *dev = &i2c->dev; in nau8825_i2c_probe()
2728 struct nau8825 *nau8825 = dev_get_platdata(&i2c->dev); in nau8825_i2c_probe()
2734 return -ENOMEM; in nau8825_i2c_probe()
2742 nau8825->regmap = devm_regmap_init_i2c(i2c, &nau8825_regmap_config); in nau8825_i2c_probe()
2743 if (IS_ERR(nau8825->regmap)) in nau8825_i2c_probe()
2744 return PTR_ERR(nau8825->regmap); in nau8825_i2c_probe()
2745 nau8825->dev = dev; in nau8825_i2c_probe()
2746 nau8825->irq = i2c->irq; in nau8825_i2c_probe()
2750 nau8825->xtalk_state = NAU8825_XTALK_DONE; in nau8825_i2c_probe()
2751 nau8825->xtalk_protect = false; in nau8825_i2c_probe()
2752 nau8825->xtalk_baktab_initialized = false; in nau8825_i2c_probe()
2753 sema_init(&nau8825->xtalk_sem, 1); in nau8825_i2c_probe()
2754 INIT_WORK(&nau8825->xtalk_work, nau8825_xtalk_work); in nau8825_i2c_probe()
2758 nau8825_reset_chip(nau8825->regmap); in nau8825_i2c_probe()
2759 ret = regmap_read(nau8825->regmap, NAU8825_REG_I2C_DEVICE_ID, &value); in nau8825_i2c_probe()
2768 return -ENODEV; in nau8825_i2c_probe()
2773 if (i2c->irq) in nau8825_i2c_probe()
2776 return devm_snd_soc_register_component(&i2c->dev, in nau8825_i2c_probe()