// SPDX-License-Identifier: GPL-2.0-only
//
// Cirrus Logic CS48L32 audio DSP.
//
// Copyright (C) 2016-2018, 2020, 2022, 2025 Cirrus Logic, Inc. and
// Cirrus Logic International Semiconductor Ltd.
#include <dt-bindings/sound/cs48l32.h>
#include <linux/array_size.h>
#include <linux/build_bug.h>
#include <linux/clk.h>
#include <linux/container_of.h>
#include <linux/delay.h>
#include <linux/err.h>
#include <linux/gcd.h>
#include <linux/gpio/consumer.h>
#include <linux/minmax.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/pm_runtime.h>
#include <linux/property.h>
#include <linux/regmap.h>
#include <linux/regulator/consumer.h>
#include <linux/slab.h>
#include <linux/spi/spi.h>
#include <linux/string_choices.h>
#include <sound/cs48l32.h>
#include <sound/cs48l32_registers.h>
#include <sound/pcm.h>
#include <sound/pcm_params.h>
#include <sound/soc.h>
#include <sound/soc-component.h>
#include <sound/soc-dai.h>
#include <sound/soc-dapm.h>
#include <sound/tlv.h>
#include "cs48l32.h"
static const char * const cs48l32_core_supplies[] = { "vdd-a" , "vdd-io" };
static const struct cs_dsp_region cs48l32_dsp1_regions[] = {
{ .type = WMFW_HALO_PM_PACKED, .base = 0x3800000 },
{ .type = WMFW_HALO_XM_PACKED, .base = 0x2000000 },
{ .type = WMFW_ADSP2_XM, .base = 0x2800000 },
{ .type = WMFW_HALO_YM_PACKED, .base = 0x2C00000 },
{ .type = WMFW_ADSP2_YM, .base = 0x3400000 },
};
static const struct cs48l32_dsp_power_reg_block cs48l32_dsp1_sram_ext_regs[] = {
{ CS48L32_DSP1_XM_SRAM_IBUS_SETUP_1, CS48L32_DSP1_XM_SRAM_IBUS_SETUP_24 },
{ CS48L32_DSP1_YM_SRAM_IBUS_SETUP_1, CS48L32_DSP1_YM_SRAM_IBUS_SETUP_8 },
{ CS48L32_DSP1_PM_SRAM_IBUS_SETUP_1, CS48L32_DSP1_PM_SRAM_IBUS_SETUP_7 },
};
static const unsigned int cs48l32_dsp1_sram_pwd_regs[] = {
CS48L32_DSP1_XM_SRAM_IBUS_SETUP_0,
CS48L32_DSP1_YM_SRAM_IBUS_SETUP_0,
CS48L32_DSP1_PM_SRAM_IBUS_SETUP_0,
};
static const struct cs48l32_dsp_power_regs cs48l32_dsp_sram_regs = {
.ext = cs48l32_dsp1_sram_ext_regs,
.n_ext = ARRAY_SIZE(cs48l32_dsp1_sram_ext_regs),
.pwd = cs48l32_dsp1_sram_pwd_regs,
.n_pwd = ARRAY_SIZE(cs48l32_dsp1_sram_pwd_regs),
};
static const char * const cs48l32_mixer_texts[] = {
"None" ,
"Tone Generator 1" ,
"Tone Generator 2" ,
"Noise Generator" ,
"IN1L" ,
"IN1R" ,
"IN2L" ,
"IN2R" ,
"ASP1RX1" ,
"ASP1RX2" ,
"ASP1RX3" ,
"ASP1RX4" ,
"ASP1RX5" ,
"ASP1RX6" ,
"ASP1RX7" ,
"ASP1RX8" ,
"ASP2RX1" ,
"ASP2RX2" ,
"ASP2RX3" ,
"ASP2RX4" ,
"ISRC1INT1" ,
"ISRC1INT2" ,
"ISRC1INT3" ,
"ISRC1INT4" ,
"ISRC1DEC1" ,
"ISRC1DEC2" ,
"ISRC1DEC3" ,
"ISRC1DEC4" ,
"ISRC2INT1" ,
"ISRC2INT2" ,
"ISRC2DEC1" ,
"ISRC2DEC2" ,
"ISRC3INT1" ,
"ISRC3INT2" ,
"ISRC3DEC1" ,
"ISRC3DEC2" ,
"EQ1" ,
"EQ2" ,
"EQ3" ,
"EQ4" ,
"DRC1L" ,
"DRC1R" ,
"DRC2L" ,
"DRC2R" ,
"LHPF1" ,
"LHPF2" ,
"LHPF3" ,
"LHPF4" ,
"Ultrasonic 1" ,
"Ultrasonic 2" ,
"DSP1.1" ,
"DSP1.2" ,
"DSP1.3" ,
"DSP1.4" ,
"DSP1.5" ,
"DSP1.6" ,
"DSP1.7" ,
"DSP1.8" ,
};
static unsigned int cs48l32_mixer_values[] = {
0x000, /* Silence (mute) */
0x004, /* Tone generator 1 */
0x005, /* Tone generator 2 */
0x00C, /* Noise Generator */
0x010, /* IN1L signal path */
0x011, /* IN1R signal path */
0x012, /* IN2L signal path */
0x013, /* IN2R signal path */
0x020, /* ASP1 RX1 */
0x021, /* ASP1 RX2 */
0x022, /* ASP1 RX3 */
0x023, /* ASP1 RX4 */
0x024, /* ASP1 RX5 */
0x025, /* ASP1 RX6 */
0x026, /* ASP1 RX7 */
0x027, /* ASP1 RX8 */
0x030, /* ASP2 RX1 */
0x031, /* ASP2 RX2 */
0x032, /* ASP2 RX3 */
0x033, /* ASP2 RX4 */
0x098, /* ISRC1 INT1 */
0x099, /* ISRC1 INT2 */
0x09a, /* ISRC1 INT3 */
0x09b, /* ISRC1 INT4 */
0x09C, /* ISRC1 DEC1 */
0x09D, /* ISRC1 DEC2 */
0x09e, /* ISRC1 DEC3 */
0x09f, /* ISRC1 DEC4 */
0x0A0, /* ISRC2 INT1 */
0x0A1, /* ISRC2 INT2 */
0x0A4, /* ISRC2 DEC1 */
0x0A5, /* ISRC2 DEC2 */
0x0A8, /* ISRC3 INT1 */
0x0A9, /* ISRC3 INT2 */
0x0AC, /* ISRC3 DEC1 */
0x0AD, /* ISRC3 DEC2 */
0x0B8, /* EQ1 */
0x0B9, /* EQ2 */
0x0BA, /* EQ3 */
0x0BB, /* EQ4 */
0x0C0, /* DRC1 Left */
0x0C1, /* DRC1 Right */
0x0C2, /* DRC2 Left */
0x0C3, /* DRC2 Right */
0x0C8, /* LHPF1 */
0x0C9, /* LHPF2 */
0x0CA, /* LHPF3 */
0x0CB, /* LHPF4 */
0x0D8, /* Ultrasonic 1 */
0x0D9, /* Ultrasonic 2 */
0x100, /* DSP1 channel 1 */
0x101, /* DSP1 channel 2 */
0x102, /* DSP1 channel 3 */
0x103, /* DSP1 channel 4 */
0x104, /* DSP1 channel 5 */
0x105, /* DSP1 channel 6 */
0x106, /* DSP1 channel 7 */
0x107, /* DSP1 channel 8 */
};
static_assert(ARRAY_SIZE(cs48l32_mixer_texts) == ARRAY_SIZE(cs48l32_mixer_values));
#define CS48L32_NUM_MIXER_INPUTS ARRAY_SIZE(cs48l32_mixer_values)
static const DECLARE_TLV_DB_SCALE(cs48l32_ana_tlv, 0, 100, 0);
static const DECLARE_TLV_DB_SCALE(cs48l32_eq_tlv, -1200, 100, 0);
static const DECLARE_TLV_DB_SCALE(cs48l32_digital_tlv, -6400, 50, 0);
static const DECLARE_TLV_DB_SCALE(cs48l32_noise_tlv, -10800, 600, 0);
static const DECLARE_TLV_DB_SCALE(cs48l32_mixer_tlv, -3200, 100, 0);
static const DECLARE_TLV_DB_SCALE(cs48l32_us_tlv, 0, 600, 0);
static void cs48l32_spin_sysclk(struct cs48l32_codec *cs48l32_codec)
{
struct cs48l32 *cs48l32 = &cs48l32_codec->core;
unsigned int val;
int ret, i;
/* Skip this if the chip is down */
if (pm_runtime_suspended(cs48l32->dev))
return ;
/*
* Just read a register a few times to ensure the internal
* oscillator sends out some clocks.
*/
for (i = 0; i < 4; i++) {
ret = regmap_read(cs48l32->regmap, CS48L32_DEVID, &val);
if (ret)
dev_err(cs48l32_codec->core.dev, "%s Failed to read register: %d (%d)\n" ,
__func__, ret, i);
}
udelay(300);
}
static const char * const cs48l32_rate_text[] = {
"Sample Rate 1" , "Sample Rate 2" , "Sample Rate 3" , "Sample Rate 4" ,
};
static const unsigned int cs48l32_rate_val[] = {
0x0, 0x1, 0x2, 0x3,
};
static_assert(ARRAY_SIZE(cs48l32_rate_val) == ARRAY_SIZE(cs48l32_rate_text));
static int cs48l32_rate_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
int ret;
/* Prevent any mixer mux changes while we do this */
mutex_lock(&cs48l32_codec->rate_lock);
/* The write must be guarded by a number of SYSCLK cycles */
cs48l32_spin_sysclk(cs48l32_codec);
ret = snd_soc_put_enum_double(kcontrol, ucontrol);
cs48l32_spin_sysclk(cs48l32_codec);
mutex_unlock(&cs48l32_codec->rate_lock);
return ret;
}
static const char * const cs48l32_sample_rate_text[] = {
"12kHz" ,
"24kHz" ,
"48kHz" ,
"96kHz" ,
"192kHz" ,
"384kHz" ,
"768kHz" ,
"11.025kHz" ,
"22.05kHz" ,
"44.1kHz" ,
"88.2kHz" ,
"176.4kHz" ,
"352.8kHz" ,
"705.6kHz" ,
"8kHz" ,
"16kHz" ,
"32kHz" ,
};
static const unsigned int cs48l32_sample_rate_val[] = {
0x01, /* 12kHz */
0x02, /* 24kHz */
0x03, /* 48kHz */
0x04, /* 96kHz */
0x05, /* 192kHz */
0x06, /* 384kHz */
0x07, /* 768kHz */
0x09, /* 11.025kHz */
0x0a, /* 22.05kHz */
0x0b, /* 44.1kHz */
0x0c, /* 88.2kHz */
0x0d, /* 176.4kHz */
0x0e, /* 352.8kHz */
0x0f, /* 705.6kHz */
0x11, /* 8kHz */
0x12, /* 16kHz */
0x13, /* 32kHz */
};
static_assert(ARRAY_SIZE(cs48l32_sample_rate_val) == ARRAY_SIZE(cs48l32_sample_rate_text));
#define CS48L32_SAMPLE_RATE_ENUM_SIZE ARRAY_SIZE(cs48l32_sample_rate_val)
static const struct soc_enum cs48l32_sample_rate[] = {
SOC_VALUE_ENUM_SINGLE(CS48L32_SAMPLE_RATE1,
CS48L32_SAMPLE_RATE_1_SHIFT,
CS48L32_SAMPLE_RATE_1_MASK >> CS48L32_SAMPLE_RATE_1_SHIFT,
CS48L32_SAMPLE_RATE_ENUM_SIZE,
cs48l32_sample_rate_text,
cs48l32_sample_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_SAMPLE_RATE2,
CS48L32_SAMPLE_RATE_1_SHIFT,
CS48L32_SAMPLE_RATE_1_MASK >> CS48L32_SAMPLE_RATE_1_SHIFT,
CS48L32_SAMPLE_RATE_ENUM_SIZE,
cs48l32_sample_rate_text,
cs48l32_sample_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_SAMPLE_RATE3,
CS48L32_SAMPLE_RATE_1_SHIFT,
CS48L32_SAMPLE_RATE_1_MASK >> CS48L32_SAMPLE_RATE_1_SHIFT,
CS48L32_SAMPLE_RATE_ENUM_SIZE,
cs48l32_sample_rate_text,
cs48l32_sample_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_SAMPLE_RATE4,
CS48L32_SAMPLE_RATE_1_SHIFT,
CS48L32_SAMPLE_RATE_1_MASK >> CS48L32_SAMPLE_RATE_1_SHIFT,
CS48L32_SAMPLE_RATE_ENUM_SIZE,
cs48l32_sample_rate_text,
cs48l32_sample_rate_val),
};
static int cs48l32_inmux_put(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_dapm_kcontrol_component(kcontrol);
struct snd_soc_dapm_context *dapm = snd_soc_dapm_kcontrol_dapm(kcontrol);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct soc_enum *e = (struct soc_enum *) kcontrol->private_value;
unsigned int mux, src_val, in_type;
int ret;
mux = ucontrol->value.enumerated.item[0];
if (mux > 1)
return -EINVAL;
switch (e->reg) {
case CS48L32_IN1L_CONTROL1:
in_type = cs48l32_codec->in_type[0][mux];
break ;
case CS48L32_IN1R_CONTROL1:
in_type = cs48l32_codec->in_type[1][mux];
break ;
default :
return -EINVAL;
}
src_val = mux << e->shift_l;
if (in_type == CS48L32_IN_TYPE_SE)
src_val |= 1 << CS48L32_INx_SRC_SHIFT;
ret = snd_soc_component_update_bits(dapm->component,
e->reg,
CS48L32_INx_SRC_MASK,
src_val);
if (ret > 0)
snd_soc_dapm_mux_update_power(dapm, kcontrol, mux, e, NULL);
return ret;
}
static const char * const cs48l32_inmux_texts[] = {
"Analog 1" , "Analog 2" ,
};
static SOC_ENUM_SINGLE_DECL(cs48l32_in1muxl_enum,
CS48L32_IN1L_CONTROL1,
CS48L32_INx_SRC_SHIFT + 1,
cs48l32_inmux_texts);
static SOC_ENUM_SINGLE_DECL(cs48l32_in1muxr_enum,
CS48L32_IN1R_CONTROL1,
CS48L32_INx_SRC_SHIFT + 1,
cs48l32_inmux_texts);
static const struct snd_kcontrol_new cs48l32_inmux[] = {
SOC_DAPM_ENUM_EXT("IN1L Mux" , cs48l32_in1muxl_enum,
snd_soc_dapm_get_enum_double, cs48l32_inmux_put),
SOC_DAPM_ENUM_EXT("IN1R Mux" , cs48l32_in1muxr_enum,
snd_soc_dapm_get_enum_double, cs48l32_inmux_put),
};
static const char * const cs48l32_dmode_texts[] = {
"Analog" , "Digital" ,
};
static int cs48l32_dmode_put(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_dapm_context *dapm = snd_soc_dapm_kcontrol_dapm(kcontrol);
struct snd_soc_component *component = snd_soc_dapm_to_component(dapm);
struct soc_enum *e = (struct soc_enum *) kcontrol->private_value;
unsigned int mode;
int ret, result;
mode = ucontrol->value.enumerated.item[0];
switch (mode) {
case 0:
ret = snd_soc_component_update_bits(component,
CS48L32_ADC1L_ANA_CONTROL1,
CS48L32_ADC1x_INT_ENA_FRC_MASK,
CS48L32_ADC1x_INT_ENA_FRC_MASK);
if (ret < 0) {
dev_err(component->dev,
"Failed to set ADC1L_INT_ENA_FRC: %d\n" , ret);
return ret;
}
ret = snd_soc_component_update_bits(component,
CS48L32_ADC1R_ANA_CONTROL1,
CS48L32_ADC1x_INT_ENA_FRC_MASK,
CS48L32_ADC1x_INT_ENA_FRC_MASK);
if (ret < 0) {
dev_err(component->dev,
"Failed to set ADC1R_INT_ENA_FRC: %d\n" , ret);
return ret;
}
result = snd_soc_component_update_bits(component,
e->reg,
BIT(CS48L32_IN1_MODE_SHIFT),
0);
if (result < 0) {
dev_err(component->dev, "Failed to set input mode: %d\n" , result);
return result;
}
usleep_range(200, 300);
ret = snd_soc_component_update_bits(component,
CS48L32_ADC1L_ANA_CONTROL1,
CS48L32_ADC1x_INT_ENA_FRC_MASK,
0);
if (ret < 0) {
dev_err(component->dev,
"Failed to clear ADC1L_INT_ENA_FRC: %d\n" , ret);
return ret;
}
ret = snd_soc_component_update_bits(component,
CS48L32_ADC1R_ANA_CONTROL1,
CS48L32_ADC1x_INT_ENA_FRC_MASK,
0);
if (ret < 0) {
dev_err(component->dev,
"Failed to clear ADC1R_INT_ENA_FRC: %d\n" , ret);
return ret;
}
if (result > 0)
snd_soc_dapm_mux_update_power(dapm, kcontrol, mode, e, NULL);
return result;
case 1:
return snd_soc_dapm_put_enum_double(kcontrol, ucontrol);
default :
return -EINVAL;
}
}
static SOC_ENUM_SINGLE_DECL(cs48l32_in1dmode_enum,
CS48L32_INPUT1_CONTROL1,
CS48L32_IN1_MODE_SHIFT,
cs48l32_dmode_texts);
static const struct snd_kcontrol_new cs48l32_dmode_mux[] = {
SOC_DAPM_ENUM_EXT("IN1 Mode" , cs48l32_in1dmode_enum,
snd_soc_dapm_get_enum_double, cs48l32_dmode_put),
};
static const char * const cs48l32_in_texts[] = {
"IN1L" , "IN1R" , "IN2L" , "IN2R" ,
};
static_assert(ARRAY_SIZE(cs48l32_in_texts) == CS48L32_MAX_INPUT);
static const char * const cs48l32_us_freq_texts[] = {
"16-24kHz" , "20-28kHz" ,
};
static const unsigned int cs48l32_us_freq_val[] = {
0x2, 0x3,
};
static const struct soc_enum cs48l32_us_freq[] = {
SOC_VALUE_ENUM_SINGLE(CS48L32_US1_CONTROL,
CS48L32_US1_FREQ_SHIFT,
CS48L32_US1_FREQ_MASK >> CS48L32_US1_FREQ_SHIFT,
ARRAY_SIZE(cs48l32_us_freq_val),
cs48l32_us_freq_texts,
cs48l32_us_freq_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_US2_CONTROL,
CS48L32_US1_FREQ_SHIFT,
CS48L32_US1_FREQ_MASK >> CS48L32_US1_FREQ_SHIFT,
ARRAY_SIZE(cs48l32_us_freq_val),
cs48l32_us_freq_texts,
cs48l32_us_freq_val),
};
static const unsigned int cs48l32_us_in_val[] = {
0x0, 0x1, 0x2, 0x3,
};
static const struct soc_enum cs48l32_us_inmux_enum[] = {
SOC_VALUE_ENUM_SINGLE(CS48L32_US1_CONTROL,
CS48L32_US1_SRC_SHIFT,
CS48L32_US1_SRC_MASK >> CS48L32_US1_SRC_SHIFT,
ARRAY_SIZE(cs48l32_us_in_val),
cs48l32_in_texts,
cs48l32_us_in_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_US2_CONTROL,
CS48L32_US1_SRC_SHIFT,
CS48L32_US1_SRC_MASK >> CS48L32_US1_SRC_SHIFT,
ARRAY_SIZE(cs48l32_us_in_val),
cs48l32_in_texts,
cs48l32_us_in_val),
};
static const struct snd_kcontrol_new cs48l32_us_inmux[] = {
SOC_DAPM_ENUM("Ultrasonic 1 Input" , cs48l32_us_inmux_enum[0]),
SOC_DAPM_ENUM("Ultrasonic 2 Input" , cs48l32_us_inmux_enum[1]),
};
static const char * const cs48l32_us_det_thr_texts[] = {
"-6dB" , "-9dB" , "-12dB" , "-15dB" , "-18dB" , "-21dB" , "-24dB" , "-27dB" ,
};
static const struct soc_enum cs48l32_us_det_thr[] = {
SOC_ENUM_SINGLE(CS48L32_US1_DET_CONTROL,
CS48L32_US1_DET_THR_SHIFT,
ARRAY_SIZE(cs48l32_us_det_thr_texts),
cs48l32_us_det_thr_texts),
SOC_ENUM_SINGLE(CS48L32_US2_DET_CONTROL,
CS48L32_US1_DET_THR_SHIFT,
ARRAY_SIZE(cs48l32_us_det_thr_texts),
cs48l32_us_det_thr_texts),
};
static const char * const cs48l32_us_det_num_texts[] = {
"1 Sample" ,
"2 Samples" ,
"4 Samples" ,
"8 Samples" ,
"16 Samples" ,
"32 Samples" ,
"64 Samples" ,
"128 Samples" ,
"256 Samples" ,
"512 Samples" ,
"1024 Samples" ,
"2048 Samples" ,
"4096 Samples" ,
"8192 Samples" ,
"16384 Samples" ,
"32768 Samples" ,
};
static const struct soc_enum cs48l32_us_det_num[] = {
SOC_ENUM_SINGLE(CS48L32_US1_DET_CONTROL,
CS48L32_US1_DET_NUM_SHIFT,
ARRAY_SIZE(cs48l32_us_det_num_texts),
cs48l32_us_det_num_texts),
SOC_ENUM_SINGLE(CS48L32_US2_DET_CONTROL,
CS48L32_US1_DET_NUM_SHIFT,
ARRAY_SIZE(cs48l32_us_det_num_texts),
cs48l32_us_det_num_texts),
};
static const char * const cs48l32_us_det_hold_texts[] = {
"0 Samples" ,
"31 Samples" ,
"63 Samples" ,
"127 Samples" ,
"255 Samples" ,
"511 Samples" ,
"1023 Samples" ,
"2047 Samples" ,
"4095 Samples" ,
"8191 Samples" ,
"16383 Samples" ,
"32767 Samples" ,
"65535 Samples" ,
"131071 Samples" ,
"262143 Samples" ,
"524287 Samples" ,
};
static const struct soc_enum cs48l32_us_det_hold[] = {
SOC_ENUM_SINGLE(CS48L32_US1_DET_CONTROL,
CS48L32_US1_DET_HOLD_SHIFT,
ARRAY_SIZE(cs48l32_us_det_hold_texts),
cs48l32_us_det_hold_texts),
SOC_ENUM_SINGLE(CS48L32_US2_DET_CONTROL,
CS48L32_US1_DET_HOLD_SHIFT,
ARRAY_SIZE(cs48l32_us_det_hold_texts),
cs48l32_us_det_hold_texts),
};
static const struct soc_enum cs48l32_us_output_rate[] = {
SOC_VALUE_ENUM_SINGLE(CS48L32_US1_CONTROL,
CS48L32_US1_RATE_SHIFT,
CS48L32_US1_RATE_MASK >> CS48L32_US1_RATE_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_US2_CONTROL,
CS48L32_US1_RATE_SHIFT,
CS48L32_US1_RATE_MASK >> CS48L32_US1_RATE_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
};
static const char * const cs48l32_us_det_lpf_cut_texts[] = {
"1722Hz" , "833Hz" , "408Hz" , "203Hz" ,
};
static const struct soc_enum cs48l32_us_det_lpf_cut[] = {
SOC_ENUM_SINGLE(CS48L32_US1_DET_CONTROL,
CS48L32_US1_DET_LPF_CUT_SHIFT,
ARRAY_SIZE(cs48l32_us_det_lpf_cut_texts),
cs48l32_us_det_lpf_cut_texts),
SOC_ENUM_SINGLE(CS48L32_US2_DET_CONTROL,
CS48L32_US1_DET_LPF_CUT_SHIFT,
ARRAY_SIZE(cs48l32_us_det_lpf_cut_texts),
cs48l32_us_det_lpf_cut_texts),
};
static const char * const cs48l32_us_det_dcy_texts[] = {
"0 ms" , "0.79 ms" , "1.58 ms" , "3.16 ms" , "6.33 ms" , "12.67 ms" , "25.34 ms" , "50.69 ms" ,
};
static const struct soc_enum cs48l32_us_det_dcy[] = {
SOC_ENUM_SINGLE(CS48L32_US1_DET_CONTROL,
CS48L32_US1_DET_DCY_SHIFT,
ARRAY_SIZE(cs48l32_us_det_dcy_texts),
cs48l32_us_det_dcy_texts),
SOC_ENUM_SINGLE(CS48L32_US2_DET_CONTROL,
CS48L32_US1_DET_DCY_SHIFT,
ARRAY_SIZE(cs48l32_us_det_dcy_texts),
cs48l32_us_det_dcy_texts),
};
static const struct snd_kcontrol_new cs48l32_us_switch[] = {
SOC_DAPM_SINGLE("Switch" , SND_SOC_NOPM, 0, 1, 0),
SOC_DAPM_SINGLE("Switch" , SND_SOC_NOPM, 0, 1, 0),
};
static const char * const cs48l32_vol_ramp_text[] = {
"0ms/6dB" , "0.5ms/6dB" , "1ms/6dB" , "2ms/6dB" , "4ms/6dB" , "8ms/6dB" , "16ms/6dB" , "32ms/6dB" ,
};
static SOC_ENUM_SINGLE_DECL(cs48l32_in_vd_ramp,
CS48L32_INPUT_VOL_CONTROL,
CS48L32_IN_VD_RAMP_SHIFT,
cs48l32_vol_ramp_text);
static SOC_ENUM_SINGLE_DECL(cs48l32_in_vi_ramp,
CS48L32_INPUT_VOL_CONTROL,
CS48L32_IN_VI_RAMP_SHIFT,
cs48l32_vol_ramp_text);
static const char * const cs48l32_in_hpf_cut_text[] = {
"2.5Hz" , "5Hz" , "10Hz" , "20Hz" , "40Hz"
};
static SOC_ENUM_SINGLE_DECL(cs48l32_in_hpf_cut_enum,
CS48L32_INPUT_HPF_CONTROL,
CS48L32_IN_HPF_CUT_SHIFT,
cs48l32_in_hpf_cut_text);
static const char * const cs48l32_in_dmic_osr_text[] = {
"384kHz" , "768kHz" , "1.536MHz" , "2.048MHz" , "2.4576MHz" , "3.072MHz" , "6.144MHz" ,
};
static const struct soc_enum cs48l32_in_dmic_osr[] = {
SOC_ENUM_SINGLE(CS48L32_INPUT1_CONTROL1,
CS48L32_IN1_OSR_SHIFT,
ARRAY_SIZE(cs48l32_in_dmic_osr_text),
cs48l32_in_dmic_osr_text),
SOC_ENUM_SINGLE(CS48L32_INPUT2_CONTROL1,
CS48L32_IN1_OSR_SHIFT,
ARRAY_SIZE(cs48l32_in_dmic_osr_text),
cs48l32_in_dmic_osr_text),
};
static bool cs48l32_is_input_enabled(struct snd_soc_component *component,
unsigned int reg)
{
unsigned int input_active;
input_active = snd_soc_component_read(component, CS48L32_INPUT_CONTROL);
switch (reg) {
case CS48L32_IN1L_CONTROL1:
return input_active & BIT(CS48L32_IN1L_EN_SHIFT);
case CS48L32_IN1R_CONTROL1:
return input_active & BIT(CS48L32_IN1R_EN_SHIFT);
case CS48L32_IN2L_CONTROL1:
return input_active & BIT(CS48L32_IN2L_EN_SHIFT);
case CS48L32_IN2R_CONTROL1:
return input_active & BIT(CS48L32_IN2R_EN_SHIFT);
default :
return false ;
}
}
static int cs48l32_in_rate_put(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
int ret;
snd_soc_dapm_mutex_lock(dapm);
/* Cannot change rate on an active input */
if (cs48l32_is_input_enabled(component, e->reg)) {
ret = -EBUSY;
goto exit ;
}
ret = snd_soc_put_enum_double(kcontrol, ucontrol);
exit :
snd_soc_dapm_mutex_unlock(dapm);
return ret;
}
static const struct soc_enum cs48l32_input_rate[] = {
SOC_VALUE_ENUM_SINGLE(CS48L32_IN1L_CONTROL1,
CS48L32_INx_RATE_SHIFT,
CS48L32_INx_RATE_MASK >> CS48L32_INx_RATE_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_IN1R_CONTROL1,
CS48L32_INx_RATE_SHIFT,
CS48L32_INx_RATE_MASK >> CS48L32_INx_RATE_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_IN2L_CONTROL1,
CS48L32_INx_RATE_SHIFT,
CS48L32_INx_RATE_MASK >> CS48L32_INx_RATE_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_IN2R_CONTROL1,
CS48L32_INx_RATE_SHIFT,
CS48L32_INx_RATE_MASK >> CS48L32_INx_RATE_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
};
static int cs48l32_low_power_mode_put(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct soc_mixer_control *mc = (struct soc_mixer_control *)kcontrol->private_value;
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
int ret;
snd_soc_dapm_mutex_lock(dapm);
/* Cannot change rate on an active input */
if (cs48l32_is_input_enabled(component, mc->reg)) {
ret = -EBUSY;
goto exit ;
}
ret = snd_soc_put_volsw(kcontrol, ucontrol);
exit :
snd_soc_dapm_mutex_unlock(dapm);
return ret;
}
static const struct soc_enum noise_gen_rate =
SOC_VALUE_ENUM_SINGLE(CS48L32_COMFORT_NOISE_GENERATOR,
CS48L32_NOISE_GEN_RATE_SHIFT,
CS48L32_NOISE_GEN_RATE_MASK >> CS48L32_NOISE_GEN_RATE_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val);
static const char * const cs48l32_auxpdm_freq_texts[] = {
"3.072MHz" , "2.048MHz" , "1.536MHz" , "768kHz" ,
};
static SOC_ENUM_SINGLE_DECL(cs48l32_auxpdm1_freq,
CS48L32_AUXPDM1_CONTROL1,
CS48L32_AUXPDM1_FREQ_SHIFT,
cs48l32_auxpdm_freq_texts);
static SOC_ENUM_SINGLE_DECL(cs48l32_auxpdm2_freq,
CS48L32_AUXPDM2_CONTROL1,
CS48L32_AUXPDM1_FREQ_SHIFT,
cs48l32_auxpdm_freq_texts);
static const char * const cs48l32_auxpdm_src_texts[] = {
"Analog" , "IN1 Digital" , "IN2 Digital" ,
};
static SOC_ENUM_SINGLE_DECL(cs48l32_auxpdm1_in,
CS48L32_AUXPDM_CTRL2,
CS48L32_AUXPDMDAT1_SRC_SHIFT,
cs48l32_auxpdm_src_texts);
static SOC_ENUM_SINGLE_DECL(cs48l32_auxpdm2_in,
CS48L32_AUXPDM_CTRL2,
CS48L32_AUXPDMDAT2_SRC_SHIFT,
cs48l32_auxpdm_src_texts);
static const struct snd_kcontrol_new cs48l32_auxpdm_inmux[] = {
SOC_DAPM_ENUM("AUXPDM1 Input" , cs48l32_auxpdm1_in),
SOC_DAPM_ENUM("AUXPDM2 Input" , cs48l32_auxpdm2_in),
};
static const unsigned int cs48l32_auxpdm_analog_in_val[] = {
0x0, 0x1,
};
static const struct soc_enum cs48l32_auxpdm_analog_inmux_enum[] = {
SOC_VALUE_ENUM_SINGLE(CS48L32_AUXPDM1_CONTROL1,
CS48L32_AUXPDM1_SRC_SHIFT,
CS48L32_AUXPDM1_SRC_MASK >> CS48L32_AUXPDM1_SRC_SHIFT,
ARRAY_SIZE(cs48l32_auxpdm_analog_in_val),
cs48l32_in_texts,
cs48l32_auxpdm_analog_in_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_AUXPDM2_CONTROL1,
CS48L32_AUXPDM1_SRC_SHIFT,
CS48L32_AUXPDM1_SRC_MASK >> CS48L32_AUXPDM1_SRC_SHIFT,
ARRAY_SIZE(cs48l32_auxpdm_analog_in_val),
cs48l32_in_texts,
cs48l32_auxpdm_analog_in_val),
};
static const struct snd_kcontrol_new cs48l32_auxpdm_analog_inmux[] = {
SOC_DAPM_ENUM("AUXPDM1 Analog Input" , cs48l32_auxpdm_analog_inmux_enum[0]),
SOC_DAPM_ENUM("AUXPDM2 Analog Input" , cs48l32_auxpdm_analog_inmux_enum[1]),
};
static const struct snd_kcontrol_new cs48l32_auxpdm_switch[] = {
SOC_DAPM_SINGLE("Switch" , SND_SOC_NOPM, 0, 1, 0),
SOC_DAPM_SINGLE("Switch" , SND_SOC_NOPM, 0, 1, 0),
};
static const struct soc_enum cs48l32_isrc_fsh[] = {
SOC_VALUE_ENUM_SINGLE(CS48L32_ISRC1_CONTROL1,
CS48L32_ISRC1_FSH_SHIFT,
CS48L32_ISRC1_FSH_MASK >> CS48L32_ISRC1_FSH_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_ISRC2_CONTROL1,
CS48L32_ISRC1_FSH_SHIFT,
CS48L32_ISRC1_FSH_MASK >> CS48L32_ISRC1_FSH_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_ISRC3_CONTROL1,
CS48L32_ISRC1_FSH_SHIFT,
CS48L32_ISRC1_FSH_MASK >> CS48L32_ISRC1_FSH_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
};
static const struct soc_enum cs48l32_isrc_fsl[] = {
SOC_VALUE_ENUM_SINGLE(CS48L32_ISRC1_CONTROL1,
CS48L32_ISRC1_FSL_SHIFT,
CS48L32_ISRC1_FSL_MASK >> CS48L32_ISRC1_FSL_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_ISRC2_CONTROL1,
CS48L32_ISRC1_FSL_SHIFT,
CS48L32_ISRC1_FSL_MASK >> CS48L32_ISRC1_FSL_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(CS48L32_ISRC3_CONTROL1,
CS48L32_ISRC1_FSL_SHIFT,
CS48L32_ISRC1_FSL_MASK >> CS48L32_ISRC1_FSL_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val),
};
static const struct soc_enum cs48l32_fx_rate =
SOC_VALUE_ENUM_SINGLE(CS48L32_FX_SAMPLE_RATE,
CS48L32_FX_RATE_SHIFT,
CS48L32_FX_RATE_MASK >> CS48L32_FX_RATE_SHIFT,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text,
cs48l32_rate_val);
static const char * const cs48l32_lhpf_mode_text[] = {
"Low-pass" , "High-pass"
};
static const struct soc_enum cs48l32_lhpf_mode[] = {
SOC_ENUM_SINGLE(CS48L32_LHPF_CONTROL2, 0,
ARRAY_SIZE(cs48l32_lhpf_mode_text), cs48l32_lhpf_mode_text),
SOC_ENUM_SINGLE(CS48L32_LHPF_CONTROL2, 1,
ARRAY_SIZE(cs48l32_lhpf_mode_text), cs48l32_lhpf_mode_text),
SOC_ENUM_SINGLE(CS48L32_LHPF_CONTROL2, 2,
ARRAY_SIZE(cs48l32_lhpf_mode_text), cs48l32_lhpf_mode_text),
SOC_ENUM_SINGLE(CS48L32_LHPF_CONTROL2, 3,
ARRAY_SIZE(cs48l32_lhpf_mode_text), cs48l32_lhpf_mode_text),
};
static int cs48l32_lhpf_coeff_put(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
__be32 *data = (__be32 *)ucontrol->value.bytes.data;
s16 val = (s16)be32_to_cpu(*data);
if (abs(val) > CS48L32_LHPF_MAX_COEFF) {
dev_err(cs48l32_codec->core.dev, "Rejecting unstable LHPF coefficients\n" );
return -EINVAL;
}
return snd_soc_bytes_put(kcontrol, ucontrol);
}
static const char * const cs48l32_eq_mode_text[] = {
"Low-pass" , "High-pass" ,
};
static const struct soc_enum cs48l32_eq_mode[] = {
SOC_ENUM_SINGLE(CS48L32_EQ_CONTROL2, 0,
ARRAY_SIZE(cs48l32_eq_mode_text),
cs48l32_eq_mode_text),
SOC_ENUM_SINGLE(CS48L32_EQ_CONTROL2, 1,
ARRAY_SIZE(cs48l32_eq_mode_text),
cs48l32_eq_mode_text),
SOC_ENUM_SINGLE(CS48L32_EQ_CONTROL2, 2,
ARRAY_SIZE(cs48l32_eq_mode_text),
cs48l32_eq_mode_text),
SOC_ENUM_SINGLE(CS48L32_EQ_CONTROL2, 3,
ARRAY_SIZE(cs48l32_eq_mode_text),
cs48l32_eq_mode_text),
};
static int cs48l32_eq_mode_get(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct soc_enum *e = (struct soc_enum *) kcontrol->private_value;
unsigned int item;
item = snd_soc_enum_val_to_item(e, cs48l32_codec->eq_mode[e->shift_l]);
ucontrol->value.enumerated.item[0] = item;
return 0;
}
static int cs48l32_eq_mode_put(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct soc_enum *e = (struct soc_enum *) kcontrol->private_value;
unsigned int *item = ucontrol->value.enumerated.item;
unsigned int val;
bool changed = false ;
if (item[0] >= e->items)
return -EINVAL;
val = snd_soc_enum_item_to_val(e, item[0]);
snd_soc_dapm_mutex_lock(dapm);
if (cs48l32_codec->eq_mode[e->shift_l] != val) {
cs48l32_codec->eq_mode[e->shift_l] = val;
changed = true ;
}
snd_soc_dapm_mutex_unlock(dapm);
return changed;
}
static int cs48l32_eq_coeff_info(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_info *uinfo)
{
struct cs48l32_eq_control *ctl = (void *) kcontrol->private_value;
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
uinfo->count = 1;
uinfo->value.integer.min = 0;
uinfo->value.integer.max = ctl->max;
return 0;
}
static int cs48l32_eq_coeff_get(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct cs48l32_eq_control *params = (void *)kcontrol->private_value;
__be16 *coeffs;
unsigned int coeff_idx;
int block_idx;
block_idx = ((int ) params->block_base - (int ) CS48L32_EQ1_BAND1_COEFF1);
block_idx /= (CS48L32_EQ2_BAND1_COEFF1 - CS48L32_EQ1_BAND1_COEFF1);
coeffs = &cs48l32_codec->eq_coefficients[block_idx][0];
coeff_idx = (params->reg - params->block_base) / 2;
/* High __be16 is in [coeff_idx] and low __be16 in [coeff_idx + 1] */
if (params->shift == 0)
coeff_idx++;
ucontrol->value.integer.value[0] = be16_to_cpu(coeffs[coeff_idx]);
return 0;
}
static int cs48l32_eq_coeff_put(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct cs48l32_eq_control *params = (void *)kcontrol->private_value;
__be16 *coeffs;
unsigned int coeff_idx;
int block_idx;
block_idx = ((int ) params->block_base - (int ) CS48L32_EQ1_BAND1_COEFF1);
block_idx /= (CS48L32_EQ2_BAND1_COEFF1 - CS48L32_EQ1_BAND1_COEFF1);
coeffs = &cs48l32_codec->eq_coefficients[block_idx][0];
coeff_idx = (params->reg - params->block_base) / 2;
/* Put high __be16 in [coeff_idx] and low __be16 in [coeff_idx + 1] */
if (params->shift == 0)
coeff_idx++;
snd_soc_dapm_mutex_lock(dapm);
coeffs[coeff_idx] = cpu_to_be16(ucontrol->value.integer.value[0]);
snd_soc_dapm_mutex_unlock(dapm);
return 0;
}
static const struct snd_kcontrol_new cs48l32_drc_activity_output_mux[] = {
SOC_DAPM_SINGLE("Switch" , SND_SOC_NOPM, 0, 1, 0),
SOC_DAPM_SINGLE("Switch" , SND_SOC_NOPM, 0, 1, 0),
};
static const struct snd_kcontrol_new cs48l32_dsp_trigger_output_mux[] = {
SOC_DAPM_SINGLE("Switch" , SND_SOC_NOPM, 0, 1, 0),
};
static int cs48l32_dsp_rate_get(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct soc_enum *e = (struct soc_enum *) kcontrol->private_value;
unsigned int cached_rate;
const unsigned int rate_num = e->mask;
int item;
if (rate_num >= ARRAY_SIZE(cs48l32_codec->dsp_dma_rates))
return -EINVAL;
cached_rate = cs48l32_codec->dsp_dma_rates[rate_num];
item = snd_soc_enum_val_to_item(e, cached_rate);
ucontrol->value.enumerated.item[0] = item;
return 0;
}
static int cs48l32_dsp_rate_put(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol);
struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct soc_enum *e = (struct soc_enum *) kcontrol->private_value;
const unsigned int rate_num = e->mask;
const unsigned int item = ucontrol->value.enumerated.item[0];
unsigned int val;
bool changed = false ;
if (item >= e->items)
return -EINVAL;
if (rate_num >= ARRAY_SIZE(cs48l32_codec->dsp_dma_rates))
return -EINVAL;
val = snd_soc_enum_item_to_val(e, item);
snd_soc_dapm_mutex_lock(dapm);
if (cs48l32_codec->dsp_dma_rates[rate_num] != val) {
cs48l32_codec->dsp_dma_rates[rate_num] = val;
changed = true ;
}
snd_soc_dapm_mutex_unlock(dapm);
return changed;
}
static const struct soc_enum cs48l32_dsp_rate_enum[] = {
/* RX rates */
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
0,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
1,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
2,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
3,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
4,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
5,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
6,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
7,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
/* TX rates */
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
8,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
9,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
10,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
11,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
12,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
13,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
14,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
SOC_VALUE_ENUM_SINGLE(SND_SOC_NOPM, 0,
15,
ARRAY_SIZE(cs48l32_rate_text),
cs48l32_rate_text, cs48l32_rate_val),
};
static int cs48l32_dsp_pre_run(struct wm_adsp *dsp)
{
struct cs48l32_codec *cs48l32_codec = container_of(dsp, struct cs48l32_codec, dsp);
unsigned int reg;
const u8 *rate = cs48l32_codec->dsp_dma_rates;
int i;
reg = dsp->cs_dsp.base + CS48L32_HALO_SAMPLE_RATE_RX1;
for (i = 0; i < CS48L32_DSP_N_RX_CHANNELS; ++i) {
regmap_update_bits(dsp->cs_dsp.regmap, reg, CS48L32_HALO_DSP_RATE_MASK, *rate);
reg += 8;
rate++;
}
reg = dsp->cs_dsp.base + CS48L32_HALO_SAMPLE_RATE_TX1;
for (i = 0; i < CS48L32_DSP_N_TX_CHANNELS; ++i) {
regmap_update_bits(dsp->cs_dsp.regmap, reg, CS48L32_HALO_DSP_RATE_MASK, *rate);
reg += 8;
rate++;
}
usleep_range(300, 600);
return 0;
}
static void cs48l32_dsp_memory_disable(struct cs48l32_codec *cs48l32_codec,
const struct cs48l32_dsp_power_regs *regs)
{
struct regmap *regmap = cs48l32_codec->core.regmap;
int i, j, ret;
for (i = 0; i < regs->n_pwd; ++i) {
ret = regmap_write(regmap, regs->pwd[i], 0);
if (ret)
goto err;
}
for (i = 0; i < regs->n_ext; ++i) {
for (j = regs->ext[i].start; j <= regs->ext[i].end; j += 4) {
ret = regmap_write(regmap, j, 0);
if (ret)
goto err;
}
}
return ;
err:
dev_warn(cs48l32_codec->core.dev, "Failed to write SRAM enables (%d)\n" , ret);
}
static int cs48l32_dsp_memory_enable(struct cs48l32_codec *cs48l32_codec,
const struct cs48l32_dsp_power_regs *regs)
{
struct regmap *regmap = cs48l32_codec->core.regmap;
int i, j, ret;
/* disable power-off */
for (i = 0; i < regs->n_ext; ++i) {
for (j = regs->ext[i].start; j <= regs->ext[i].end; j += 4) {
ret = regmap_write(regmap, j, 0x3);
if (ret)
goto err;
}
}
/* power-up the banks in sequence */
for (i = 0; i < regs->n_pwd; ++i) {
ret = regmap_write(regmap, regs->pwd[i], 0x1);
if (ret)
goto err;
udelay(1); /* allow bank to power-up */
ret = regmap_write(regmap, regs->pwd[i], 0x3);
if (ret)
goto err;
udelay(1); /* allow bank to power-up */
}
return 0;
err:
dev_err(cs48l32_codec->core.dev, "Failed to write SRAM enables (%d)\n" , ret);
cs48l32_dsp_memory_disable(cs48l32_codec, regs);
return ret;
}
static int cs48l32_dsp_freq_update(struct snd_soc_dapm_widget *w, unsigned int freq_reg,
unsigned int freqsel_reg)
{
struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm);
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct regmap *regmap = cs48l32_codec->core.regmap;
struct wm_adsp *dsp = &cs48l32_codec->dsp;
int ret;
unsigned int freq, freq_sel, freq_sts;
if (!freq_reg)
return -EINVAL;
ret = regmap_read(regmap, freq_reg, &freq);
if (ret) {
dev_err(component->dev, "Failed to read #%x: %d\n" , freq_reg, ret);
return ret;
}
if (freqsel_reg) {
freq_sts = (freq & CS48L32_SYSCLK_FREQ_STS_MASK) >> CS48L32_SYSCLK_FREQ_STS_SHIFT;
ret = regmap_read(regmap, freqsel_reg, &freq_sel);
if (ret) {
dev_err(component->dev, "Failed to read #%x: %d\n" , freqsel_reg, ret);
return ret;
}
freq_sel = (freq_sel & CS48L32_SYSCLK_FREQ_MASK) >> CS48L32_SYSCLK_FREQ_SHIFT;
if (freq_sts != freq_sel) {
dev_err(component->dev, "SYSCLK FREQ (#%x) != FREQ STS (#%x)\n" ,
freq_sel, freq_sts);
return -ETIMEDOUT;
}
}
freq &= CS48L32_DSP_CLK_FREQ_MASK;
freq >>= CS48L32_DSP_CLK_FREQ_SHIFT;
ret = regmap_write(dsp->cs_dsp.regmap,
dsp->cs_dsp.base + CS48L32_DSP_CLOCK_FREQ_OFFS, freq);
if (ret) {
dev_err(component->dev, "Failed to set HALO clock freq: %d\n" , ret);
return ret;
}
return 0;
}
static int cs48l32_dsp_freq_ev(struct snd_soc_dapm_widget *w,
struct snd_kcontrol *kcontrol, int event)
{
switch (event) {
case SND_SOC_DAPM_POST_PMU:
return cs48l32_dsp_freq_update(w, CS48L32_SYSTEM_CLOCK2, CS48L32_SYSTEM_CLOCK1);
default :
return 0;
}
}
static irqreturn_t cs48l32_irq(int irq, void *data)
{
static const unsigned int eint1_regs[] = {
CS48L32_IRQ1_EINT_9, CS48L32_IRQ1_MASK_9,
CS48L32_IRQ1_EINT_7, CS48L32_IRQ1_MASK_7
};
u32 reg_vals[4];
struct cs48l32_codec *cs48l32_codec = data;
struct regmap *regmap = cs48l32_codec->core.regmap;
irqreturn_t result = IRQ_NONE;
unsigned int eint_pending;
int i, ret;
static_assert(ARRAY_SIZE(eint1_regs) == ARRAY_SIZE(reg_vals));
ret = pm_runtime_resume_and_get(cs48l32_codec->core.dev);
if (ret) {
dev_warn(cs48l32_codec->core.dev, "irq could not get pm runtime: %d\n" , ret);
return IRQ_NONE;
}
ret = regmap_read(regmap, CS48L32_IRQ1_STATUS, &eint_pending);
if (ret) {
dev_warn(cs48l32_codec->core.dev, "Read IRQ1_STATUS failed: %d\n" , ret);
return IRQ_NONE;
}
if ((eint_pending & CS48L32_IRQ1_STS_MASK) == 0)
goto out;
ret = regmap_multi_reg_read(regmap, eint1_regs, reg_vals, ARRAY_SIZE(reg_vals));
if (ret) {
dev_warn(cs48l32_codec->core.dev, "Read IRQ regs failed: %d\n" , ret);
return IRQ_NONE;
}
for (i = 0; i < ARRAY_SIZE(reg_vals); i += 2) {
reg_vals[i] &= ~reg_vals[i + 1];
regmap_write(regmap, eint1_regs[i], reg_vals[i]);
}
if (reg_vals[0] & CS48L32_DSP1_IRQ0_EINT1_MASK)
wm_adsp_compr_handle_irq(&cs48l32_codec->dsp);
if (reg_vals[2] & CS48L32_DSP1_MPU_ERR_EINT1_MASK) {
dev_warn(cs48l32_codec->core.dev, "MPU err IRQ\n" );
wm_halo_bus_error(irq, &cs48l32_codec->dsp);
}
if (reg_vals[2] & CS48L32_DSP1_WDT_EXPIRE_EINT1_MASK) {
dev_warn(cs48l32_codec->core.dev, "WDT expire IRQ\n" );
wm_halo_wdt_expire(irq, &cs48l32_codec->dsp);
}
result = IRQ_HANDLED;
out:
pm_runtime_put_autosuspend(cs48l32_codec->core.dev);
return result;
}
static int cs48l32_get_dspclk_setting(struct cs48l32_codec *cs48l32_codec, unsigned int freq,
int src, unsigned int *val)
{
freq /= 15625; /* convert to 1/64ths of 1MHz */
*val |= freq << CS48L32_DSP_CLK_FREQ_SHIFT;
return 0;
}
static int cs48l32_get_sysclk_setting(unsigned int freq)
{
switch (freq) {
case 0:
case 5644800:
case 6144000:
return CS48L32_SYSCLK_RATE_6MHZ;
case 11289600:
case 12288000:
return CS48L32_SYSCLK_RATE_12MHZ << CS48L32_SYSCLK_FREQ_SHIFT;
case 22579200:
case 24576000:
return CS48L32_SYSCLK_RATE_24MHZ << CS48L32_SYSCLK_FREQ_SHIFT;
case 45158400:
case 49152000:
return CS48L32_SYSCLK_RATE_49MHZ << CS48L32_SYSCLK_FREQ_SHIFT;
case 90316800:
case 98304000:
return CS48L32_SYSCLK_RATE_98MHZ << CS48L32_SYSCLK_FREQ_SHIFT;
default :
return -EINVAL;
}
}
static int cs48l32_set_pdm_fllclk(struct snd_soc_component *component, int source)
{
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct regmap *regmap = cs48l32_codec->core.regmap;
unsigned int val;
switch (source) {
case CS48L32_PDMCLK_SRC_IN1_PDMCLK:
case CS48L32_PDMCLK_SRC_IN2_PDMCLK:
case CS48L32_PDMCLK_SRC_IN3_PDMCLK:
case CS48L32_PDMCLK_SRC_IN4_PDMCLK:
case CS48L32_PDMCLK_SRC_AUXPDM1_CLK:
case CS48L32_PDMCLK_SRC_AUXPDM2_CLK:
val = source << CS48L32_PDM_FLLCLK_SRC_SHIFT;
break ;
default :
dev_err(cs48l32_codec->core.dev, "Invalid PDM FLLCLK src %d\n" , source);
return -EINVAL;
}
return regmap_update_bits(regmap, CS48L32_INPUT_CONTROL2,
CS48L32_PDM_FLLCLK_SRC_MASK, val);
}
static int cs48l32_set_sysclk(struct snd_soc_component *component, int clk_id, int source,
unsigned int freq, int dir)
{
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct regmap *regmap = cs48l32_codec->core.regmap;
char *name;
unsigned int reg;
unsigned int mask = CS48L32_SYSCLK_SRC_MASK;
unsigned int val = source << CS48L32_SYSCLK_SRC_SHIFT;
int clk_freq_sel, *clk;
switch (clk_id) {
case CS48L32_CLK_SYSCLK_1:
name = "SYSCLK" ;
reg = CS48L32_SYSTEM_CLOCK1;
clk = &cs48l32_codec->sysclk;
clk_freq_sel = cs48l32_get_sysclk_setting(freq);
mask |= CS48L32_SYSCLK_FREQ_MASK | CS48L32_SYSCLK_FRAC_MASK;
break ;
case CS48L32_CLK_DSPCLK:
name = "DSPCLK" ;
reg = CS48L32_DSP_CLOCK1;
clk = &cs48l32_codec->dspclk;
clk_freq_sel = cs48l32_get_dspclk_setting(cs48l32_codec, freq, source, &val);
mask |= CS48L32_DSP_CLK_FREQ_MASK;
break ;
case CS48L32_CLK_PDM_FLLCLK:
return cs48l32_set_pdm_fllclk(component, source);
default :
return -EINVAL;
}
if (clk_freq_sel < 0) {
dev_err(cs48l32_codec->core.dev, "Failed to get %s setting for %dHZ\n" , name, freq);
return clk_freq_sel;
}
*clk = freq;
if (freq == 0) {
dev_dbg(cs48l32_codec->core.dev, "%s cleared\n" , name);
return 0;
}
val |= clk_freq_sel;
if (freq % 6144000)
val |= CS48L32_SYSCLK_FRAC_MASK;
dev_dbg(cs48l32_codec->core.dev, "%s set to %uHz" , name, freq);
return regmap_update_bits(regmap, reg, mask, val);
}
static int cs48l32_is_enabled_fll(struct cs48l32_fll *fll, int base)
{
struct regmap *regmap = fll->codec->core.regmap;
unsigned int reg;
int ret;
ret = regmap_read(regmap, base + CS48L32_FLL_CONTROL1_OFFS, ®);
if (ret != 0) {
cs48l32_fll_err(fll, "Failed to read current state: %d\n" , ret);
return ret;
}
return reg & CS48L32_FLL_EN_MASK;
}
static int cs48l32_wait_for_fll(struct cs48l32_fll *fll, bool requested)
{
struct regmap *regmap = fll->codec->core.regmap;
unsigned int val = 0;
int i;
cs48l32_fll_dbg(fll, "Waiting for FLL...\n" );
for (i = 0; i < 30; i++) {
regmap_read(regmap, fll->sts_addr, &val);
if (!!(val & fll->sts_mask) == requested)
return 0;
switch (i) {
case 0 ... 5:
usleep_range(75, 125);
break ;
case 6 ... 20:
usleep_range(750, 1250);
break ;
default :
fsleep(20000);
break ;
}
}
cs48l32_fll_warn(fll, "Timed out waiting for %s\n" , requested ? "lock" : "unlock" );
return -ETIMEDOUT;
}
static int cs48l32_fllhj_disable(struct cs48l32_fll *fll)
{
struct cs48l32 *cs48l32 = &fll->codec->core;
bool change;
cs48l32_fll_dbg(fll, "Disabling FLL\n" );
/*
* Disable lockdet, but don't set ctrl_upd update bit. This allows the
* lock status bit to clear as normal, but should the FLL be enabled
* again due to a control clock being required, the lock won't re-assert
* as the FLL config registers are automatically applied when the FLL
* enables.
*/
regmap_set_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL1_OFFS,
CS48L32_FLL_HOLD_MASK);
regmap_clear_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL2_OFFS,
CS48L32_FLL_LOCKDET_MASK);
regmap_set_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL5_OFFS,
CS48L32_FLL_FRC_INTEG_UPD_MASK);
regmap_update_bits_check(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL1_OFFS,
CS48L32_FLL_EN_MASK,
0,
&change);
cs48l32_wait_for_fll(fll, false );
/*
* ctrl_up gates the writes to all the fll's registers, setting it to 0
* here ensures that after a runtime suspend/resume cycle when one
* enables the fll then ctrl_up is the last bit that is configured
* by the fll enable code rather than the cache sync operation which
* would have updated it much earlier before writing out all fll
* registers
*/
regmap_clear_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL1_OFFS,
CS48L32_FLL_CTRL_UPD_MASK);
if (change)
pm_runtime_put_autosuspend(cs48l32->dev);
return 0;
}
static int cs48l32_fllhj_apply(struct cs48l32_fll *fll, int fin)
{
struct regmap *regmap = fll->codec->core.regmap;
int refdiv, fref, fout, lockdet_thr, fbdiv, fllgcd;
bool frac = false ;
unsigned int fll_n, min_n, max_n, ratio, theta, lambda, hp;
unsigned int gains, num;
cs48l32_fll_dbg(fll, "fin=%d, fout=%d\n" , fin, fll->fout);
for (refdiv = 0; refdiv < 4; refdiv++) {
if ((fin / (1 << refdiv)) <= CS48L32_FLLHJ_MAX_THRESH)
break ;
}
fref = fin / (1 << refdiv);
fout = fll->fout;
frac = fout % fref;
/*
* Use simple heuristic approach to find a configuration that
* should work for most input clocks.
*/
if (fref < CS48L32_FLLHJ_LOW_THRESH) {
lockdet_thr = 2;
gains = CS48L32_FLLHJ_LOW_GAINS;
if (frac)
fbdiv = 256;
else
fbdiv = 4;
} else if (fref < CS48L32_FLLHJ_MID_THRESH) {
lockdet_thr = 8;
gains = CS48L32_FLLHJ_MID_GAINS;
fbdiv = (frac) ? 16 : 2;
} else {
lockdet_thr = 8;
gains = CS48L32_FLLHJ_HIGH_GAINS;
fbdiv = 1;
}
/* Use high performance mode for fractional configurations. */
if (frac) {
hp = 3;
min_n = CS48L32_FLLHJ_FRAC_MIN_N;
max_n = CS48L32_FLLHJ_FRAC_MAX_N;
} else {
if (fref < CS48L32_FLLHJ_LP_INT_MODE_THRESH)
hp = 0;
else
hp = 1;
min_n = CS48L32_FLLHJ_INT_MIN_N;
max_n = CS48L32_FLLHJ_INT_MAX_N;
}
ratio = fout / fref;
cs48l32_fll_dbg(fll, "refdiv=%d, fref=%d, frac:%d\n" , refdiv, fref, frac);
while (ratio / fbdiv < min_n) {
fbdiv /= 2;
if (fbdiv < min_n) {
cs48l32_fll_err(fll, "FBDIV (%u) < minimum N (%u)\n" , fbdiv, min_n);
return -EINVAL;
}
}
while (frac && (ratio / fbdiv > max_n)) {
fbdiv *= 2;
if (fbdiv >= 1024) {
cs48l32_fll_err(fll, "FBDIV (%u) >= 1024\n" , fbdiv);
return -EINVAL;
}
}
cs48l32_fll_dbg(fll, "lockdet=%d, hp=#%x, fbdiv:%d\n" , lockdet_thr, hp, fbdiv);
/* Calculate N.K values */
fllgcd = gcd(fout, fbdiv * fref);
num = fout / fllgcd;
lambda = (fref * fbdiv) / fllgcd;
fll_n = num / lambda;
theta = num % lambda;
cs48l32_fll_dbg(fll, "fll_n=%d, gcd=%d, theta=%d, lambda=%d\n" ,
fll_n, fllgcd, theta, lambda);
/* Some sanity checks before any registers are written. */
if (fll_n < min_n || fll_n > max_n) {
cs48l32_fll_err(fll, "N not in valid %s mode range %d-%d: %d\n" ,
frac ? "fractional" : "integer" , min_n, max_n, fll_n);
return -EINVAL;
}
if (fbdiv < 1 || (frac && fbdiv >= 1024) || (!frac && fbdiv >= 256)) {
cs48l32_fll_err(fll, "Invalid fbdiv for %s mode (%u)\n" ,
frac ? "fractional" : "integer" , fbdiv);
return -EINVAL;
}
/* clear the ctrl_upd bit to guarantee we write to it later. */
regmap_update_bits(regmap,
fll->base + CS48L32_FLL_CONTROL2_OFFS,
CS48L32_FLL_LOCKDET_THR_MASK |
CS48L32_FLL_PHASEDET_MASK |
CS48L32_FLL_REFCLK_DIV_MASK |
CS48L32_FLL_N_MASK |
CS48L32_FLL_CTRL_UPD_MASK,
(lockdet_thr << CS48L32_FLL_LOCKDET_THR_SHIFT) |
(1 << CS48L32_FLL_PHASEDET_SHIFT) |
(refdiv << CS48L32_FLL_REFCLK_DIV_SHIFT) |
(fll_n << CS48L32_FLL_N_SHIFT));
regmap_update_bits(regmap,
fll->base + CS48L32_FLL_CONTROL3_OFFS,
CS48L32_FLL_LAMBDA_MASK |
CS48L32_FLL_THETA_MASK,
(lambda << CS48L32_FLL_LAMBDA_SHIFT) |
(theta << CS48L32_FLL_THETA_SHIFT));
regmap_update_bits(regmap,
fll->base + CS48L32_FLL_CONTROL4_OFFS,
(0xffff << CS48L32_FLL_FD_GAIN_COARSE_SHIFT) |
CS48L32_FLL_HP_MASK |
CS48L32_FLL_FB_DIV_MASK,
(gains << CS48L32_FLL_FD_GAIN_COARSE_SHIFT) |
(hp << CS48L32_FLL_HP_SHIFT) |
(fbdiv << CS48L32_FLL_FB_DIV_SHIFT));
return 0;
}
static int cs48l32_fllhj_enable(struct cs48l32_fll *fll)
{
struct cs48l32 *cs48l32 = &fll->codec->core;
int already_enabled = cs48l32_is_enabled_fll(fll, fll->base);
int ret;
if (already_enabled < 0)
return already_enabled;
if (!already_enabled)
pm_runtime_get_sync(cs48l32->dev);
cs48l32_fll_dbg(fll, "Enabling FLL, initially %s\n" ,
str_enabled_disabled(already_enabled));
/* FLLn_HOLD must be set before configuring any registers */
regmap_set_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL1_OFFS,
CS48L32_FLL_HOLD_MASK);
/* Apply refclk */
ret = cs48l32_fllhj_apply(fll, fll->ref_freq);
if (ret) {
cs48l32_fll_err(fll, "Failed to set FLL: %d\n" , ret);
goto out;
}
regmap_update_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL2_OFFS,
CS48L32_FLL_REFCLK_SRC_MASK,
fll->ref_src << CS48L32_FLL_REFCLK_SRC_SHIFT);
regmap_set_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL1_OFFS,
CS48L32_FLL_EN_MASK);
out:
regmap_set_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL2_OFFS,
CS48L32_FLL_LOCKDET_MASK);
regmap_set_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL1_OFFS,
CS48L32_FLL_CTRL_UPD_MASK);
/* Release the hold so that flln locks to external frequency */
regmap_clear_bits(cs48l32->regmap,
fll->base + CS48L32_FLL_CONTROL1_OFFS,
CS48L32_FLL_HOLD_MASK);
if (!already_enabled)
cs48l32_wait_for_fll(fll, true );
return 0;
}
static int cs48l32_fllhj_validate(struct cs48l32_fll *fll,
unsigned int ref_in,
unsigned int fout)
{
if (fout && !ref_in) {
cs48l32_fll_err(fll, "fllout set without valid input clk\n" );
return -EINVAL;
}
if (fll->fout && fout != fll->fout) {
cs48l32_fll_err(fll, "Can't change output on active FLL\n" );
return -EINVAL;
}
if (ref_in / CS48L32_FLL_MAX_REFDIV > CS48L32_FLLHJ_MAX_THRESH) {
cs48l32_fll_err(fll, "Can't scale %dMHz to <=13MHz\n" , ref_in);
return -EINVAL;
}
if (fout > CS48L32_FLL_MAX_FOUT) {
cs48l32_fll_err(fll, "Fout=%dMHz exceeds maximum %dMHz\n" ,
fout, CS48L32_FLL_MAX_FOUT);
return -EINVAL;
}
return 0;
}
static int cs48l32_fllhj_set_refclk(struct cs48l32_fll *fll, int source,
unsigned int fin, unsigned int fout)
{
int ret = 0;
if (fll->ref_src == source && fll->ref_freq == fin && fll->fout == fout)
return 0;
if (fin && fout && cs48l32_fllhj_validate(fll, fin, fout))
return -EINVAL;
fll->ref_src = source;
fll->ref_freq = fin;
fll->fout = fout;
if (fout)
ret = cs48l32_fllhj_enable(fll);
else
cs48l32_fllhj_disable(fll);
return ret;
}
static int cs48l32_init_fll(struct cs48l32_fll *fll)
{
fll->ref_src = CS48L32_FLL_SRC_NONE;
return 0;
}
static int cs48l32_set_fll(struct snd_soc_component *component, int fll_id,
int source, unsigned int fref, unsigned int fout)
{
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
switch (fll_id) {
case CS48L32_FLL1_REFCLK:
break ;
default :
return -EINVAL;
}
return cs48l32_fllhj_set_refclk(&cs48l32_codec->fll, source, fref, fout);
}
static int cs48l32_asp_dai_probe(struct snd_soc_dai *dai)
{
struct snd_soc_component *component = dai->component;
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct regmap *regmap = cs48l32_codec->core.regmap;
unsigned int pin_reg, last_pin_reg, hiz_reg;
switch (dai->id) {
case 1:
pin_reg = CS48L32_GPIO3_CTRL1;
hiz_reg = CS48L32_ASP1_CONTROL3;
break ;
case 2:
pin_reg = CS48L32_GPIO7_CTRL1;
hiz_reg = CS48L32_ASP2_CONTROL3;
break ;
default :
return -EINVAL;
}
for (last_pin_reg = pin_reg + 12; pin_reg <= last_pin_reg; ++pin_reg)
regmap_clear_bits(regmap, pin_reg, CS48L32_GPIOX_CTRL1_FN_MASK);
/* DOUT high-impendance when not transmitting */
regmap_set_bits(regmap, hiz_reg, CS48L32_ASP_DOUT_HIZ_MASK);
return 0;
}
static int cs48l32_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
{
struct snd_soc_component *component = dai->component;
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct regmap *regmap = cs48l32_codec->core.regmap;
unsigned int val = 0U;
unsigned int base = dai->driver->base;
unsigned int mask = CS48L32_ASP_FMT_MASK | CS48L32_ASP_BCLK_INV_MASK |
CS48L32_ASP_BCLK_MSTR_MASK |
CS48L32_ASP_FSYNC_INV_MASK |
CS48L32_ASP_FSYNC_MSTR_MASK;
switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
case SND_SOC_DAIFMT_DSP_A:
val |= (CS48L32_ASP_FMT_DSP_MODE_A << CS48L32_ASP_FMT_SHIFT);
break ;
case SND_SOC_DAIFMT_DSP_B:
if ((fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) != SND_SOC_DAIFMT_BP_FP) {
cs48l32_asp_err(dai, "DSP_B cannot be clock consumer\n" );
return -EINVAL;
}
val |= (CS48L32_ASP_FMT_DSP_MODE_B << CS48L32_ASP_FMT_SHIFT);
break ;
case SND_SOC_DAIFMT_I2S:
val |= (CS48L32_ASP_FMT_I2S_MODE << CS48L32_ASP_FMT_SHIFT);
break ;
case SND_SOC_DAIFMT_LEFT_J:
if ((fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) != SND_SOC_DAIFMT_BP_FP) {
cs48l32_asp_err(dai, "LEFT_J cannot be clock consumer\n" );
return -EINVAL;
}
val |= (CS48L32_ASP_FMT_LEFT_JUSTIFIED_MODE << CS48L32_ASP_FMT_SHIFT);
break ;
default :
cs48l32_asp_err(dai, "Unsupported DAI format %d\n" ,
fmt & SND_SOC_DAIFMT_FORMAT_MASK);
return -EINVAL;
}
switch (fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) {
case SND_SOC_DAIFMT_BC_FC:
break ;
case SND_SOC_DAIFMT_BC_FP:
val |= CS48L32_ASP_FSYNC_MSTR_MASK;
break ;
case SND_SOC_DAIFMT_BP_FC:
val |= CS48L32_ASP_BCLK_MSTR_MASK;
break ;
case SND_SOC_DAIFMT_BP_FP:
val |= CS48L32_ASP_BCLK_MSTR_MASK;
val |= CS48L32_ASP_FSYNC_MSTR_MASK;
break ;
default :
cs48l32_asp_err(dai, "Unsupported clock direction %d\n" ,
fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK);
return -EINVAL;
}
switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
case SND_SOC_DAIFMT_NB_NF:
break ;
case SND_SOC_DAIFMT_IB_IF:
val |= CS48L32_ASP_BCLK_INV_MASK;
val |= CS48L32_ASP_FSYNC_INV_MASK;
break ;
case SND_SOC_DAIFMT_IB_NF:
val |= CS48L32_ASP_BCLK_INV_MASK;
break ;
case SND_SOC_DAIFMT_NB_IF:
val |= CS48L32_ASP_FSYNC_INV_MASK;
break ;
default :
return -EINVAL;
}
regmap_update_bits(regmap, base + CS48L32_ASP_CONTROL2, mask, val);
return 0;
}
static const struct {
u32 freq;
u32 id;
} cs48l32_sclk_rates[] = {
{ 128000, 12 },
{ 176400, 13 },
{ 192000, 14 },
{ 256000, 15 },
{ 352800, 16 },
{ 384000, 17 },
{ 512000, 18 },
{ 705600, 19 },
{ 768000, 21 },
{ 1024000, 23 },
{ 1411200, 25 },
{ 1536000, 27 },
{ 2048000, 29 },
{ 2822400, 31 },
{ 3072000, 33 },
{ 4096000, 36 },
{ 5644800, 38 },
{ 6144000, 40 },
{ 8192000, 47 },
{ 11289600, 49 },
{ 12288000, 51 },
{ 22579200, 57 },
{ 24576000, 59 },
};
#define CS48L32_48K_RATE_MASK 0x0e00fe
#define CS48L32_44K1_RATE_MASK 0x00fe00
#define CS48L32_RATE_MASK (CS48L32_48K_RATE_MASK | CS48L32_44K1_RATE_MASK)
static const unsigned int cs48l32_sr_vals[] = {
0,
12000, /* CS48L32_48K_RATE_MASK */
24000, /* CS48L32_48K_RATE_MASK */
48000, /* CS48L32_48K_RATE_MASK */
96000, /* CS48L32_48K_RATE_MASK */
192000, /* CS48L32_48K_RATE_MASK */
384000, /* CS48L32_48K_RATE_MASK */
768000, /* CS48L32_48K_RATE_MASK */
0,
11025, /* CS48L32_44K1_RATE_MASK */
22050, /* CS48L32_44K1_RATE_MASK */
44100, /* CS48L32_44K1_RATE_MASK */
88200, /* CS48L32_44K1_RATE_MASK */
176400, /* CS48L32_44K1_RATE_MASK */
352800, /* CS48L32_44K1_RATE_MASK */
705600, /* CS48L32_44K1_RATE_MASK */
0,
8000, /* CS48L32_48K_RATE_MASK */
16000, /* CS48L32_48K_RATE_MASK */
32000, /* CS48L32_48K_RATE_MASK */
};
static const struct snd_pcm_hw_constraint_list cs48l32_constraint = {
.count = ARRAY_SIZE(cs48l32_sr_vals),
.list = cs48l32_sr_vals,
};
static int cs48l32_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
{
struct snd_soc_component *component = dai->component;
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct cs48l32_dai_priv *dai_priv = &cs48l32_codec->dai[dai->id - 1];
unsigned int base_rate;
if (!substream->runtime)
return 0;
switch (dai_priv->clk) {
case CS48L32_CLK_SYSCLK_1:
case CS48L32_CLK_SYSCLK_2:
case CS48L32_CLK_SYSCLK_3:
case CS48L32_CLK_SYSCLK_4:
base_rate = cs48l32_codec->sysclk;
break ;
default :
return 0;
}
if (base_rate == 0)
dai_priv->constraint.mask = CS48L32_RATE_MASK;
else if (base_rate % 4000)
dai_priv->constraint.mask = CS48L32_44K1_RATE_MASK;
else
dai_priv->constraint.mask = CS48L32_48K_RATE_MASK;
return snd_pcm_hw_constraint_list(substream->runtime, 0,
SNDRV_PCM_HW_PARAM_RATE,
&dai_priv->constraint);
}
static int cs48l32_hw_params_rate(struct snd_pcm_substream *substream,
struct snd_pcm_hw_params *params,
struct snd_soc_dai *dai)
{
struct snd_soc_component *component = dai->component;
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct cs48l32_dai_priv *dai_priv = &cs48l32_codec->dai[dai->id - 1];
unsigned int sr_val, sr_reg, rate;
rate = params_rate(params);
for (sr_val = 0; sr_val < ARRAY_SIZE(cs48l32_sr_vals); sr_val++)
if (cs48l32_sr_vals[sr_val] == rate)
break ;
if (sr_val == ARRAY_SIZE(cs48l32_sr_vals)) {
cs48l32_asp_err(dai, "Unsupported sample rate %dHz\n" , rate);
return -EINVAL;
}
switch (dai_priv->clk) {
case CS48L32_CLK_SYSCLK_1:
sr_reg = CS48L32_SAMPLE_RATE1;
break ;
case CS48L32_CLK_SYSCLK_2:
sr_reg = CS48L32_SAMPLE_RATE2;
break ;
case CS48L32_CLK_SYSCLK_3:
sr_reg = CS48L32_SAMPLE_RATE3;
break ;
case CS48L32_CLK_SYSCLK_4:
sr_reg = CS48L32_SAMPLE_RATE4;
break ;
default :
return -EINVAL;
}
snd_soc_component_update_bits(component, sr_reg, CS48L32_SAMPLE_RATE_1_MASK, sr_val);
return 0;
}
static bool cs48l32_asp_cfg_changed(struct snd_soc_component *component,
unsigned int base, unsigned int sclk,
unsigned int slotws, unsigned int dataw)
{
unsigned int val;
val = snd_soc_component_read(component, base + CS48L32_ASP_CONTROL1);
if (sclk != (val & CS48L32_ASP_BCLK_FREQ_MASK))
return true ;
val = snd_soc_component_read(component, base + CS48L32_ASP_CONTROL2);
if (slotws != (val & (CS48L32_ASP_RX_WIDTH_MASK | CS48L32_ASP_TX_WIDTH_MASK)))
return true ;
val = snd_soc_component_read(component, base + CS48L32_ASP_DATA_CONTROL1);
if (dataw != (val & (CS48L32_ASP_TX_WL_MASK)))
return true ;
val = snd_soc_component_read(component, base + CS48L32_ASP_DATA_CONTROL5);
if (dataw != (val & (CS48L32_ASP_RX_WL_MASK)))
return true ;
return false ;
}
static int cs48l32_hw_params(struct snd_pcm_substream *substream,
struct snd_pcm_hw_params *params,
struct snd_soc_dai *dai)
{
struct snd_soc_component *component = dai->component;
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct regmap *regmap = cs48l32_codec->core.regmap;
int base = dai->driver->base;
int dai_id = dai->id - 1;
unsigned int rate = params_rate(params);
unsigned int dataw = snd_pcm_format_width(params_format(params));
unsigned int asp_state = 0;
int sclk, sclk_target;
unsigned int slotw, n_slots, n_slots_multiple, val;
int i, ret;
cs48l32_asp_dbg(dai, "hwparams in: ch:%u dataw:%u rate:%u\n" ,
params_channels(params), dataw, rate);
/*
* The following calculations hold only under the assumption that
* symmetric_[rates|channels|samplebits] are set to 1
*/
if (cs48l32_codec->tdm_slots[dai_id]) {
n_slots = cs48l32_codec->tdm_slots[dai_id];
slotw = cs48l32_codec->tdm_width[dai_id];
} else {
n_slots = params_channels(params);
slotw = dataw;
}
val = snd_soc_component_read(component, base + CS48L32_ASP_CONTROL2);
val = (val & CS48L32_ASP_FMT_MASK) >> CS48L32_ASP_FMT_SHIFT;
if (val == CS48L32_ASP_FMT_I2S_MODE)
n_slots_multiple = 2;
else
n_slots_multiple = 1;
sclk_target = snd_soc_tdm_params_to_bclk(params, slotw, n_slots, n_slots_multiple);
if (sclk_target < 0) {
cs48l32_asp_err(dai, "Invalid parameters\n" );
return sclk_target;
}
for (i = 0; i < ARRAY_SIZE(cs48l32_sclk_rates); i++) {
if ((cs48l32_sclk_rates[i].freq >= sclk_target) &&
(cs48l32_sclk_rates[i].freq % rate == 0)) {
sclk = cs48l32_sclk_rates[i].id;
break ;
}
}
if (i == ARRAY_SIZE(cs48l32_sclk_rates)) {
cs48l32_asp_err(dai, "Unsupported sample rate %dHz\n" , rate);
return -EINVAL;
}
cs48l32_asp_dbg(dai, "hwparams out: n_slots:%u dataw:%u slotw:%u bclk:%u bclkid:%u\n" ,
n_slots, dataw, slotw, sclk_target, sclk);
slotw = (slotw << CS48L32_ASP_TX_WIDTH_SHIFT) |
(slotw << CS48L32_ASP_RX_WIDTH_SHIFT);
if (!cs48l32_asp_cfg_changed(component, base, sclk, slotw, dataw))
return cs48l32_hw_params_rate(substream, params, dai);
/* ASP must be disabled while changing configuration */
asp_state = snd_soc_component_read(component, base + CS48L32_ASP_ENABLES1);
regmap_clear_bits(regmap, base + CS48L32_ASP_ENABLES1, 0xff00ff);
ret = cs48l32_hw_params_rate(substream, params, dai);
if (ret != 0)
goto restore_asp;
regmap_update_bits_async(regmap,
base + CS48L32_ASP_CONTROL1,
CS48L32_ASP_BCLK_FREQ_MASK,
sclk);
regmap_update_bits_async(regmap,
base + CS48L32_ASP_CONTROL2,
CS48L32_ASP_RX_WIDTH_MASK | CS48L32_ASP_TX_WIDTH_MASK,
slotw);
regmap_update_bits_async(regmap,
base + CS48L32_ASP_DATA_CONTROL1,
CS48L32_ASP_TX_WL_MASK,
dataw);
regmap_update_bits(regmap,
base + CS48L32_ASP_DATA_CONTROL5,
CS48L32_ASP_RX_WL_MASK,
dataw);
restore_asp:
/* Restore ASP TX/RX enable state */
regmap_update_bits(regmap,
base + CS48L32_ASP_ENABLES1,
0xff00ff,
asp_state);
return ret;
}
static const char *cs48l32_dai_clk_str(int clk_id)
{
switch (clk_id) {
case CS48L32_CLK_SYSCLK_1:
case CS48L32_CLK_SYSCLK_2:
case CS48L32_CLK_SYSCLK_3:
case CS48L32_CLK_SYSCLK_4:
return "SYSCLK" ;
default :
return "Unknown clock" ;
}
}
static int cs48l32_dai_set_sysclk(struct snd_soc_dai *dai,
int clk_id, unsigned int freq, int dir)
{
struct snd_soc_component *component = dai->component;
struct cs48l32_codec *cs48l32_codec = snd_soc_component_get_drvdata(component);
struct cs48l32_dai_priv *dai_priv = &cs48l32_codec->dai[dai->id - 1];
unsigned int base = dai->driver->base;
unsigned int current_asp_rate, target_asp_rate;
bool change_rate_domain = false ;
int ret;
if (clk_id == dai_priv->clk)
return 0;
if (snd_soc_dai_active(dai)) {
cs48l32_asp_err(dai, "Can't change clock on active DAI\n" );
return -EBUSY;
}
switch (clk_id) {
case CS48L32_CLK_SYSCLK_1:
target_asp_rate = 0U << CS48L32_ASP_RATE_SHIFT;
break ;
case CS48L32_CLK_SYSCLK_2:
target_asp_rate = 1U << CS48L32_ASP_RATE_SHIFT;
break ;
case CS48L32_CLK_SYSCLK_3:
target_asp_rate = 2U << CS48L32_ASP_RATE_SHIFT;
break ;
case CS48L32_CLK_SYSCLK_4:
target_asp_rate = 3U << CS48L32_ASP_RATE_SHIFT;
break ;
default :
return -EINVAL;
}
dai_priv->clk = clk_id;
cs48l32_asp_dbg(dai, "Setting to %s\n" , cs48l32_dai_clk_str(clk_id));
if (base) {
ret = regmap_read(cs48l32_codec->core.regmap,
base + CS48L32_ASP_CONTROL1,
¤t_asp_rate);
if (ret != 0) {
cs48l32_asp_err(dai, "Failed to check rate: %d\n" , ret);
return ret;
}
if ((current_asp_rate & CS48L32_ASP_RATE_MASK) !=
(target_asp_rate & CS48L32_ASP_RATE_MASK)) {
change_rate_domain = true ;
mutex_lock(&cs48l32_codec->rate_lock);
/* Guard the rate change with SYSCLK cycles */
cs48l32_spin_sysclk(cs48l32_codec);
}
snd_soc_component_update_bits(component, base + CS48L32_ASP_CONTROL1,
CS48L32_ASP_RATE_MASK, target_asp_rate);
if (change_rate_domain) {
cs48l32_spin_sysclk(cs48l32_codec);
mutex_unlock(&cs48l32_codec->rate_lock);
}
}
return 0;
}
static void cs48l32_set_channels_to_mask(struct snd_soc_dai *dai,
unsigned int base,
--> --------------------
--> maximum size reached
--> --------------------
Messung V0.5 C=95 H=95 G=94
¤ Dauer der Verarbeitung: 0.30 Sekunden
(vorverarbeitet)
¤
*© Formatika GbR, Deutschland