|  | // SPDX-License-Identifier: GPL-2.0-only | 
|  | /* | 
|  | * Copyright (C) 2021 Stephan Gerhold | 
|  | * | 
|  | * Register definitions/sequences taken from various tfa98xx kernel drivers: | 
|  | * Copyright (C) 2014-2020 NXP Semiconductors, All Rights Reserved. | 
|  | * Copyright (C) 2013 Sony Mobile Communications Inc. | 
|  | */ | 
|  |  | 
|  | #include <linux/gpio/consumer.h> | 
|  | #include <linux/i2c.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/regmap.h> | 
|  | #include <linux/regulator/consumer.h> | 
|  | #include <sound/soc.h> | 
|  |  | 
|  | #define TFA989X_STATUSREG		0x00 | 
|  | #define TFA989X_BATTERYVOLTAGE		0x01 | 
|  | #define TFA989X_TEMPERATURE		0x02 | 
|  | #define TFA989X_REVISIONNUMBER		0x03 | 
|  | #define TFA989X_REVISIONNUMBER_REV_MSK	GENMASK(7, 0)	/* device revision */ | 
|  | #define TFA989X_I2SREG			0x04 | 
|  | #define TFA989X_I2SREG_RCV		2	/* receiver mode */ | 
|  | #define TFA989X_I2SREG_CHSA		6	/* amplifier input select */ | 
|  | #define TFA989X_I2SREG_CHSA_MSK		GENMASK(7, 6) | 
|  | #define TFA989X_I2SREG_I2SSR		12	/* sample rate */ | 
|  | #define TFA989X_I2SREG_I2SSR_MSK	GENMASK(15, 12) | 
|  | #define TFA989X_BAT_PROT		0x05 | 
|  | #define TFA989X_AUDIO_CTR		0x06 | 
|  | #define TFA989X_DCDCBOOST		0x07 | 
|  | #define TFA989X_SPKR_CALIBRATION	0x08 | 
|  | #define TFA989X_SYS_CTRL		0x09 | 
|  | #define TFA989X_SYS_CTRL_PWDN		0	/* power down */ | 
|  | #define TFA989X_SYS_CTRL_I2CR		1	/* I2C reset */ | 
|  | #define TFA989X_SYS_CTRL_CFE		2	/* enable CoolFlux DSP */ | 
|  | #define TFA989X_SYS_CTRL_AMPE		3	/* enable amplifier */ | 
|  | #define TFA989X_SYS_CTRL_DCA		4	/* enable boost */ | 
|  | #define TFA989X_SYS_CTRL_SBSL		5	/* DSP configured */ | 
|  | #define TFA989X_SYS_CTRL_AMPC		6	/* amplifier enabled by DSP */ | 
|  | #define TFA989X_I2S_SEL_REG		0x0a | 
|  | #define TFA989X_I2S_SEL_REG_SPKR_MSK	GENMASK(10, 9)	/* speaker impedance */ | 
|  | #define TFA989X_I2S_SEL_REG_DCFG_MSK	GENMASK(14, 11)	/* DCDC compensation */ | 
|  | #define TFA989X_HIDE_UNHIDE_KEY	0x40 | 
|  | #define TFA989X_PWM_CONTROL		0x41 | 
|  | #define TFA989X_CURRENTSENSE1		0x46 | 
|  | #define TFA989X_CURRENTSENSE2		0x47 | 
|  | #define TFA989X_CURRENTSENSE3		0x48 | 
|  | #define TFA989X_CURRENTSENSE4		0x49 | 
|  |  | 
|  | #define TFA9890_REVISION		0x80 | 
|  | #define TFA9895_REVISION		0x12 | 
|  | #define TFA9897_REVISION		0x97 | 
|  |  | 
|  | struct tfa989x_rev { | 
|  | unsigned int rev; | 
|  | int (*init)(struct regmap *regmap); | 
|  | }; | 
|  |  | 
|  | struct tfa989x { | 
|  | const struct tfa989x_rev *rev; | 
|  | struct regulator *vddd_supply; | 
|  | struct gpio_desc *rcv_gpiod; | 
|  | }; | 
|  |  | 
|  | static bool tfa989x_writeable_reg(struct device *dev, unsigned int reg) | 
|  | { | 
|  | return reg > TFA989X_REVISIONNUMBER; | 
|  | } | 
|  |  | 
|  | static bool tfa989x_volatile_reg(struct device *dev, unsigned int reg) | 
|  | { | 
|  | return reg < TFA989X_REVISIONNUMBER; | 
|  | } | 
|  |  | 
|  | static const struct regmap_config tfa989x_regmap = { | 
|  | .reg_bits = 8, | 
|  | .val_bits = 16, | 
|  |  | 
|  | .writeable_reg	= tfa989x_writeable_reg, | 
|  | .volatile_reg	= tfa989x_volatile_reg, | 
|  | .cache_type	= REGCACHE_RBTREE, | 
|  | }; | 
|  |  | 
|  | static const char * const chsa_text[] = { "Left", "Right", /* "DSP" */ }; | 
|  | static SOC_ENUM_SINGLE_DECL(chsa_enum, TFA989X_I2SREG, TFA989X_I2SREG_CHSA, chsa_text); | 
|  | static const struct snd_kcontrol_new chsa_mux = SOC_DAPM_ENUM("Amp Input", chsa_enum); | 
|  |  | 
|  | static const struct snd_soc_dapm_widget tfa989x_dapm_widgets[] = { | 
|  | SND_SOC_DAPM_OUTPUT("OUT"), | 
|  | SND_SOC_DAPM_SUPPLY("POWER", TFA989X_SYS_CTRL, TFA989X_SYS_CTRL_PWDN, 1, NULL, 0), | 
|  | SND_SOC_DAPM_OUT_DRV("AMPE", TFA989X_SYS_CTRL, TFA989X_SYS_CTRL_AMPE, 0, NULL, 0), | 
|  |  | 
|  | SND_SOC_DAPM_MUX("Amp Input", SND_SOC_NOPM, 0, 0, &chsa_mux), | 
|  | SND_SOC_DAPM_AIF_IN("AIFINL", "HiFi Playback", 0, SND_SOC_NOPM, 0, 0), | 
|  | SND_SOC_DAPM_AIF_IN("AIFINR", "HiFi Playback", 1, SND_SOC_NOPM, 0, 0), | 
|  | }; | 
|  |  | 
|  | static const struct snd_soc_dapm_route tfa989x_dapm_routes[] = { | 
|  | {"OUT", NULL, "AMPE"}, | 
|  | {"AMPE", NULL, "POWER"}, | 
|  | {"AMPE", NULL, "Amp Input"}, | 
|  | {"Amp Input", "Left", "AIFINL"}, | 
|  | {"Amp Input", "Right", "AIFINR"}, | 
|  | }; | 
|  |  | 
|  | static int tfa989x_put_mode(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) | 
|  | { | 
|  | struct snd_soc_component *component = snd_soc_kcontrol_component(kcontrol); | 
|  | struct tfa989x *tfa989x = snd_soc_component_get_drvdata(component); | 
|  |  | 
|  | gpiod_set_value_cansleep(tfa989x->rcv_gpiod, ucontrol->value.enumerated.item[0]); | 
|  |  | 
|  | return snd_soc_put_enum_double(kcontrol, ucontrol); | 
|  | } | 
|  |  | 
|  | static const char * const mode_text[] = { "Speaker", "Receiver" }; | 
|  | static SOC_ENUM_SINGLE_DECL(mode_enum, TFA989X_I2SREG, TFA989X_I2SREG_RCV, mode_text); | 
|  | static const struct snd_kcontrol_new tfa989x_mode_controls[] = { | 
|  | SOC_ENUM_EXT("Mode", mode_enum, snd_soc_get_enum_double, tfa989x_put_mode), | 
|  | }; | 
|  |  | 
|  | static int tfa989x_probe(struct snd_soc_component *component) | 
|  | { | 
|  | struct tfa989x *tfa989x = snd_soc_component_get_drvdata(component); | 
|  |  | 
|  | if (tfa989x->rev->rev == TFA9897_REVISION) | 
|  | return snd_soc_add_component_controls(component, tfa989x_mode_controls, | 
|  | ARRAY_SIZE(tfa989x_mode_controls)); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct snd_soc_component_driver tfa989x_component = { | 
|  | .probe			= tfa989x_probe, | 
|  | .dapm_widgets		= tfa989x_dapm_widgets, | 
|  | .num_dapm_widgets	= ARRAY_SIZE(tfa989x_dapm_widgets), | 
|  | .dapm_routes		= tfa989x_dapm_routes, | 
|  | .num_dapm_routes	= ARRAY_SIZE(tfa989x_dapm_routes), | 
|  | .use_pmdown_time	= 1, | 
|  | .endianness		= 1, | 
|  | }; | 
|  |  | 
|  | static const unsigned int tfa989x_rates[] = { | 
|  | 8000, 11025, 12000, 16000, 22050, 24000, 32000, 44100, 48000 | 
|  | }; | 
|  |  | 
|  | static int tfa989x_find_sample_rate(unsigned int rate) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(tfa989x_rates); ++i) | 
|  | if (tfa989x_rates[i] == rate) | 
|  | return i; | 
|  |  | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | static int tfa989x_hw_params(struct snd_pcm_substream *substream, | 
|  | struct snd_pcm_hw_params *params, | 
|  | struct snd_soc_dai *dai) | 
|  | { | 
|  | struct snd_soc_component *component = dai->component; | 
|  | int sr; | 
|  |  | 
|  | sr = tfa989x_find_sample_rate(params_rate(params)); | 
|  | if (sr < 0) | 
|  | return sr; | 
|  |  | 
|  | return snd_soc_component_update_bits(component, TFA989X_I2SREG, | 
|  | TFA989X_I2SREG_I2SSR_MSK, | 
|  | sr << TFA989X_I2SREG_I2SSR); | 
|  | } | 
|  |  | 
|  | static const struct snd_soc_dai_ops tfa989x_dai_ops = { | 
|  | .hw_params = tfa989x_hw_params, | 
|  | }; | 
|  |  | 
|  | static struct snd_soc_dai_driver tfa989x_dai = { | 
|  | .name = "tfa989x-hifi", | 
|  | .playback = { | 
|  | .stream_name	= "HiFi Playback", | 
|  | .formats	= SNDRV_PCM_FMTBIT_S16_LE, | 
|  | .rates		= SNDRV_PCM_RATE_8000_48000, | 
|  | .rate_min	= 8000, | 
|  | .rate_max	= 48000, | 
|  | .channels_min	= 1, | 
|  | .channels_max	= 2, | 
|  | }, | 
|  | .ops = &tfa989x_dai_ops, | 
|  | }; | 
|  |  | 
|  | static int tfa9890_init(struct regmap *regmap) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | /* temporarily allow access to hidden registers */ | 
|  | ret = regmap_write(regmap, TFA989X_HIDE_UNHIDE_KEY, 0x5a6b); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* update PLL registers */ | 
|  | ret = regmap_set_bits(regmap, 0x59, 0x3); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* hide registers again */ | 
|  | ret = regmap_write(regmap, TFA989X_HIDE_UNHIDE_KEY, 0x0000); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | return regmap_write(regmap, TFA989X_CURRENTSENSE2, 0x7BE1); | 
|  | } | 
|  |  | 
|  | static const struct tfa989x_rev tfa9890_rev = { | 
|  | .rev	= TFA9890_REVISION, | 
|  | .init	= tfa9890_init, | 
|  | }; | 
|  |  | 
|  | static const struct reg_sequence tfa9895_reg_init[] = { | 
|  | /* some other registers must be set for optimal amplifier behaviour */ | 
|  | { TFA989X_BAT_PROT, 0x13ab }, | 
|  | { TFA989X_AUDIO_CTR, 0x001f }, | 
|  |  | 
|  | /* peak voltage protection is always on, but may be written */ | 
|  | { TFA989X_SPKR_CALIBRATION, 0x3c4e }, | 
|  |  | 
|  | /* TFA989X_SYSCTRL_DCA = 0 */ | 
|  | { TFA989X_SYS_CTRL, 0x024d }, | 
|  | { TFA989X_PWM_CONTROL, 0x0308 }, | 
|  | { TFA989X_CURRENTSENSE4, 0x0e82 }, | 
|  | }; | 
|  |  | 
|  | static int tfa9895_init(struct regmap *regmap) | 
|  | { | 
|  | return regmap_multi_reg_write(regmap, tfa9895_reg_init, | 
|  | ARRAY_SIZE(tfa9895_reg_init)); | 
|  | } | 
|  |  | 
|  | static const struct tfa989x_rev tfa9895_rev = { | 
|  | .rev	= TFA9895_REVISION, | 
|  | .init	= tfa9895_init, | 
|  | }; | 
|  |  | 
|  | static int tfa9897_init(struct regmap *regmap) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | /* Reduce slewrate by clearing iddqtestbst to avoid booster damage */ | 
|  | ret = regmap_write(regmap, TFA989X_CURRENTSENSE3, 0x0300); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* Enable clipping */ | 
|  | ret = regmap_clear_bits(regmap, TFA989X_CURRENTSENSE4, 0x1); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* Set required TDM configuration */ | 
|  | return regmap_write(regmap, 0x14, 0x0); | 
|  | } | 
|  |  | 
|  | static const struct tfa989x_rev tfa9897_rev = { | 
|  | .rev	= TFA9897_REVISION, | 
|  | .init	= tfa9897_init, | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * Note: At the moment this driver bypasses the "CoolFlux DSP" built into the | 
|  | * TFA989X amplifiers. Unfortunately, there seems to be absolutely | 
|  | * no documentation for it - the public "short datasheets" do not provide | 
|  | * any information about the DSP or available registers. | 
|  | * | 
|  | * Usually the TFA989X amplifiers are configured through proprietary userspace | 
|  | * libraries. There are also some (rather complex) kernel drivers but even those | 
|  | * rely on obscure firmware blobs for configuration (so-called "containers"). | 
|  | * They seem to contain different "profiles" with tuned speaker settings, sample | 
|  | * rates and volume steps (which would be better exposed as separate ALSA mixers). | 
|  | * | 
|  | * Bypassing the DSP disables volume control (and perhaps some speaker | 
|  | * optimization?), but at least allows using the speaker without obscure | 
|  | * kernel drivers and firmware. | 
|  | * | 
|  | * Ideally NXP (or now Goodix) should release proper documentation for these | 
|  | * amplifiers so that support for the "CoolFlux DSP" can be implemented properly. | 
|  | */ | 
|  | static int tfa989x_dsp_bypass(struct regmap *regmap) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | /* Clear CHSA to bypass DSP and take input from I2S 1 left channel */ | 
|  | ret = regmap_clear_bits(regmap, TFA989X_I2SREG, TFA989X_I2SREG_CHSA_MSK); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* Set DCDC compensation to off and speaker impedance to 8 ohm */ | 
|  | ret = regmap_update_bits(regmap, TFA989X_I2S_SEL_REG, | 
|  | TFA989X_I2S_SEL_REG_DCFG_MSK | | 
|  | TFA989X_I2S_SEL_REG_SPKR_MSK, | 
|  | TFA989X_I2S_SEL_REG_SPKR_MSK); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* Set DCDC to follower mode and disable CoolFlux DSP */ | 
|  | return regmap_clear_bits(regmap, TFA989X_SYS_CTRL, | 
|  | BIT(TFA989X_SYS_CTRL_DCA) | | 
|  | BIT(TFA989X_SYS_CTRL_CFE) | | 
|  | BIT(TFA989X_SYS_CTRL_AMPC)); | 
|  | } | 
|  |  | 
|  | static void tfa989x_regulator_disable(void *data) | 
|  | { | 
|  | struct tfa989x *tfa989x = data; | 
|  |  | 
|  | regulator_disable(tfa989x->vddd_supply); | 
|  | } | 
|  |  | 
|  | static int tfa989x_i2c_probe(struct i2c_client *i2c) | 
|  | { | 
|  | struct device *dev = &i2c->dev; | 
|  | const struct tfa989x_rev *rev; | 
|  | struct tfa989x *tfa989x; | 
|  | struct regmap *regmap; | 
|  | unsigned int val; | 
|  | int ret; | 
|  |  | 
|  | rev = device_get_match_data(dev); | 
|  | if (!rev) { | 
|  | dev_err(dev, "unknown device revision\n"); | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | tfa989x = devm_kzalloc(dev, sizeof(*tfa989x), GFP_KERNEL); | 
|  | if (!tfa989x) | 
|  | return -ENOMEM; | 
|  |  | 
|  | tfa989x->rev = rev; | 
|  | i2c_set_clientdata(i2c, tfa989x); | 
|  |  | 
|  | tfa989x->vddd_supply = devm_regulator_get(dev, "vddd"); | 
|  | if (IS_ERR(tfa989x->vddd_supply)) | 
|  | return dev_err_probe(dev, PTR_ERR(tfa989x->vddd_supply), | 
|  | "Failed to get vddd regulator\n"); | 
|  |  | 
|  | if (tfa989x->rev->rev == TFA9897_REVISION) { | 
|  | tfa989x->rcv_gpiod = devm_gpiod_get_optional(dev, "rcv", GPIOD_OUT_LOW); | 
|  | if (IS_ERR(tfa989x->rcv_gpiod)) | 
|  | return PTR_ERR(tfa989x->rcv_gpiod); | 
|  | } | 
|  |  | 
|  | regmap = devm_regmap_init_i2c(i2c, &tfa989x_regmap); | 
|  | if (IS_ERR(regmap)) | 
|  | return PTR_ERR(regmap); | 
|  |  | 
|  | ret = regulator_enable(tfa989x->vddd_supply); | 
|  | if (ret) { | 
|  | dev_err(dev, "Failed to enable vddd regulator: %d\n", ret); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | ret = devm_add_action_or_reset(dev, tfa989x_regulator_disable, tfa989x); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* Bypass regcache for reset and init sequence */ | 
|  | regcache_cache_bypass(regmap, true); | 
|  |  | 
|  | /* Dummy read to generate i2c clocks, required on some devices */ | 
|  | regmap_read(regmap, TFA989X_REVISIONNUMBER, &val); | 
|  |  | 
|  | ret = regmap_read(regmap, TFA989X_REVISIONNUMBER, &val); | 
|  | if (ret) { | 
|  | dev_err(dev, "failed to read revision number: %d\n", ret); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | val &= TFA989X_REVISIONNUMBER_REV_MSK; | 
|  | if (val != rev->rev) { | 
|  | dev_err(dev, "invalid revision number, expected %#x, got %#x\n", | 
|  | rev->rev, val); | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | ret = regmap_write(regmap, TFA989X_SYS_CTRL, BIT(TFA989X_SYS_CTRL_I2CR)); | 
|  | if (ret) { | 
|  | dev_err(dev, "failed to reset I2C registers: %d\n", ret); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | ret = rev->init(regmap); | 
|  | if (ret) { | 
|  | dev_err(dev, "failed to initialize registers: %d\n", ret); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | ret = tfa989x_dsp_bypass(regmap); | 
|  | if (ret) { | 
|  | dev_err(dev, "failed to enable DSP bypass: %d\n", ret); | 
|  | return ret; | 
|  | } | 
|  | regcache_cache_bypass(regmap, false); | 
|  |  | 
|  | return devm_snd_soc_register_component(dev, &tfa989x_component, | 
|  | &tfa989x_dai, 1); | 
|  | } | 
|  |  | 
|  | static const struct of_device_id tfa989x_of_match[] = { | 
|  | { .compatible = "nxp,tfa9890", .data = &tfa9890_rev }, | 
|  | { .compatible = "nxp,tfa9895", .data = &tfa9895_rev }, | 
|  | { .compatible = "nxp,tfa9897", .data = &tfa9897_rev }, | 
|  | { } | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(of, tfa989x_of_match); | 
|  |  | 
|  | static struct i2c_driver tfa989x_i2c_driver = { | 
|  | .driver = { | 
|  | .name = "tfa989x", | 
|  | .of_match_table = tfa989x_of_match, | 
|  | }, | 
|  | .probe = tfa989x_i2c_probe, | 
|  | }; | 
|  | module_i2c_driver(tfa989x_i2c_driver); | 
|  |  | 
|  | MODULE_DESCRIPTION("ASoC NXP/Goodix TFA989X (TFA1) driver"); | 
|  | MODULE_AUTHOR("Stephan Gerhold <stephan@gerhold.net>"); | 
|  | MODULE_LICENSE("GPL"); |