blob: 6f89a422f3c62a1f4ca33126d1b48090ec4da03c [file]
// SPDX-License-Identifier: GPL-2.0
/*
* ALSA SoC Texas Instruments TAS67524 Quad-Channel Audio Amplifier
*
* Copyright (C) 2026 Texas Instruments Incorporated - https://www.ti.com/
* Author: Sen Wang <sen@ti.com>
*/
#include <linux/bitfield.h>
#include <linux/module.h>
#include <linux/mutex.h>
#include <linux/i2c.h>
#include <linux/regmap.h>
#include <linux/gpio/consumer.h>
#include <linux/regulator/consumer.h>
#include <linux/delay.h>
#include <linux/property.h>
#include <linux/interrupt.h>
#include <linux/workqueue.h>
#include <linux/pm_runtime.h>
#include <linux/iopoll.h>
#include <sound/soc.h>
#include <sound/tlv.h>
#include <sound/pcm_params.h>
#include "tas675x.h"
#define TAS675X_FAULT_CHECK_INTERVAL_MS 200
enum tas675x_type {
TAS67524,
};
struct tas675x_reg_param {
u8 page;
u8 reg;
u32 val;
};
struct tas675x_priv {
struct device *dev;
struct regmap *regmap;
enum tas675x_type dev_type;
/* Custom regmap lock; protects writes across books */
struct mutex io_lock;
struct gpio_desc *pd_gpio;
struct gpio_desc *stby_gpio;
struct regulator_bulk_data supplies[2];
struct regulator *vbat;
bool fast_boot;
int audio_slot;
int llp_slot;
int vpredict_slot;
int isense_slot;
int bclk_offset;
int slot_width;
unsigned int tx_mask;
int gpio1_func;
int gpio2_func;
unsigned long active_playback_dais;
unsigned long active_capture_dais;
unsigned int rate;
unsigned int saved_rtldg_en;
#define TAS675X_DSP_PARAM_NUM 2
struct tas675x_reg_param dsp_params[TAS675X_DSP_PARAM_NUM];
/* Fault monitor, disabled when Fault IRQ is used */
struct delayed_work fault_check_work;
#define TAS675X_FAULT_REGS_NUM 9
unsigned int last_status[TAS675X_FAULT_REGS_NUM];
};
static const char * const tas675x_supply_names[] = {
"dvdd", /* Digital power supply */
"pvdd", /* Output powerstage supply */
};
/* Page 1 setup initialization defaults */
static const struct reg_sequence tas675x_page1_init[] = {
REG_SEQ0(TAS675X_PAGE_REG(1, 0xC8), 0x20), /* Charge pump clock */
REG_SEQ0(TAS675X_PAGE_REG(1, 0x2F), 0x90), /* VBAT idle */
REG_SEQ0(TAS675X_PAGE_REG(1, 0x29), 0x40), /* OC/CBC threshold */
REG_SEQ0(TAS675X_PAGE_REG(1, 0x2E), 0x0C), /* OC/CBC config */
REG_SEQ0(TAS675X_PAGE_REG(1, 0xC5), 0x02), /* OC/CBC config */
REG_SEQ0(TAS675X_PAGE_REG(1, 0xC6), 0x10), /* OC/CBC config */
REG_SEQ0(TAS675X_PAGE_REG(1, 0x1F), 0x20), /* OC/CBC config */
REG_SEQ0(TAS675X_PAGE_REG(1, 0x16), 0x01), /* OC/CBC config */
REG_SEQ0(TAS675X_PAGE_REG(1, 0x1E), 0x04), /* OC/CBC config */
REG_SEQ0(TAS675X_PAGE_REG(1, 0xC1), 0x00), /* CH1 DC fault */
REG_SEQ0(TAS675X_PAGE_REG(1, 0xC2), 0x04), /* CH2 DC fault */
REG_SEQ0(TAS675X_PAGE_REG(1, 0xC3), 0x00), /* CH3 DC fault */
REG_SEQ0(TAS675X_PAGE_REG(1, 0xC4), 0x00), /* CH4 DC fault */
};
static inline const char *tas675x_state_name(unsigned int state)
{
switch (state & 0x0F) {
case TAS675X_STATE_DEEPSLEEP: return "DEEPSLEEP";
case TAS675X_STATE_LOAD_DIAG: return "LOAD_DIAG";
case TAS675X_STATE_SLEEP: return "SLEEP";
case TAS675X_STATE_HIZ: return "HIZ";
case TAS675X_STATE_PLAY: return "PLAY";
case TAS675X_STATE_FAULT: return "FAULT";
case TAS675X_STATE_AUTOREC: return "AUTOREC";
default: return "UNKNOWN";
}
}
static inline int tas675x_set_state_all(struct tas675x_priv *tas, u8 state)
{
const struct reg_sequence seq[] = {
REG_SEQ0(TAS675X_STATE_CTRL_CH1_CH2_REG, state),
REG_SEQ0(TAS675X_STATE_CTRL_CH3_CH4_REG, state),
};
return regmap_multi_reg_write(tas->regmap, seq, ARRAY_SIZE(seq));
}
static inline int tas675x_select_book(struct regmap *regmap, u8 book)
{
int ret;
/* Reset page to 0 before switching books */
ret = regmap_write(regmap, TAS675X_PAGE_CTRL_REG, 0x00);
if (!ret)
ret = regmap_write(regmap, TAS675X_BOOK_CTRL_REG, book);
return ret;
}
/* Raw I2C version of tas675x_select_book, must be called with io_lock held */
static inline int __tas675x_select_book(struct tas675x_priv *tas, u8 book)
{
struct i2c_client *client = to_i2c_client(tas->dev);
int ret;
/* Reset page to 0 before switching books */
ret = i2c_smbus_write_byte_data(client, TAS675X_PAGE_CTRL_REG, 0x00);
if (ret)
return ret;
return i2c_smbus_write_byte_data(client, TAS675X_BOOK_CTRL_REG, book);
}
static int tas675x_dsp_mem_write(struct tas675x_priv *tas, u8 page, u8 reg, u32 val)
{
struct i2c_client *client = to_i2c_client(tas->dev);
u8 buf[4];
int ret;
/* DSP registers are 32 bit big-endian */
buf[0] = (val >> 24) & 0xFF;
buf[1] = (val >> 16) & 0xFF;
buf[2] = (val >> 8) & 0xFF;
buf[3] = val & 0xFF;
/*
* DSP regs in a different book, therefore block
* regmap access before completion.
*/
mutex_lock(&tas->io_lock);
ret = __tas675x_select_book(tas, TAS675X_BOOK_DSP);
if (ret)
goto out;
ret = i2c_smbus_write_byte_data(client, TAS675X_PAGE_CTRL_REG, page);
if (ret)
goto out;
ret = i2c_smbus_write_i2c_block_data(client, reg, sizeof(buf), buf);
out:
__tas675x_select_book(tas, TAS675X_BOOK_DEFAULT);
mutex_unlock(&tas->io_lock);
return ret;
}
static int tas675x_dsp_mem_read(struct tas675x_priv *tas, u8 page, u8 reg, u32 *val)
{
struct i2c_client *client = to_i2c_client(tas->dev);
u8 buf[4];
int ret;
/*
* DSP regs in a different book, therefore block
* regmap access before completion.
*/
mutex_lock(&tas->io_lock);
ret = __tas675x_select_book(tas, TAS675X_BOOK_DSP);
if (ret)
goto out;
ret = i2c_smbus_write_byte_data(client, TAS675X_PAGE_CTRL_REG, page);
if (ret)
goto out;
ret = i2c_smbus_read_i2c_block_data(client, reg, sizeof(buf), buf);
if (ret == sizeof(buf)) {
*val = (buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3];
ret = 0;
} else if (ret >= 0) {
ret = -EIO;
}
out:
__tas675x_select_book(tas, TAS675X_BOOK_DEFAULT);
mutex_unlock(&tas->io_lock);
return ret;
}
static const struct {
const char *name;
int val;
} tas675x_gpio_func_map[] = {
/* Output functions */
{ "low", TAS675X_GPIO_SEL_LOW },
{ "auto-mute", TAS675X_GPIO_SEL_AUTO_MUTE_ALL },
{ "auto-mute-ch4", TAS675X_GPIO_SEL_AUTO_MUTE_CH4 },
{ "auto-mute-ch3", TAS675X_GPIO_SEL_AUTO_MUTE_CH3 },
{ "auto-mute-ch2", TAS675X_GPIO_SEL_AUTO_MUTE_CH2 },
{ "auto-mute-ch1", TAS675X_GPIO_SEL_AUTO_MUTE_CH1 },
{ "sdout2", TAS675X_GPIO_SEL_SDOUT2 },
{ "sdout1", TAS675X_GPIO_SEL_SDOUT1 },
{ "warn", TAS675X_GPIO_SEL_WARN },
{ "fault", TAS675X_GPIO_SEL_FAULT },
{ "clock-sync", TAS675X_GPIO_SEL_CLOCK_SYNC },
{ "invalid-clock", TAS675X_GPIO_SEL_INVALID_CLK },
{ "high", TAS675X_GPIO_SEL_HIGH },
/* Input functions */
{ "mute", TAS675X_GPIO_IN_MUTE },
{ "phase-sync", TAS675X_GPIO_IN_PHASE_SYNC },
{ "sdin2", TAS675X_GPIO_IN_SDIN2 },
{ "deep-sleep", TAS675X_GPIO_IN_DEEP_SLEEP },
{ "hiz", TAS675X_GPIO_IN_HIZ },
{ "play", TAS675X_GPIO_IN_PLAY },
{ "sleep", TAS675X_GPIO_IN_SLEEP },
};
static int tas675x_gpio_func_parse(struct device *dev, const char *propname)
{
const char *str;
int i, ret;
ret = device_property_read_string(dev, propname, &str);
if (ret)
return -1;
for (i = 0; i < ARRAY_SIZE(tas675x_gpio_func_map); i++) {
if (!strcmp(str, tas675x_gpio_func_map[i].name))
return tas675x_gpio_func_map[i].val;
}
dev_warn(dev, "Invalid %s value '%s'\n", propname, str);
return -1;
}
static const struct {
unsigned int reg;
unsigned int mask;
} tas675x_gpio_input_table[TAS675X_GPIO_IN_NUM] = {
[TAS675X_GPIO_IN_ID_MUTE] = {
TAS675X_GPIO_INPUT_MUTE_REG, TAS675X_GPIO_IN_MUTE_MASK },
[TAS675X_GPIO_IN_ID_PHASE_SYNC] = {
TAS675X_GPIO_INPUT_SYNC_REG, TAS675X_GPIO_IN_SYNC_MASK },
[TAS675X_GPIO_IN_ID_SDIN2] = {
TAS675X_GPIO_INPUT_SDIN2_REG, TAS675X_GPIO_IN_SDIN2_MASK },
[TAS675X_GPIO_IN_ID_DEEP_SLEEP] = {
TAS675X_GPIO_INPUT_SLEEP_HIZ_REG, TAS675X_GPIO_IN_DEEP_SLEEP_MASK },
[TAS675X_GPIO_IN_ID_HIZ] = {
TAS675X_GPIO_INPUT_SLEEP_HIZ_REG, TAS675X_GPIO_IN_HIZ_MASK },
[TAS675X_GPIO_IN_ID_PLAY] = {
TAS675X_GPIO_INPUT_PLAY_SLEEP_REG, TAS675X_GPIO_IN_PLAY_MASK },
[TAS675X_GPIO_IN_ID_SLEEP] = {
TAS675X_GPIO_INPUT_PLAY_SLEEP_REG, TAS675X_GPIO_IN_SLEEP_MASK },
};
static void tas675x_config_gpio_pin(struct regmap *regmap, int func_id,
unsigned int out_sel_reg,
unsigned int pin_idx,
unsigned int *gpio_ctrl)
{
int id;
if (func_id < 0)
return;
if (func_id & TAS675X_GPIO_FUNC_INPUT) {
/* 3-bit mux: 0 = disabled, 0b1 = GPIO1, 0b10 = GPIO2 */
id = func_id & ~TAS675X_GPIO_FUNC_INPUT;
regmap_update_bits(regmap,
tas675x_gpio_input_table[id].reg,
tas675x_gpio_input_table[id].mask,
(pin_idx + 1) << __ffs(tas675x_gpio_input_table[id].mask));
} else {
/* Output GPIO, update selection register and enable bit */
regmap_write(regmap, out_sel_reg, func_id);
*gpio_ctrl |= pin_idx ? TAS675X_GPIO2_OUTPUT_EN : TAS675X_GPIO1_OUTPUT_EN;
}
}
static int tas675x_rtldg_thresh_info(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_info *uinfo)
{
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
uinfo->count = 1;
uinfo->value.integer.min = 0;
/* threshold reg ranges up to 24bit */
uinfo->value.integer.max = 0x00FFFFFF;
return 0;
}
static int tas675x_set_rtldg_thresh(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *comp = snd_kcontrol_chip(kcontrol);
struct tas675x_priv *tas = snd_soc_component_get_drvdata(comp);
const struct tas675x_reg_param *t =
(const struct tas675x_reg_param *)kcontrol->private_value;
u32 val = ucontrol->value.integer.value[0];
int ret;
ret = tas675x_dsp_mem_write(tas, t->page, t->reg, val);
/* Cache the value */
if (!ret) {
int i;
for (i = 0; i < ARRAY_SIZE(tas->dsp_params); i++) {
if (tas->dsp_params[i].page == t->page &&
tas->dsp_params[i].reg == t->reg) {
tas->dsp_params[i].val = val;
break;
}
}
}
/* Return 1 to notify change, or propagate error */
return ret ? ret : 1;
}
static int tas675x_get_rtldg_thresh(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *comp = snd_kcontrol_chip(kcontrol);
struct tas675x_priv *tas = snd_soc_component_get_drvdata(comp);
const struct tas675x_reg_param *t =
(const struct tas675x_reg_param *)kcontrol->private_value;
u32 val = 0;
int ret;
ret = tas675x_dsp_mem_read(tas, t->page, t->reg, &val);
if (!ret)
ucontrol->value.integer.value[0] = val;
return ret;
}
static const struct tas675x_reg_param tas675x_dsp_defaults[] = {
[TAS675X_DSP_PARAM_ID_OL_THRESH] = {
TAS675X_DSP_PAGE_RTLDG, TAS675X_DSP_RTLDG_OL_THRESH_REG },
[TAS675X_DSP_PARAM_ID_SL_THRESH] = {
TAS675X_DSP_PAGE_RTLDG, TAS675X_DSP_RTLDG_SL_THRESH_REG },
};
static_assert(ARRAY_SIZE(tas675x_dsp_defaults) == TAS675X_DSP_PARAM_NUM);
static int tas675x_set_dcldg_trigger(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *comp = snd_kcontrol_chip(kcontrol);
struct tas675x_priv *tas = snd_soc_component_get_drvdata(comp);
unsigned int state, state34;
int ret;
if (!ucontrol->value.integer.value[0])
return 0;
if (snd_soc_component_active(comp))
return -EBUSY;
ret = pm_runtime_resume_and_get(tas->dev);
if (ret < 0)
return ret;
/*
* Abort automatic DC LDG retry loops (startup or init-after-fault)
* and clear faults before manual diagnostics.
*/
regmap_update_bits(tas->regmap, TAS675X_DC_LDG_CTRL_REG,
TAS675X_LDG_ABORT_BIT | TAS675X_LDG_BYPASS_BIT,
TAS675X_LDG_ABORT_BIT | TAS675X_LDG_BYPASS_BIT);
regmap_write(tas->regmap, TAS675X_RESET_REG, TAS675X_FAULT_CLEAR);
/* Wait for LOAD_DIAG to exit */
ret = regmap_read_poll_timeout(tas->regmap, TAS675X_STATE_REPORT_CH1_CH2_REG,
state, (state & 0x0F) != TAS675X_STATE_LOAD_DIAG &&
(state >> 4) != TAS675X_STATE_LOAD_DIAG,
TAS675X_POLL_INTERVAL_US,
TAS675X_STATE_TRANSITION_TIMEOUT_US);
ret |= regmap_read_poll_timeout(tas->regmap, TAS675X_STATE_REPORT_CH3_CH4_REG,
state34, (state34 & 0x0F) != TAS675X_STATE_LOAD_DIAG &&
(state34 >> 4) != TAS675X_STATE_LOAD_DIAG,
TAS675X_POLL_INTERVAL_US,
TAS675X_STATE_TRANSITION_TIMEOUT_US);
if (ret) {
dev_err(tas->dev,
"DC LDG: abort timeout (CH1/2=0x%02x [%s/%s], CH3/4=0x%02x [%s/%s])\n",
state, tas675x_state_name(state), tas675x_state_name(state >> 4),
state34, tas675x_state_name(state34), tas675x_state_name(state34 >> 4));
goto out_restore_ldg_ctrl;
}
/* Transition to HIZ state */
ret = tas675x_set_state_all(tas, TAS675X_STATE_HIZ_BOTH);
if (ret)
goto out_restore_ldg_ctrl;
/* Set LOAD_DIAG state for manual DC LDG */
ret = tas675x_set_state_all(tas, TAS675X_STATE_LOAD_DIAG_BOTH);
if (ret)
goto out_restore_ldg_ctrl;
/* Wait for device to transition to LOAD_DIAG state */
ret = regmap_read_poll_timeout(tas->regmap, TAS675X_STATE_REPORT_CH1_CH2_REG,
state, state == TAS675X_STATE_LOAD_DIAG_BOTH,
TAS675X_POLL_INTERVAL_US,
TAS675X_STATE_TRANSITION_TIMEOUT_US);
ret |= regmap_read_poll_timeout(tas->regmap, TAS675X_STATE_REPORT_CH3_CH4_REG,
state34, state34 == TAS675X_STATE_LOAD_DIAG_BOTH,
TAS675X_POLL_INTERVAL_US,
TAS675X_STATE_TRANSITION_TIMEOUT_US);
if (ret) {
dev_err(tas->dev,
"DC LDG: LOAD_DIAG timeout (CH1/2=0x%02x [%s/%s], CH3/4=0x%02x [%s/%s])\n",
state, tas675x_state_name(state), tas675x_state_name(state >> 4),
state34, tas675x_state_name(state34), tas675x_state_name(state34 >> 4));
goto out_restore_hiz;
}
/* Clear ABORT and BYPASS bits to enable manual DC LDG */
ret = regmap_update_bits(tas->regmap, TAS675X_DC_LDG_CTRL_REG,
TAS675X_LDG_ABORT_BIT | TAS675X_LDG_BYPASS_BIT,
0);
if (ret)
goto out_restore_hiz;
dev_dbg(tas->dev, "DC LDG: Started\n");
/* Poll all channels for SLEEP state */
ret = regmap_read_poll_timeout(tas->regmap, TAS675X_STATE_REPORT_CH1_CH2_REG,
state, state == TAS675X_STATE_SLEEP_BOTH,
TAS675X_POLL_INTERVAL_US,
TAS675X_DC_LDG_TIMEOUT_US);
ret |= regmap_read_poll_timeout(tas->regmap, TAS675X_STATE_REPORT_CH3_CH4_REG,
state34, state34 == TAS675X_STATE_SLEEP_BOTH,
TAS675X_POLL_INTERVAL_US,
TAS675X_DC_LDG_TIMEOUT_US);
if (ret) {
dev_err(tas->dev,
"DC LDG: SLEEP timeout (CH1/2=0x%02x [%s/%s], CH3/4=0x%02x [%s/%s])\n",
state, tas675x_state_name(state), tas675x_state_name(state >> 4),
state34, tas675x_state_name(state34), tas675x_state_name(state34 >> 4));
goto out_restore_hiz;
}
dev_dbg(tas->dev, "DC LDG: Completed successfully (CH1/2=0x%02x, CH3/4=0x%02x)\n",
state, state34);
out_restore_hiz:
tas675x_set_state_all(tas, TAS675X_STATE_HIZ_BOTH);
out_restore_ldg_ctrl:
regmap_update_bits(tas->regmap, TAS675X_DC_LDG_CTRL_REG,
TAS675X_LDG_ABORT_BIT | TAS675X_LDG_BYPASS_BIT,
0);
pm_runtime_mark_last_busy(tas->dev);
pm_runtime_put_autosuspend(tas->dev);
return ret;
}
static int tas675x_set_acldg_trigger(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *comp = snd_kcontrol_chip(kcontrol);
struct tas675x_priv *tas = snd_soc_component_get_drvdata(comp);
unsigned int state, state34;
int ret;
if (!ucontrol->value.integer.value[0])
return 0;
if (snd_soc_component_active(comp))
return -EBUSY;
ret = pm_runtime_resume_and_get(tas->dev);
if (ret < 0)
return ret;
/* AC Load Diagnostics requires SLEEP state */
ret = tas675x_set_state_all(tas, TAS675X_STATE_SLEEP_BOTH);
if (ret) {
dev_err(tas->dev, "AC LDG: Failed to set SLEEP state: %d\n", ret);
goto out;
}
/* Start AC LDG on all 4 channels (0x0F) */
ret = regmap_write(tas->regmap, TAS675X_AC_LDG_CTRL_REG, 0x0F);
if (ret) {
dev_err(tas->dev, "AC LDG: Failed to start: %d\n", ret);
goto out;
}
dev_dbg(tas->dev, "AC LDG: Started\n");
/* Poll all channels for SLEEP state */
ret = regmap_read_poll_timeout(tas->regmap, TAS675X_STATE_REPORT_CH1_CH2_REG,
state, (state == TAS675X_STATE_SLEEP_BOTH),
TAS675X_POLL_INTERVAL_US,
TAS675X_AC_LDG_TIMEOUT_US);
if (ret) {
dev_err(tas->dev,
"AC LDG: CH1/CH2 timeout: %d (state=0x%02x [%s/%s])\n",
ret, state, tas675x_state_name(state),
tas675x_state_name(state >> 4));
regmap_write(tas->regmap, TAS675X_AC_LDG_CTRL_REG, 0x00);
goto out;
}
ret = regmap_read_poll_timeout(tas->regmap, TAS675X_STATE_REPORT_CH3_CH4_REG,
state34, (state34 == TAS675X_STATE_SLEEP_BOTH),
TAS675X_POLL_INTERVAL_US,
TAS675X_AC_LDG_TIMEOUT_US);
if (ret) {
dev_err(tas->dev,
"AC LDG: CH3/CH4 timeout: %d (state=0x%02x [%s/%s])\n",
ret, state34, tas675x_state_name(state34),
tas675x_state_name(state34 >> 4));
regmap_write(tas->regmap, TAS675X_AC_LDG_CTRL_REG, 0x00);
goto out;
}
dev_dbg(tas->dev, "AC LDG: Completed successfully (CH1/2=0x%02x, CH3/4=0x%02x)\n",
state, state34);
regmap_write(tas->regmap, TAS675X_AC_LDG_CTRL_REG, 0x00);
out:
pm_runtime_mark_last_busy(tas->dev);
pm_runtime_put_autosuspend(tas->dev);
return ret;
}
static int tas675x_rtldg_impedance_info(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_info *uinfo)
{
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
uinfo->count = 1;
uinfo->value.integer.min = 0;
uinfo->value.integer.max = 0xFFFF;
return 0;
}
static int tas675x_get_rtldg_impedance(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *comp = snd_kcontrol_chip(kcontrol);
struct tas675x_priv *tas = snd_soc_component_get_drvdata(comp);
unsigned int msb_reg = (unsigned int)kcontrol->private_value;
u8 buf[2];
int ret;
ret = regmap_bulk_read(tas->regmap, msb_reg, buf, 2);
if (ret)
return ret;
ucontrol->value.integer.value[0] = (buf[0] << 8) | buf[1];
return 0;
}
static int tas675x_dc_resistance_info(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_info *uinfo)
{
/* 10-bit: 2-bit MSB + 8-bit LSB, 0.1 ohm/code, 0-102.3 ohm */
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
uinfo->count = 1;
uinfo->value.integer.min = 0;
uinfo->value.integer.max = 1023;
return 0;
}
static int tas675x_get_dc_resistance(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct snd_soc_component *comp = snd_kcontrol_chip(kcontrol);
struct tas675x_priv *tas = snd_soc_component_get_drvdata(comp);
unsigned int lsb_reg = (unsigned int)kcontrol->private_value;
unsigned int msb, lsb, shift;
int ret;
ret = regmap_read(tas->regmap, TAS675X_DC_LDG_DCR_MSB_REG, &msb);
if (ret)
return ret;
ret = regmap_read(tas->regmap, lsb_reg, &lsb);
if (ret)
return ret;
/* 2-bit MSB: CH1=[7:6], CH2=[5:4], CH3=[3:2], CH4=[1:0] */
shift = 6 - (lsb_reg - TAS675X_CH1_DC_LDG_DCR_LSB_REG) * 2;
msb = (msb >> shift) & 0x3;
ucontrol->value.integer.value[0] = (msb << 8) | lsb;
return 0;
}
/* Counterparts with read-only access */
#define SOC_SINGLE_RO(xname, xreg, xshift, xmax) \
{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
.name = xname, \
.access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE, \
.info = snd_soc_info_volsw, \
.get = snd_soc_get_volsw, \
.private_value = SOC_SINGLE_VALUE(xreg, xshift, 0, xmax, 0, 0) }
#define SOC_DC_RESIST_RO(xname, xlsb_reg) \
{ .name = xname, \
.iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
.access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE, \
.info = tas675x_dc_resistance_info, \
.get = tas675x_get_dc_resistance, \
.private_value = (xlsb_reg) }
#define SOC_RTLDG_IMP_RO(xname, xreg) \
{ .name = xname, \
.iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
.access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE, \
.info = tas675x_rtldg_impedance_info, \
.get = tas675x_get_rtldg_impedance, \
.private_value = (xreg) }
#define SOC_DSP_THRESH_EXT(xname, xthresh) \
{ .name = xname, \
.iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
.info = tas675x_rtldg_thresh_info, \
.get = tas675x_get_rtldg_thresh, \
.put = tas675x_set_rtldg_thresh, \
.private_value = (unsigned long)&(xthresh) }
/*
* DAC digital volumes. From -103 to 0 dB in 0.5 dB steps, -103.5 dB means mute.
* DAC analog gain. From -15.5 to 0 dB in 0.5 dB steps, no mute.
*/
static const DECLARE_TLV_DB_SCALE(tas675x_dig_vol_tlv, -10350, 50, 1);
static const DECLARE_TLV_DB_SCALE(tas675x_ana_gain_tlv, -1550, 50, 0);
static const char * const tas675x_ss_texts[] = {
"Disabled", "Triangle", "Random", "Triangle and Random"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ss_enum, TAS675X_SS_CTRL_REG, 0, tas675x_ss_texts);
static const char * const tas675x_ss_tri_range_texts[] = {
"6.5%", "13.5%", "5%", "10%"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ss_tri_range_enum,
TAS675X_SS_RANGE_CTRL_REG, 0,
tas675x_ss_tri_range_texts);
static const char * const tas675x_ss_rdm_range_texts[] = {
"0.83%", "2.50%", "5.83%", "12.50%", "25.83%"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ss_rdm_range_enum,
TAS675X_SS_RANGE_CTRL_REG, 4,
tas675x_ss_rdm_range_texts);
static const char * const tas675x_ss_rdm_dwell_texts[] = {
"1/FSS to 2/FSS", "1/FSS to 4/FSS", "1/FSS to 8/FSS", "1/FSS to 15/FSS"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ss_rdm_dwell_enum,
TAS675X_SS_RANGE_CTRL_REG, 2,
tas675x_ss_rdm_dwell_texts);
static const char * const tas675x_oc_limit_texts[] = {
"Level 4", "Level 3", "Level 2", "Level 1"
};
static SOC_ENUM_SINGLE_DECL(tas675x_oc_limit_enum, TAS675X_CURRENT_LIMIT_CTRL_REG,
0, tas675x_oc_limit_texts);
static const char * const tas675x_otw_texts[] = {
"Disabled", ">95C", ">110C", ">125C", ">135C", ">145C", ">155C", ">165C"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ch1_otw_enum,
TAS675X_OTW_CTRL_CH1_CH2_REG, 4,
tas675x_otw_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ch2_otw_enum,
TAS675X_OTW_CTRL_CH1_CH2_REG, 0,
tas675x_otw_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ch3_otw_enum,
TAS675X_OTW_CTRL_CH3_CH4_REG, 4,
tas675x_otw_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ch4_otw_enum,
TAS675X_OTW_CTRL_CH3_CH4_REG, 0,
tas675x_otw_texts);
static const char * const tas675x_dc_ldg_sl_texts[] = {
"0.5 Ohm", "1 Ohm", "1.5 Ohm", "2 Ohm", "2.5 Ohm",
"3 Ohm", "3.5 Ohm", "4 Ohm", "4.5 Ohm", "5 Ohm"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ch1_dc_ldg_sl_enum,
TAS675X_DC_LDG_SL_CH1_CH2_CTRL_REG, 4,
tas675x_dc_ldg_sl_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ch2_dc_ldg_sl_enum,
TAS675X_DC_LDG_SL_CH1_CH2_CTRL_REG, 0,
tas675x_dc_ldg_sl_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ch3_dc_ldg_sl_enum,
TAS675X_DC_LDG_SL_CH3_CH4_CTRL_REG, 4,
tas675x_dc_ldg_sl_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ch4_dc_ldg_sl_enum,
TAS675X_DC_LDG_SL_CH3_CH4_CTRL_REG, 0,
tas675x_dc_ldg_sl_texts);
static const char * const tas675x_dc_slol_ramp_texts[] = {
"15 ms", "30 ms", "10 ms", "20 ms"
};
static SOC_ENUM_SINGLE_DECL(tas675x_dc_slol_ramp_enum,
TAS675X_DC_LDG_TIME_CTRL_REG, 6,
tas675x_dc_slol_ramp_texts);
static const char * const tas675x_dc_slol_settling_texts[] = {
"10 ms", "5 ms", "20 ms", "15 ms"
};
static SOC_ENUM_SINGLE_DECL(tas675x_dc_slol_settling_enum,
TAS675X_DC_LDG_TIME_CTRL_REG, 4,
tas675x_dc_slol_settling_texts);
static const char * const tas675x_dc_s2pg_ramp_texts[] = {
"5 ms", "2.5 ms", "10 ms", "15 ms"
};
static SOC_ENUM_SINGLE_DECL(tas675x_dc_s2pg_ramp_enum,
TAS675X_DC_LDG_TIME_CTRL_REG, 2,
tas675x_dc_s2pg_ramp_texts);
static const char * const tas675x_dc_s2pg_settling_texts[] = {
"10 ms", "5 ms", "20 ms", "30 ms"
};
static SOC_ENUM_SINGLE_DECL(tas675x_dc_s2pg_settling_enum,
TAS675X_DC_LDG_TIME_CTRL_REG, 0,
tas675x_dc_s2pg_settling_texts);
static const char * const tas675x_dsp_mode_texts[] = {
"Normal", "LLP", "FFLP"
};
static SOC_ENUM_SINGLE_DECL(tas675x_dsp_mode_enum,
TAS675X_LL_EN_REG, 0,
tas675x_dsp_mode_texts);
static const char * const tas675x_ana_ramp_texts[] = {
"15us", "60us", "200us", "400us"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ana_ramp_enum,
TAS675X_ANALOG_GAIN_RAMP_CTRL_REG, 2,
tas675x_ana_ramp_texts);
static const char * const tas675x_ramp_rate_texts[] = {
"4 FS", "16 FS", "32 FS", "Instant"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ramp_down_rate_enum,
TAS675X_DIG_VOL_RAMP_CTRL_REG, 6,
tas675x_ramp_rate_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ramp_up_rate_enum,
TAS675X_DIG_VOL_RAMP_CTRL_REG, 2,
tas675x_ramp_rate_texts);
static const char * const tas675x_ramp_step_texts[] = {
"4dB", "2dB", "1dB", "0.5dB"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ramp_down_step_enum,
TAS675X_DIG_VOL_RAMP_CTRL_REG, 4,
tas675x_ramp_step_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ramp_up_step_enum,
TAS675X_DIG_VOL_RAMP_CTRL_REG, 0,
tas675x_ramp_step_texts);
static const char * const tas675x_vol_combine_ch12_texts[] = {
"Independent", "CH2 follows CH1", "CH1 follows CH2"
};
static SOC_ENUM_SINGLE_DECL(tas675x_vol_combine_ch12_enum,
TAS675X_DIG_VOL_COMBINE_CTRL_REG, 0,
tas675x_vol_combine_ch12_texts);
static const char * const tas675x_vol_combine_ch34_texts[] = {
"Independent", "CH4 follows CH3", "CH3 follows CH4"
};
static SOC_ENUM_SINGLE_DECL(tas675x_vol_combine_ch34_enum,
TAS675X_DIG_VOL_COMBINE_CTRL_REG, 2,
tas675x_vol_combine_ch34_texts);
static const char * const tas675x_auto_mute_time_texts[] = {
"11.5ms", "53ms", "106.5ms", "266.5ms",
"535ms", "1065ms", "2665ms", "5330ms"
};
static SOC_ENUM_SINGLE_DECL(tas675x_ch1_mute_time_enum,
TAS675X_AUTO_MUTE_TIMING_CH1_CH2_REG, 4,
tas675x_auto_mute_time_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ch2_mute_time_enum,
TAS675X_AUTO_MUTE_TIMING_CH1_CH2_REG, 0,
tas675x_auto_mute_time_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ch3_mute_time_enum,
TAS675X_AUTO_MUTE_TIMING_CH3_CH4_REG, 4,
tas675x_auto_mute_time_texts);
static SOC_ENUM_SINGLE_DECL(tas675x_ch4_mute_time_enum,
TAS675X_AUTO_MUTE_TIMING_CH3_CH4_REG, 0,
tas675x_auto_mute_time_texts);
/*
* ALSA Mixer Controls
*
* For detailed documentation of each control see:
* Documentation/sound/codecs/tas675x.rst
*/
static const struct snd_kcontrol_new tas675x_snd_controls[] = {
/* Volume & Gain Control */
SOC_DOUBLE_R_TLV("Analog Playback Volume", TAS675X_ANALOG_GAIN_CH1_CH2_REG,
TAS675X_ANALOG_GAIN_CH3_CH4_REG, 1, 0x1F, 1, tas675x_ana_gain_tlv),
SOC_ENUM("Analog Gain Ramp Step", tas675x_ana_ramp_enum),
SOC_SINGLE_RANGE_TLV("CH1 Digital Playback Volume",
TAS675X_DIG_VOL_CH1_REG, 0, 0x30, 0xFF, 1,
tas675x_dig_vol_tlv),
SOC_SINGLE_RANGE_TLV("CH2 Digital Playback Volume",
TAS675X_DIG_VOL_CH2_REG, 0, 0x30, 0xFF, 1,
tas675x_dig_vol_tlv),
SOC_SINGLE_RANGE_TLV("CH3 Digital Playback Volume",
TAS675X_DIG_VOL_CH3_REG, 0, 0x30, 0xFF, 1,
tas675x_dig_vol_tlv),
SOC_SINGLE_RANGE_TLV("CH4 Digital Playback Volume",
TAS675X_DIG_VOL_CH4_REG, 0, 0x30, 0xFF, 1,
tas675x_dig_vol_tlv),
SOC_ENUM("Volume Ramp Down Rate", tas675x_ramp_down_rate_enum),
SOC_ENUM("Volume Ramp Down Step", tas675x_ramp_down_step_enum),
SOC_ENUM("Volume Ramp Up Rate", tas675x_ramp_up_rate_enum),
SOC_ENUM("Volume Ramp Up Step", tas675x_ramp_up_step_enum),
SOC_ENUM("CH1/2 Volume Combine", tas675x_vol_combine_ch12_enum),
SOC_ENUM("CH3/4 Volume Combine", tas675x_vol_combine_ch34_enum),
/* Auto Mute & Silence Detection */
SOC_SINGLE("CH1 Auto Mute Switch", TAS675X_AUTO_MUTE_EN_REG, 0, 1, 0),
SOC_SINGLE("CH2 Auto Mute Switch", TAS675X_AUTO_MUTE_EN_REG, 1, 1, 0),
SOC_SINGLE("CH3 Auto Mute Switch", TAS675X_AUTO_MUTE_EN_REG, 2, 1, 0),
SOC_SINGLE("CH4 Auto Mute Switch", TAS675X_AUTO_MUTE_EN_REG, 3, 1, 0),
SOC_SINGLE("Auto Mute Combine Switch", TAS675X_AUTO_MUTE_EN_REG, 4, 1, 0),
SOC_ENUM("CH1 Auto Mute Time", tas675x_ch1_mute_time_enum),
SOC_ENUM("CH2 Auto Mute Time", tas675x_ch2_mute_time_enum),
SOC_ENUM("CH3 Auto Mute Time", tas675x_ch3_mute_time_enum),
SOC_ENUM("CH4 Auto Mute Time", tas675x_ch4_mute_time_enum),
/* Clock & EMI Management */
SOC_ENUM("Spread Spectrum Mode", tas675x_ss_enum),
SOC_ENUM("SS Triangle Range", tas675x_ss_tri_range_enum),
SOC_ENUM("SS Random Range", tas675x_ss_rdm_range_enum),
SOC_ENUM("SS Random Dwell Range", tas675x_ss_rdm_dwell_enum),
SOC_SINGLE("SS Triangle Dwell Min", TAS675X_SS_DWELL_CTRL_REG, 4, 15, 0),
SOC_SINGLE("SS Triangle Dwell Max", TAS675X_SS_DWELL_CTRL_REG, 0, 15, 0),
/* Hardware Protection */
SOC_SINGLE("OTSD Auto Recovery Switch", TAS675X_OTSD_RECOVERY_EN_REG, 1, 1, 0),
SOC_ENUM("Overcurrent Limit Level", tas675x_oc_limit_enum),
SOC_ENUM("CH1 OTW Threshold", tas675x_ch1_otw_enum),
SOC_ENUM("CH2 OTW Threshold", tas675x_ch2_otw_enum),
SOC_ENUM("CH3 OTW Threshold", tas675x_ch3_otw_enum),
SOC_ENUM("CH4 OTW Threshold", tas675x_ch4_otw_enum),
/* DSP Signal Path & Mode */
SOC_ENUM("DSP Signal Path Mode", tas675x_dsp_mode_enum),
/* DC Load Diagnostics */
{
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
.name = "DC LDG Trigger",
.access = SNDRV_CTL_ELEM_ACCESS_WRITE,
.info = snd_ctl_boolean_mono_info,
.put = tas675x_set_dcldg_trigger,
},
SOC_SINGLE("DC LDG Auto Diagnostics Switch", TAS675X_DC_LDG_CTRL_REG, 0, 1, 1),
SOC_SINGLE("CH1 LO LDG Switch", TAS675X_DC_LDG_LO_CTRL_REG, 3, 1, 0),
SOC_SINGLE("CH2 LO LDG Switch", TAS675X_DC_LDG_LO_CTRL_REG, 2, 1, 0),
SOC_SINGLE("CH3 LO LDG Switch", TAS675X_DC_LDG_LO_CTRL_REG, 1, 1, 0),
SOC_SINGLE("CH4 LO LDG Switch", TAS675X_DC_LDG_LO_CTRL_REG, 0, 1, 0),
SOC_ENUM("DC LDG SLOL Ramp Time", tas675x_dc_slol_ramp_enum),
SOC_ENUM("DC LDG SLOL Settling Time", tas675x_dc_slol_settling_enum),
SOC_ENUM("DC LDG S2PG Ramp Time", tas675x_dc_s2pg_ramp_enum),
SOC_ENUM("DC LDG S2PG Settling Time", tas675x_dc_s2pg_settling_enum),
SOC_ENUM("CH1 DC LDG SL Threshold", tas675x_ch1_dc_ldg_sl_enum),
SOC_ENUM("CH2 DC LDG SL Threshold", tas675x_ch2_dc_ldg_sl_enum),
SOC_ENUM("CH3 DC LDG SL Threshold", tas675x_ch3_dc_ldg_sl_enum),
SOC_ENUM("CH4 DC LDG SL Threshold", tas675x_ch4_dc_ldg_sl_enum),
SOC_SINGLE_RO("DC LDG Result", TAS675X_DC_LDG_RESULT_REG, 0, 0xFF),
SOC_SINGLE_RO("CH1 DC LDG Report", TAS675X_DC_LDG_REPORT_CH1_CH2_REG, 4, 0x0F),
SOC_SINGLE_RO("CH2 DC LDG Report", TAS675X_DC_LDG_REPORT_CH1_CH2_REG, 0, 0x0F),
SOC_SINGLE_RO("CH3 DC LDG Report", TAS675X_DC_LDG_REPORT_CH3_CH4_REG, 4, 0x0F),
SOC_SINGLE_RO("CH4 DC LDG Report", TAS675X_DC_LDG_REPORT_CH3_CH4_REG, 0, 0x0F),
SOC_SINGLE_RO("CH1 LO LDG Report", TAS675X_DC_LDG_RESULT_REG, 7, 1),
SOC_SINGLE_RO("CH2 LO LDG Report", TAS675X_DC_LDG_RESULT_REG, 6, 1),
SOC_SINGLE_RO("CH3 LO LDG Report", TAS675X_DC_LDG_RESULT_REG, 5, 1),
SOC_SINGLE_RO("CH4 LO LDG Report", TAS675X_DC_LDG_RESULT_REG, 4, 1),
SOC_DC_RESIST_RO("CH1 DC Resistance", TAS675X_CH1_DC_LDG_DCR_LSB_REG),
SOC_DC_RESIST_RO("CH2 DC Resistance", TAS675X_CH2_DC_LDG_DCR_LSB_REG),
SOC_DC_RESIST_RO("CH3 DC Resistance", TAS675X_CH3_DC_LDG_DCR_LSB_REG),
SOC_DC_RESIST_RO("CH4 DC Resistance", TAS675X_CH4_DC_LDG_DCR_LSB_REG),
/* AC Load Diagnostics */
{
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
.name = "AC LDG Trigger",
.access = SNDRV_CTL_ELEM_ACCESS_WRITE,
.info = snd_ctl_boolean_mono_info,
.put = tas675x_set_acldg_trigger,
},
SOC_SINGLE("AC LDG Gain", TAS675X_AC_LDG_CTRL_REG, 4, 1, 0),
SOC_SINGLE("AC LDG Test Frequency", TAS675X_AC_LDG_FREQ_CTRL_REG, 0, 0xFF, 0),
SOC_SINGLE_RO("CH1 AC LDG Real", TAS675X_AC_LDG_REPORT_CH1_R_REG, 0, 0xFF),
SOC_SINGLE_RO("CH1 AC LDG Imag", TAS675X_AC_LDG_REPORT_CH1_I_REG, 0, 0xFF),
SOC_SINGLE_RO("CH2 AC LDG Real", TAS675X_AC_LDG_REPORT_CH2_R_REG, 0, 0xFF),
SOC_SINGLE_RO("CH2 AC LDG Imag", TAS675X_AC_LDG_REPORT_CH2_I_REG, 0, 0xFF),
SOC_SINGLE_RO("CH3 AC LDG Real", TAS675X_AC_LDG_REPORT_CH3_R_REG, 0, 0xFF),
SOC_SINGLE_RO("CH3 AC LDG Imag", TAS675X_AC_LDG_REPORT_CH3_I_REG, 0, 0xFF),
SOC_SINGLE_RO("CH4 AC LDG Real", TAS675X_AC_LDG_REPORT_CH4_R_REG, 0, 0xFF),
SOC_SINGLE_RO("CH4 AC LDG Imag", TAS675X_AC_LDG_REPORT_CH4_I_REG, 0, 0xFF),
/* Temperature and Voltage Monitoring */
SOC_SINGLE_RO("PVDD Sense", TAS675X_PVDD_SENSE_REG, 0, 0xFF),
SOC_SINGLE_RO("Global Temperature", TAS675X_TEMP_GLOBAL_REG, 0, 0xFF),
SOC_SINGLE_RO("CH1 Temperature Range", TAS675X_TEMP_CH1_CH2_REG, 6, 3),
SOC_SINGLE_RO("CH2 Temperature Range", TAS675X_TEMP_CH1_CH2_REG, 4, 3),
SOC_SINGLE_RO("CH3 Temperature Range", TAS675X_TEMP_CH3_CH4_REG, 2, 3),
SOC_SINGLE_RO("CH4 Temperature Range", TAS675X_TEMP_CH3_CH4_REG, 0, 3),
/* Speaker Protection & Detection */
SOC_SINGLE("Tweeter Detection Switch", TAS675X_TWEETER_DETECT_CTRL_REG, 0, 1, 1),
SOC_SINGLE("Tweeter Detect Threshold", TAS675X_TWEETER_DETECT_THRESH_REG, 0, 0xFF, 0),
SOC_SINGLE_RO("CH1 Tweeter Detect Report", TAS675X_TWEETER_REPORT_REG, 3, 1),
SOC_SINGLE_RO("CH2 Tweeter Detect Report", TAS675X_TWEETER_REPORT_REG, 2, 1),
SOC_SINGLE_RO("CH3 Tweeter Detect Report", TAS675X_TWEETER_REPORT_REG, 1, 1),
SOC_SINGLE_RO("CH4 Tweeter Detect Report", TAS675X_TWEETER_REPORT_REG, 0, 1),
/*
* Unavailable in LLP, available in Normal & FFLP
*/
SOC_SINGLE("Thermal Foldback Switch", TAS675X_DSP_CTRL_REG, 0, 1, 0),
SOC_SINGLE("PVDD Foldback Switch", TAS675X_DSP_CTRL_REG, 4, 1, 0),
SOC_SINGLE("DC Blocker Bypass Switch", TAS675X_DC_BLOCK_BYP_REG, 0, 1, 0),
SOC_SINGLE("Clip Detect Switch", TAS675X_CLIP_DETECT_CTRL_REG, 6, 1, 0),
SOC_SINGLE("Audio SDOUT Switch", TAS675X_DSP_CTRL_REG, 5, 1, 0),
/*
* Unavailable in both FFLP and LLP, Normal mode only
*/
/* Real-Time Load Diagnostics */
SOC_SINGLE("CH1 RTLDG Switch", TAS675X_RTLDG_EN_REG, 3, 1, 0),
SOC_SINGLE("CH2 RTLDG Switch", TAS675X_RTLDG_EN_REG, 2, 1, 0),
SOC_SINGLE("CH3 RTLDG Switch", TAS675X_RTLDG_EN_REG, 1, 1, 0),
SOC_SINGLE("CH4 RTLDG Switch", TAS675X_RTLDG_EN_REG, 0, 1, 0),
SOC_SINGLE("RTLDG Clip Mask Switch", TAS675X_RTLDG_EN_REG, 4, 1, 0),
SOC_SINGLE("ISENSE Calibration Switch", TAS675X_ISENSE_CAL_REG, 3, 1, 0),
SOC_DSP_THRESH_EXT("RTLDG Open Load Threshold",
tas675x_dsp_defaults[TAS675X_DSP_PARAM_ID_OL_THRESH]),
SOC_DSP_THRESH_EXT("RTLDG Short Load Threshold",
tas675x_dsp_defaults[TAS675X_DSP_PARAM_ID_SL_THRESH]),
SOC_RTLDG_IMP_RO("CH1 RTLDG Impedance", TAS675X_CH1_RTLDG_IMP_MSB_REG),
SOC_RTLDG_IMP_RO("CH2 RTLDG Impedance", TAS675X_CH2_RTLDG_IMP_MSB_REG),
SOC_RTLDG_IMP_RO("CH3 RTLDG Impedance", TAS675X_CH3_RTLDG_IMP_MSB_REG),
SOC_RTLDG_IMP_RO("CH4 RTLDG Impedance", TAS675X_CH4_RTLDG_IMP_MSB_REG),
};
static const struct snd_kcontrol_new tas675x_audio_path_switch =
SOC_DAPM_SINGLE("Switch", SND_SOC_NOPM, 0, 1, 1);
static const struct snd_kcontrol_new tas675x_anc_path_switch =
SOC_DAPM_SINGLE("Switch", SND_SOC_NOPM, 0, 1, 1);
static const struct snd_soc_dapm_widget tas675x_dapm_widgets[] = {
SND_SOC_DAPM_SUPPLY("Analog Core", SND_SOC_NOPM, 0, 0, NULL, 0),
SND_SOC_DAPM_SUPPLY("SDOUT Vpredict", SND_SOC_NOPM, 0, 0, NULL, 0),
SND_SOC_DAPM_SUPPLY("SDOUT Isense", SND_SOC_NOPM, 0, 0, NULL, 0),
SND_SOC_DAPM_DAC("Audio DAC", "Playback", SND_SOC_NOPM, 0, 0),
SND_SOC_DAPM_DAC("ANC DAC", "ANC Playback", SND_SOC_NOPM, 0, 0),
SND_SOC_DAPM_ADC("Feedback ADC", "Feedback Capture", SND_SOC_NOPM, 0, 0),
SND_SOC_DAPM_SWITCH("Audio Path", SND_SOC_NOPM, 0, 0,
&tas675x_audio_path_switch),
SND_SOC_DAPM_SWITCH("ANC Path", SND_SOC_NOPM, 0, 0,
&tas675x_anc_path_switch),
/*
* Even though all channels are coupled in terms of power control,
* use logical outputs for each channel to allow independent routing
* and DAPM controls if needed.
*/
SND_SOC_DAPM_OUTPUT("OUT_CH1"),
SND_SOC_DAPM_OUTPUT("OUT_CH2"),
SND_SOC_DAPM_OUTPUT("OUT_CH3"),
SND_SOC_DAPM_OUTPUT("OUT_CH4"),
SND_SOC_DAPM_INPUT("SPEAKER_LOAD"),
};
static const struct snd_soc_dapm_route tas675x_dapm_routes[] = {
{ "Audio DAC", NULL, "Analog Core" },
{ "Audio Path", "Switch", "Audio DAC" },
{ "OUT_CH1", NULL, "Audio Path" },
{ "OUT_CH2", NULL, "Audio Path" },
{ "OUT_CH3", NULL, "Audio Path" },
{ "OUT_CH4", NULL, "Audio Path" },
{ "ANC DAC", NULL, "Analog Core" },
{ "ANC Path", "Switch", "ANC DAC" },
{ "OUT_CH1", NULL, "ANC Path" },
{ "OUT_CH2", NULL, "ANC Path" },
{ "OUT_CH3", NULL, "ANC Path" },
{ "OUT_CH4", NULL, "ANC Path" },
{ "Feedback ADC", NULL, "Analog Core" },
{ "Feedback ADC", NULL, "SDOUT Vpredict" },
{ "Feedback ADC", NULL, "SDOUT Isense" },
{ "Feedback ADC", NULL, "SPEAKER_LOAD" },
};
static void tas675x_program_slot_offsets(struct tas675x_priv *tas,
int dai_id, int slot_width)
{
int offset = 0;
switch (dai_id) {
case 0:
/* Standard Audio on SDIN */
if (tas->audio_slot >= 0)
offset = tas->audio_slot * slot_width;
else if (tas->tx_mask)
offset = __ffs(tas->tx_mask) * slot_width;
else
return;
offset += tas->bclk_offset;
regmap_update_bits(tas->regmap, TAS675X_SDIN_OFFSET_MSB_REG,
TAS675X_SDIN_AUDIO_OFF_MSB_MASK,
FIELD_PREP(TAS675X_SDIN_AUDIO_OFF_MSB_MASK, offset >> 8));
regmap_write(tas->regmap, TAS675X_SDIN_AUDIO_OFFSET_REG,
offset & 0xFF);
break;
case 1:
/*
* Low-Latency Playback on SDIN, **only** enabled in LLP mode
* and to be mixed with main audio before output amplification
* to achieve ANC/RNC.
*/
if (tas->llp_slot >= 0)
offset = tas->llp_slot * slot_width;
else if (tas->tx_mask)
offset = __ffs(tas->tx_mask) * slot_width;
else
return;
offset += tas->bclk_offset;
regmap_update_bits(tas->regmap, TAS675X_SDIN_OFFSET_MSB_REG,
TAS675X_SDIN_LL_OFF_MSB_MASK,
FIELD_PREP(TAS675X_SDIN_LL_OFF_MSB_MASK, offset >> 8));
regmap_write(tas->regmap, TAS675X_SDIN_LL_OFFSET_REG,
offset & 0xFF);
break;
case 2:
/* SDOUT Data Output (Vpredict + Isense feedback) */
if (!tas->slot_width)
break;
if (tas->vpredict_slot >= 0) {
offset = tas->vpredict_slot * slot_width;
offset += tas->bclk_offset;
regmap_update_bits(tas->regmap, TAS675X_SDOUT_OFFSET_MSB_REG,
TAS675X_SDOUT_VP_OFF_MSB_MASK,
FIELD_PREP(TAS675X_SDOUT_VP_OFF_MSB_MASK, offset >> 8));
regmap_write(tas->regmap, TAS675X_VPREDICT_OFFSET_REG,
offset & 0xFF);
}
if (tas->isense_slot >= 0) {
offset = tas->isense_slot * slot_width;
offset += tas->bclk_offset;
regmap_update_bits(tas->regmap, TAS675X_SDOUT_OFFSET_MSB_REG,
TAS675X_SDOUT_IS_OFF_MSB_MASK,
FIELD_PREP(TAS675X_SDOUT_IS_OFF_MSB_MASK, offset >> 8));
regmap_write(tas->regmap, TAS675X_ISENSE_OFFSET_REG,
offset & 0xFF);
}
break;
}
if (offset > 511)
dev_warn(tas->dev,
"DAI %d slot offset %d exceeds 511 SCLK limit\n",
dai_id, offset);
}
static int tas675x_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 tas675x_priv *tas = snd_soc_component_get_drvdata(component);
unsigned int rate = params_rate(params);
u8 word_length;
/*
* Single clock domain: SDIN and SDOUT share one SCLK/FSYNC pair,
* so all active DAIs must use the same sample rate.
*/
if ((tas->active_playback_dais || tas->active_capture_dais) &&
tas->rate && tas->rate != rate) {
dev_err(component->dev,
"Rate %u conflicts with active rate %u\n",
rate, tas->rate);
return -EINVAL;
}
switch (params_width(params)) {
case 16:
word_length = TAS675X_WL_16BIT;
break;
case 20:
word_length = TAS675X_WL_20BIT;
break;
case 24:
word_length = TAS675X_WL_24BIT;
break;
case 32:
word_length = TAS675X_WL_32BIT;
break;
default:
return -EINVAL;
}
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
/*
* RTLDG is not supported above 96kHz. Auto-disable to
* prevent DSP overload and restore when rate drops back.
*/
if (rate > 96000) {
unsigned int val;
regmap_read(component->regmap, TAS675X_RTLDG_EN_REG,
&val);
if (val & TAS675X_RTLDG_CH_EN_MASK) {
tas->saved_rtldg_en = val;
dev_dbg(component->dev,
"Sample rate %dHz > 96kHz: Auto-disabling RTLDG\n",
rate);
regmap_update_bits(component->regmap,
TAS675X_RTLDG_EN_REG,
TAS675X_RTLDG_CH_EN_MASK,
0x00);
}
} else if (tas->saved_rtldg_en) {
unsigned int cur;
/*
* Respect overrides and only restore if RTLDG is still auto-disabled
*/
regmap_read(component->regmap, TAS675X_RTLDG_EN_REG,
&cur);
if (!(cur & TAS675X_RTLDG_CH_EN_MASK)) {
dev_dbg(component->dev,
"Restoring RTLDG config after high-rate stream\n");
regmap_update_bits(component->regmap,
TAS675X_RTLDG_EN_REG,
TAS675X_RTLDG_CH_EN_MASK,
TAS675X_RTLDG_CH_EN_MASK &
tas->saved_rtldg_en);
}
tas->saved_rtldg_en = 0;
}
/* Set SDIN word length (audio path + low-latency path) */
regmap_update_bits(component->regmap, TAS675X_SDIN_CTRL_REG,
TAS675X_SDIN_WL_MASK,
FIELD_PREP(TAS675X_SDIN_AUDIO_WL_MASK, word_length) |
FIELD_PREP(TAS675X_SDIN_LL_WL_MASK, word_length));
} else {
/* Set SDOUT word length (VPREDICT + ISENSE) for capture */
regmap_update_bits(component->regmap, TAS675X_SDOUT_CTRL_REG,
TAS675X_SDOUT_WL_MASK,
FIELD_PREP(TAS675X_SDOUT_VP_WL_MASK, word_length) |
FIELD_PREP(TAS675X_SDOUT_IS_WL_MASK, word_length));
}
tas675x_program_slot_offsets(tas, dai->id,
tas->slot_width ?: params_width(params));
tas->rate = rate;
return 0;
}
static int tas675x_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
{
struct snd_soc_component *component = dai->component;
struct tas675x_priv *tas = snd_soc_component_get_drvdata(component);
bool tdm_mode = false, i2s_mode = false;
/* Enforce Clocking Direction (Codec is strictly a consumer) */
switch (fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) {
case SND_SOC_DAIFMT_BC_FC:
break;
default:
dev_err(component->dev, "Unsupported clock provider format\n");
return -EINVAL;
}
/* SCLK polarity: NB_NF or IB_NF only (no FSYNC inversion support) */
switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
case SND_SOC_DAIFMT_NB_NF:
regmap_update_bits(component->regmap, TAS675X_SCLK_INV_CTRL_REG,
TAS675X_SCLK_INV_MASK, 0x00);
break;
case SND_SOC_DAIFMT_IB_NF:
regmap_update_bits(component->regmap, TAS675X_SCLK_INV_CTRL_REG,
TAS675X_SCLK_INV_MASK, TAS675X_SCLK_INV_MASK);
break;
default:
dev_err(component->dev, "Unsupported clock inversion\n");
return -EINVAL;
}
/* Configure Audio Format and TDM Enable */
switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
case SND_SOC_DAIFMT_I2S:
i2s_mode = true;
tas->bclk_offset = 0;
regmap_update_bits(component->regmap, TAS675X_AUDIO_IF_CTRL_REG,
TAS675X_TDM_EN_BIT | TAS675X_SAP_FMT_MASK |
TAS675X_FS_PULSE_MASK,
TAS675X_SAP_FMT_I2S);
regmap_update_bits(component->regmap, TAS675X_SDOUT_CTRL_REG,
TAS675X_SDOUT_SELECT_MASK,
TAS675X_SDOUT_SELECT_NON_TDM);
break;
case SND_SOC_DAIFMT_RIGHT_J:
tas->bclk_offset = 0;
regmap_update_bits(component->regmap, TAS675X_AUDIO_IF_CTRL_REG,
TAS675X_TDM_EN_BIT | TAS675X_SAP_FMT_MASK |
TAS675X_FS_PULSE_MASK,
TAS675X_SAP_FMT_RIGHT_J);
regmap_update_bits(component->regmap, TAS675X_SDOUT_CTRL_REG,
TAS675X_SDOUT_SELECT_MASK,
TAS675X_SDOUT_SELECT_NON_TDM);
break;
case SND_SOC_DAIFMT_LEFT_J:
tas->bclk_offset = 0;
regmap_update_bits(component->regmap, TAS675X_AUDIO_IF_CTRL_REG,
TAS675X_TDM_EN_BIT | TAS675X_SAP_FMT_MASK |
TAS675X_FS_PULSE_MASK,
TAS675X_SAP_FMT_LEFT_J);
regmap_update_bits(component->regmap, TAS675X_SDOUT_CTRL_REG,
TAS675X_SDOUT_SELECT_MASK,
TAS675X_SDOUT_SELECT_NON_TDM);
break;
case SND_SOC_DAIFMT_DSP_A:
tdm_mode = true;
tas->bclk_offset = 1;
regmap_update_bits(component->regmap, TAS675X_AUDIO_IF_CTRL_REG,
TAS675X_TDM_EN_BIT | TAS675X_SAP_FMT_MASK |
TAS675X_FS_PULSE_MASK,
TAS675X_TDM_EN_BIT | TAS675X_SAP_FMT_TDM |
TAS675X_FS_PULSE_SHORT);
regmap_update_bits(component->regmap, TAS675X_SDOUT_CTRL_REG,
TAS675X_SDOUT_SELECT_MASK,
TAS675X_SDOUT_SELECT_TDM_SDOUT1);
break;
case SND_SOC_DAIFMT_DSP_B:
tdm_mode = true;
tas->bclk_offset = 0;
regmap_update_bits(component->regmap, TAS675X_AUDIO_IF_CTRL_REG,
TAS675X_TDM_EN_BIT | TAS675X_SAP_FMT_MASK |
TAS675X_FS_PULSE_MASK,
TAS675X_TDM_EN_BIT | TAS675X_SAP_FMT_TDM |
TAS675X_FS_PULSE_SHORT);
regmap_update_bits(component->regmap, TAS675X_SDOUT_CTRL_REG,
TAS675X_SDOUT_SELECT_MASK,
TAS675X_SDOUT_SELECT_TDM_SDOUT1);
break;
default:
dev_err(component->dev, "Unsupported DAI format\n");
return -EINVAL;
}
/* Setup Vpredict and Isense outputs */
if (dai->id == 2) {
unsigned int sdout_en = 0;
if (tdm_mode) {
/* TDM: Vpredict and Isense may coexist on separate slots */
if (tas->vpredict_slot >= 0)
sdout_en |= TAS675X_SDOUT_EN_VPREDICT;
if (tas->isense_slot >= 0)
sdout_en |= TAS675X_SDOUT_EN_ISENSE;
regmap_update_bits(component->regmap,
TAS675X_SDOUT_EN_REG,
TAS675X_SDOUT_EN_VPREDICT |
TAS675X_SDOUT_EN_ISENSE,
sdout_en);
if (tas->vpredict_slot >= 0 && tas->isense_slot >= 0 &&
abs(tas->vpredict_slot - tas->isense_slot) < 4)
dev_warn(component->dev,
"ti,vpredict-slot-no and ti,isense-slot-no overlaps (each occupies 4 consecutive slots)\n");
} else if (i2s_mode) {
/* I2S: only one source at a time; Vpredict takes priority */
if (tas->vpredict_slot >= 0)
sdout_en = TAS675X_SDOUT_NON_TDM_SEL_VPREDICT |
TAS675X_SDOUT_EN_NON_TDM_ALL;
else if (tas->isense_slot >= 0)
sdout_en = TAS675X_SDOUT_NON_TDM_SEL_ISENSE |
TAS675X_SDOUT_EN_NON_TDM_ALL;
regmap_update_bits(component->regmap,
TAS675X_SDOUT_EN_REG,
TAS675X_SDOUT_NON_TDM_SEL_MASK |
TAS675X_SDOUT_EN_NON_TDM_ALL,
sdout_en);
if (sdout_en &&
tas->gpio1_func != TAS675X_GPIO_SEL_SDOUT2 &&
tas->gpio2_func != TAS675X_GPIO_SEL_SDOUT2)
dev_warn(component->dev,
"sdout enabled in I2S mode but no GPIO configured as SDOUT2; Ch3/Ch4 will be absent\n");
}
}
return 0;
}
static int tas675x_set_tdm_slot(struct snd_soc_dai *dai, unsigned int tx_mask,
unsigned int rx_mask, int slots, int slot_width)
{
struct tas675x_priv *tas = snd_soc_component_get_drvdata(dai->component);
if (slots == 0) {
tas->slot_width = 0;
tas->tx_mask = 0;
return 0;
}
/* No rx_mask as hardware does not support channel muxing for capture */
tas->slot_width = slot_width;
tas->tx_mask = tx_mask;
return 0;
}
static int tas675x_mute_stream(struct snd_soc_dai *dai, int mute, int direction)
{
struct snd_soc_component *component = dai->component;
struct tas675x_priv *tas = snd_soc_component_get_drvdata(component);
unsigned int discard;
int ret;
if (direction == SNDRV_PCM_STREAM_CAPTURE) {
if (mute)
clear_bit(dai->id, &tas->active_capture_dais);
else
set_bit(dai->id, &tas->active_capture_dais);
return 0;
}
/*
* Track which playback DAIs are active.
* The TAS675x has two playback DAIs (main audio and LLP).
* Only transition to SLEEP when ALL are muted.
*/
if (mute)
clear_bit(dai->id, &tas->active_playback_dais);
else
set_bit(dai->id, &tas->active_playback_dais);
/* Last playback stream */
if (mute && !tas->active_playback_dais) {
ret = tas675x_set_state_all(tas, TAS675X_STATE_SLEEP_BOTH);
regmap_read(tas->regmap, TAS675X_CLK_FAULT_LATCHED_REG, &discard);
return ret;
}
return tas675x_set_state_all(tas,
tas->active_playback_dais ?
TAS675X_STATE_PLAY_BOTH :
TAS675X_STATE_SLEEP_BOTH);
}
static const struct snd_soc_dai_ops tas675x_dai_ops = {
.hw_params = tas675x_hw_params,
.set_fmt = tas675x_set_fmt,
.set_tdm_slot = tas675x_set_tdm_slot,
.mute_stream = tas675x_mute_stream,
};
static struct snd_soc_dai_driver tas675x_dais[] = {
{
.name = "tas675x-audio",
.id = 0,
.playback = {
.stream_name = "Playback",
.channels_min = 2,
.channels_max = 4,
.rates = SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000 |
SNDRV_PCM_RATE_96000 | SNDRV_PCM_RATE_192000,
.formats = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_LE |
SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE,
},
.ops = &tas675x_dai_ops,
},
/* Only available when Low Latency Path (LLP) is enabled */
{
.name = "tas675x-anc",
.id = 1,
.playback = {
.stream_name = "ANC Playback",
.channels_min = 2,
.channels_max = 4,
.rates = SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_96000,
.formats = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_LE |
SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE,
},
.ops = &tas675x_dai_ops,
},
{
.name = "tas675x-feedback",
.id = 2,
.capture = {
.stream_name = "Feedback Capture",
.channels_min = 2,
.channels_max = 8,
.rates = SNDRV_PCM_RATE_48000,
.formats = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_LE |
SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE,
},
.ops = &tas675x_dai_ops,
}
};
/*
* Enable regulators and release hardware reset GPIOs.
* The device is not I2C-accessible until this returns.
*/
static int tas675x_hw_enable(struct tas675x_priv *tas)
{
int ret;
ret = regulator_bulk_enable(ARRAY_SIZE(tas->supplies), tas->supplies);
if (ret) {
dev_err(tas->dev, "Failed to enable regulators: %d\n", ret);
return ret;
}
if (!IS_ERR(tas->vbat)) {
ret = regulator_enable(tas->vbat);
if (ret) {
dev_err(tas->dev, "Failed to enable vbat: %d\n", ret);
regulator_bulk_disable(ARRAY_SIZE(tas->supplies), tas->supplies);
return ret;
}
}
if (tas->pd_gpio && tas->stby_gpio) {
/*
* Independent Pin Control
* Deassert PD first to boot digital, then STBY for analog.
*/
/* Min 4ms digital boot wait */
gpiod_set_value_cansleep(tas->pd_gpio, 0);
usleep_range(4000, 5000);
/* ~2ms analog stabilization */
gpiod_set_value_cansleep(tas->stby_gpio, 0);
usleep_range(2000, 3000);
} else if (tas->pd_gpio) {
/*
* Simultaneous Pin Release
* STBY tied to PD or hardwired HIGH.
*/
/* 6ms wait for simultaneous release transition */
gpiod_set_value_cansleep(tas->pd_gpio, 0);
usleep_range(6000, 7000);
} else {
/*
* PD hardwired, device in DEEP_SLEEP.
* Digital core already booted, I2C active. Deassert STBY
* to bring up the analog output stage.
*/
/* ~2ms analog stabilization */
gpiod_set_value_cansleep(tas->stby_gpio, 0);
usleep_range(2000, 3000);
}
return 0;
}
static void tas675x_hw_disable(struct tas675x_priv *tas)
{
if (tas->stby_gpio)
gpiod_set_value_cansleep(tas->stby_gpio, 1);
if (tas->pd_gpio)
gpiod_set_value_cansleep(tas->pd_gpio, 1);
/*
* Hold PD/STBY asserted for at least 10ms
* before removing PVDD, VBAT or DVDD.
*/
usleep_range(10000, 11000);
if (!IS_ERR(tas->vbat))
regulator_disable(tas->vbat);
regulator_bulk_disable(ARRAY_SIZE(tas->supplies), tas->supplies);
}
/*
* Write device start-up defaults.
* Must be called after tas675x_hw_enable() and after regcache is enabled.
*/
static int tas675x_init_device(struct tas675x_priv *tas)
{
struct regmap *regmap = tas->regmap;
unsigned int val;
int ret, i;
/* Clear POR fault flag to prevent IRQ storm */
regmap_read(regmap, TAS675X_POWER_FAULT_LATCHED_REG, &val);
/* Bypass DC Load Diagnostics for fast boot */
if (tas->fast_boot)
regmap_update_bits(regmap, TAS675X_DC_LDG_CTRL_REG,
TAS675X_LDG_ABORT_BIT | TAS675X_LDG_BYPASS_BIT,
TAS675X_LDG_ABORT_BIT | TAS675X_LDG_BYPASS_BIT);
tas675x_select_book(regmap, TAS675X_BOOK_DEFAULT);
/* Enter setup mode */
ret = regmap_write(regmap, TAS675X_SETUP_REG1, TAS675X_SETUP_ENTER_VAL1);
if (ret)
goto err;
ret = regmap_write(regmap, TAS675X_SETUP_REG2, TAS675X_SETUP_ENTER_VAL2);
if (ret)
goto err;
/* Set all channels to Sleep (required before Page 1 config) */
tas675x_set_state_all(tas, TAS675X_STATE_SLEEP_BOTH);
/* Set DAC clock per TRM startup script */
regmap_write(regmap, TAS675X_DAC_CLK_REG, 0x00);
/*
* Switch to Page 1 for safety-critical OC/CBC configuration,
* while bypassing regcache. (Page 1 not accessible post setup)
*/
regcache_cache_bypass(regmap, true);
ret = regmap_multi_reg_write(regmap, tas675x_page1_init,
ARRAY_SIZE(tas675x_page1_init));
regcache_cache_bypass(regmap, false);
if (ret)
goto err_setup;
/* Resync regmap's cached page selector */
regmap_write(regmap, TAS675X_PAGE_CTRL_REG, 0x00);
/* Exit setup mode */
regmap_write(regmap, TAS675X_SETUP_REG1, TAS675X_SETUP_EXIT_VAL);
regmap_write(regmap, TAS675X_SETUP_REG2, TAS675X_SETUP_EXIT_VAL);
/* Write DSP parameters if cached */
for (i = 0; i < ARRAY_SIZE(tas->dsp_params); i++) {
if (tas->dsp_params[i].val)
tas675x_dsp_mem_write(tas,
tas->dsp_params[i].page,
tas->dsp_params[i].reg,
tas->dsp_params[i].val);
}
/*
* Configure fault and warning event routing:
*
* ROUTING_1: CP fault/UVLO latch, OUTM soft short latch
* ROUTING_2: CBC latch, OTSD latch, OTSD, power fault
* ROUTING_3: CBC latch, OTSD latch, power latch, DC LDG,
* OTSD, power warnings
* ROUTING_4: OC latch, DC latch, protection shutdown
* OTW latch, OTW, clip latch
* ROUTING_5: clock latch+non-latch, RTLDG latch
* CBC warning, clip warning
*/
regmap_write(regmap, TAS675X_REPORT_ROUTING_1_REG, 0x70);
regmap_write(regmap, TAS675X_REPORT_ROUTING_2_REG, 0xA3);
regmap_write(regmap, TAS675X_REPORT_ROUTING_3_REG, 0xBB);
regmap_write(regmap, TAS675X_REPORT_ROUTING_4_REG, 0x7E);
regmap_write(regmap, TAS675X_REPORT_ROUTING_5_REG, 0xF3);
/* Configure GPIO pins if specified in DT */
if (tas->gpio1_func >= 0 || tas->gpio2_func >= 0) {
unsigned int gpio_ctrl = TAS675X_GPIO_CTRL_RSTVAL;
tas675x_config_gpio_pin(regmap, tas->gpio1_func,
TAS675X_GPIO1_OUTPUT_SEL_REG,
0, &gpio_ctrl);
tas675x_config_gpio_pin(regmap, tas->gpio2_func,
TAS675X_GPIO2_OUTPUT_SEL_REG,
1, &gpio_ctrl);
regmap_write(regmap, TAS675X_GPIO_CTRL_REG, gpio_ctrl);
}
/* Clear fast boot bits */
if (tas->fast_boot)
regmap_update_bits(regmap, TAS675X_DC_LDG_CTRL_REG,
TAS675X_LDG_ABORT_BIT | TAS675X_LDG_BYPASS_BIT,
0);
/* Clear any stale faults from the boot sequence */
regmap_read(regmap, TAS675X_POWER_FAULT_STATUS_1_REG, &val);
regmap_read(regmap, TAS675X_POWER_FAULT_LATCHED_REG, &val);
regmap_read(regmap, TAS675X_CLK_FAULT_LATCHED_REG, &val);
regmap_write(regmap, TAS675X_RESET_REG, TAS675X_FAULT_CLEAR);
return 0;
err_setup:
regmap_write(regmap, TAS675X_SETUP_REG1, TAS675X_SETUP_EXIT_VAL);
regmap_write(regmap, TAS675X_SETUP_REG2, TAS675X_SETUP_EXIT_VAL);
err:
dev_err(tas->dev, "Init device failed: %d\n", ret);
return ret;
}
static void tas675x_power_off(struct tas675x_priv *tas)
{
regcache_cache_only(tas->regmap, true);
regcache_mark_dirty(tas->regmap);
tas675x_hw_disable(tas);
}
static int tas675x_power_on(struct tas675x_priv *tas)
{
int ret;
ret = tas675x_hw_enable(tas);
if (ret)
return ret;
regcache_cache_only(tas->regmap, false);
regcache_mark_dirty(tas->regmap);
ret = tas675x_init_device(tas);
if (ret)
goto err_disable;
ret = regcache_sync(tas->regmap);
if (ret) {
dev_err(tas->dev, "Failed to sync regcache: %d\n", ret);
goto err_disable;
}
/* Reset fault tracking */
memset(tas->last_status, 0, sizeof(tas->last_status));
return 0;
err_disable:
tas675x_power_off(tas);
return ret;
}
static int tas675x_runtime_suspend(struct device *dev)
{
struct tas675x_priv *tas = dev_get_drvdata(dev);
disable_delayed_work_sync(&tas->fault_check_work);
tas675x_set_state_all(tas, TAS675X_STATE_SLEEP_BOTH);
return 0;
}
static int tas675x_runtime_resume(struct device *dev)
{
struct tas675x_priv *tas = dev_get_drvdata(dev);
tas675x_set_state_all(tas, TAS675X_STATE_SLEEP_BOTH);
if (!to_i2c_client(dev)->irq) {
enable_delayed_work(&tas->fault_check_work);
schedule_delayed_work(&tas->fault_check_work,
msecs_to_jiffies(TAS675X_FAULT_CHECK_INTERVAL_MS));
}
return 0;
}
static int tas675x_system_suspend(struct device *dev)
{
struct tas675x_priv *tas = dev_get_drvdata(dev);
int ret;
ret = tas675x_runtime_suspend(dev);
if (ret)
return ret;
if (to_i2c_client(dev)->irq)
disable_irq(to_i2c_client(dev)->irq);
tas675x_power_off(tas);
return 0;
}
static int tas675x_system_resume(struct device *dev)
{
struct tas675x_priv *tas = dev_get_drvdata(dev);
int ret;
ret = tas675x_power_on(tas);
if (ret)
return ret;
if (to_i2c_client(dev)->irq)
enable_irq(to_i2c_client(dev)->irq);
return tas675x_runtime_resume(dev);
}
static const struct snd_soc_component_driver soc_codec_dev_tas675x = {
.controls = tas675x_snd_controls,
.num_controls = ARRAY_SIZE(tas675x_snd_controls),
.dapm_widgets = tas675x_dapm_widgets,
.num_dapm_widgets = ARRAY_SIZE(tas675x_dapm_widgets),
.dapm_routes = tas675x_dapm_routes,
.num_dapm_routes = ARRAY_SIZE(tas675x_dapm_routes),
.endianness = 1,
};
/* Fault register flags */
#define TAS675X_FAULT_CRITICAL BIT(0) /* causes FAULT state, FAULT_CLEAR required */
#define TAS675X_FAULT_TRACK BIT(1) /* track last value, only log on change */
#define TAS675X_FAULT_ACTIVE BIT(2) /* skip when no stream is active */
struct tas675x_fault_reg {
unsigned int reg;
unsigned int flags;
const char *name;
};
static const struct tas675x_fault_reg tas675x_fault_table[] = {
/* Critical */
{ TAS675X_OTSD_LATCHED_REG, TAS675X_FAULT_CRITICAL | TAS675X_FAULT_TRACK,
"Overtemperature Shutdown" },
{ TAS675X_OC_DC_FAULT_LATCHED_REG, TAS675X_FAULT_CRITICAL | TAS675X_FAULT_TRACK,
"Overcurrent / DC Fault" },
{ TAS675X_RTLDG_OL_SL_FAULT_LATCHED_REG, TAS675X_FAULT_CRITICAL | TAS675X_FAULT_TRACK,
"Real-Time Load Diagnostic Fault" },
{ TAS675X_CBC_FAULT_WARN_LATCHED_REG, TAS675X_FAULT_CRITICAL | TAS675X_FAULT_TRACK,
"CBC Fault/Warning" },
/* Warning */
{ TAS675X_POWER_FAULT_STATUS_1_REG, TAS675X_FAULT_TRACK,
"CP / OUTM Fault" },
{ TAS675X_POWER_FAULT_LATCHED_REG, TAS675X_FAULT_TRACK,
"Power Fault" },
{ TAS675X_CLK_FAULT_LATCHED_REG, TAS675X_FAULT_TRACK | TAS675X_FAULT_ACTIVE,
"Clock Fault" },
{ TAS675X_OTW_LATCHED_REG, TAS675X_FAULT_TRACK,
"Overtemperature Warning" },
{ TAS675X_CLIP_WARN_LATCHED_REG, TAS675X_FAULT_ACTIVE,
"Clip Warning" },
};
static_assert(ARRAY_SIZE(tas675x_fault_table) == TAS675X_FAULT_REGS_NUM);
/*
* Read and log all latched fault registers.
* Shared by both the polled fault_check_work and IRQ handler paths
* (which are mutually exclusive, only one is active per device).
* Returns true if any fault register needs to be cleared.
*
* For deciphering fault messages, see "Fault Monitoring" in
* Documentation/sound/codecs/tas675x.rst
*/
static bool tas675x_check_faults(struct tas675x_priv *tas)
{
struct device *dev = tas->dev;
bool needs_clear = false;
unsigned int reg;
int i, ret;
for (i = 0; i < ARRAY_SIZE(tas675x_fault_table); i++) {
const struct tas675x_fault_reg *f = &tas675x_fault_table[i];
ret = regmap_read(tas->regmap, f->reg, &reg);
if (ret) {
if (f->flags & TAS675X_FAULT_CRITICAL) {
dev_err(dev, "failed to read %s: %d\n", f->name, ret);
return needs_clear;
}
continue;
}
if (reg)
needs_clear = true;
/* Skip logging stream-dependent events when no stream is active */
if ((f->flags & TAS675X_FAULT_ACTIVE) &&
!READ_ONCE(tas->active_playback_dais) &&
!READ_ONCE(tas->active_capture_dais))
continue;
/* Log on change or on every non-zero read */
if (reg && (!(f->flags & TAS675X_FAULT_TRACK) ||
reg != tas->last_status[i])) {
if (f->flags & TAS675X_FAULT_CRITICAL)
dev_crit(dev, "%s Latched: 0x%02x\n", f->name, reg);
else
dev_warn(dev, "%s Latched: 0x%02x\n", f->name, reg);
}
if (f->flags & TAS675X_FAULT_TRACK)
tas->last_status[i] = reg;
}
return needs_clear;
}
static void tas675x_fault_check_work(struct work_struct *work)
{
struct tas675x_priv *tas = container_of(work, struct tas675x_priv,
fault_check_work.work);
if (tas675x_check_faults(tas))
regmap_write(tas->regmap, TAS675X_RESET_REG, TAS675X_FAULT_CLEAR);
schedule_delayed_work(&tas->fault_check_work,
msecs_to_jiffies(TAS675X_FAULT_CHECK_INTERVAL_MS));
}
static irqreturn_t tas675x_irq_handler(int irq, void *data)
{
struct tas675x_priv *tas = data;
irqreturn_t ret = IRQ_NONE;
if (pm_runtime_resume_and_get(tas->dev) < 0)
return IRQ_NONE;
if (tas675x_check_faults(tas)) {
regmap_write(tas->regmap, TAS675X_RESET_REG, TAS675X_FAULT_CLEAR);
ret = IRQ_HANDLED;
}
pm_runtime_mark_last_busy(tas->dev);
pm_runtime_put_autosuspend(tas->dev);
return ret;
}
static const struct reg_default tas675x_reg_defaults[] = {
{ TAS675X_PAGE_CTRL_REG, 0x00 },
{ TAS675X_OUTPUT_CTRL_REG, 0x00 },
{ TAS675X_STATE_CTRL_CH1_CH2_REG, TAS675X_STATE_SLEEP_BOTH },
{ TAS675X_STATE_CTRL_CH3_CH4_REG, TAS675X_STATE_SLEEP_BOTH },
{ TAS675X_ISENSE_CTRL_REG, 0x0F },
{ TAS675X_DC_DETECT_CTRL_REG, 0x00 },
{ TAS675X_SCLK_INV_CTRL_REG, 0x00 },
{ TAS675X_AUDIO_IF_CTRL_REG, 0x00 },
{ TAS675X_SDIN_CTRL_REG, 0x0A },
{ TAS675X_SDOUT_CTRL_REG, 0x1A },
{ TAS675X_SDIN_OFFSET_MSB_REG, 0x00 },
{ TAS675X_SDIN_AUDIO_OFFSET_REG, 0x00 },
{ TAS675X_SDIN_LL_OFFSET_REG, 0x60 },
{ TAS675X_SDIN_CH_SWAP_REG, 0x00 },
{ TAS675X_SDOUT_OFFSET_MSB_REG, 0xCF },
{ TAS675X_VPREDICT_OFFSET_REG, 0xFF },
{ TAS675X_ISENSE_OFFSET_REG, 0x00 },
{ TAS675X_SDOUT_EN_REG, 0x00 },
{ TAS675X_LL_EN_REG, 0x00 },
{ TAS675X_RTLDG_EN_REG, 0x10 },
{ TAS675X_DC_BLOCK_BYP_REG, 0x00 },
{ TAS675X_DSP_CTRL_REG, 0x00 },
{ TAS675X_PAGE_AUTO_INC_REG, 0x00 },
{ TAS675X_DIG_VOL_CH1_REG, 0x30 },
{ TAS675X_DIG_VOL_CH2_REG, 0x30 },
{ TAS675X_DIG_VOL_CH3_REG, 0x30 },
{ TAS675X_DIG_VOL_CH4_REG, 0x30 },
{ TAS675X_DIG_VOL_RAMP_CTRL_REG, 0x77 },
{ TAS675X_DIG_VOL_COMBINE_CTRL_REG, 0x00 },
{ TAS675X_AUTO_MUTE_EN_REG, 0x00 },
{ TAS675X_AUTO_MUTE_TIMING_CH1_CH2_REG, 0x00 },
{ TAS675X_AUTO_MUTE_TIMING_CH3_CH4_REG, 0x00 },
{ TAS675X_ANALOG_GAIN_CH1_CH2_REG, 0x00 },
{ TAS675X_ANALOG_GAIN_CH3_CH4_REG, 0x00 },
{ TAS675X_ANALOG_GAIN_RAMP_CTRL_REG, 0x00 },
{ TAS675X_PULSE_INJECTION_EN_REG, 0x03 },
{ TAS675X_CBC_CTRL_REG, 0x07 },
{ TAS675X_CURRENT_LIMIT_CTRL_REG, 0x00 },
{ TAS675X_ISENSE_CAL_REG, 0x00 },
{ TAS675X_PWM_PHASE_CTRL_REG, 0x00 },
{ TAS675X_SS_CTRL_REG, 0x00 },
{ TAS675X_SS_RANGE_CTRL_REG, 0x00 },
{ TAS675X_SS_DWELL_CTRL_REG, 0x00 },
{ TAS675X_RAMP_PHASE_CTRL_GPO_REG, 0x00 },
{ TAS675X_PWM_PHASE_M_CTRL_CH1_REG, 0x00 },
{ TAS675X_PWM_PHASE_M_CTRL_CH2_REG, 0x00 },
{ TAS675X_PWM_PHASE_M_CTRL_CH3_REG, 0x00 },
{ TAS675X_PWM_PHASE_M_CTRL_CH4_REG, 0x00 },
{ TAS675X_DC_LDG_CTRL_REG, 0x00 },
{ TAS675X_DC_LDG_LO_CTRL_REG, 0x00 },
{ TAS675X_DC_LDG_TIME_CTRL_REG, 0x00 },
{ TAS675X_DC_LDG_SL_CH1_CH2_CTRL_REG, 0x11 },
{ TAS675X_DC_LDG_SL_CH3_CH4_CTRL_REG, 0x11 },
{ TAS675X_AC_LDG_CTRL_REG, 0x10 },
{ TAS675X_TWEETER_DETECT_CTRL_REG, 0x08 },
{ TAS675X_TWEETER_DETECT_THRESH_REG, 0x00 },
{ TAS675X_AC_LDG_FREQ_CTRL_REG, 0xC8 },
{ TAS675X_REPORT_ROUTING_1_REG, 0x00 },
{ TAS675X_OTSD_RECOVERY_EN_REG, 0x00 },
{ TAS675X_REPORT_ROUTING_2_REG, 0xA2 },
{ TAS675X_REPORT_ROUTING_3_REG, 0x00 },
{ TAS675X_REPORT_ROUTING_4_REG, 0x06 },
{ TAS675X_CLIP_DETECT_CTRL_REG, 0x00 },
{ TAS675X_REPORT_ROUTING_5_REG, 0x00 },
{ TAS675X_GPIO1_OUTPUT_SEL_REG, 0x00 },
{ TAS675X_GPIO2_OUTPUT_SEL_REG, 0x00 },
{ TAS675X_GPIO_CTRL_REG, TAS675X_GPIO_CTRL_RSTVAL },
{ TAS675X_OTW_CTRL_CH1_CH2_REG, 0x11 },
{ TAS675X_OTW_CTRL_CH3_CH4_REG, 0x11 },
};
static bool tas675x_is_readable_register(struct device *dev, unsigned int reg)
{
switch (reg) {
case TAS675X_RESET_REG:
return false;
default:
return true;
}
}
static bool tas675x_is_volatile_register(struct device *dev, unsigned int reg)
{
switch (reg) {
case TAS675X_RESET_REG:
case TAS675X_BOOK_CTRL_REG:
case TAS675X_AUTO_MUTE_STATUS_REG:
case TAS675X_STATE_REPORT_CH1_CH2_REG:
case TAS675X_STATE_REPORT_CH3_CH4_REG:
case TAS675X_PVDD_SENSE_REG:
case TAS675X_TEMP_GLOBAL_REG:
case TAS675X_TEMP_CH1_CH2_REG:
case TAS675X_TEMP_CH3_CH4_REG:
case TAS675X_FS_MON_REG:
case TAS675X_SCLK_MON_REG:
case TAS675X_POWER_FAULT_STATUS_1_REG:
case TAS675X_POWER_FAULT_STATUS_2_REG:
case TAS675X_OT_FAULT_REG:
case TAS675X_OTW_STATUS_REG:
case TAS675X_CLIP_WARN_STATUS_REG:
case TAS675X_CBC_WARNING_STATUS_REG:
case TAS675X_POWER_FAULT_LATCHED_REG:
case TAS675X_OTSD_LATCHED_REG:
case TAS675X_OTW_LATCHED_REG:
case TAS675X_CLIP_WARN_LATCHED_REG:
case TAS675X_CLK_FAULT_LATCHED_REG:
case TAS675X_RTLDG_OL_SL_FAULT_LATCHED_REG:
case TAS675X_CBC_FAULT_WARN_LATCHED_REG:
case TAS675X_OC_DC_FAULT_LATCHED_REG:
case TAS675X_WARN_OT_MAX_FLAG_REG:
case TAS675X_DC_LDG_REPORT_CH1_CH2_REG ... TAS675X_TWEETER_REPORT_REG:
case TAS675X_CH1_RTLDG_IMP_MSB_REG ... TAS675X_CH4_DC_LDG_DCR_LSB_REG:
return true;
default:
return false;
}
}
static const struct regmap_range_cfg tas675x_ranges[] = {
{
.name = "Pages",
.range_min = 0,
.range_max = TAS675X_PAGE_SIZE * TAS675X_PAGE_SIZE - 1,
.selector_reg = TAS675X_PAGE_CTRL_REG,
.selector_mask = 0xff,
.selector_shift = 0,
.window_start = 0,
.window_len = TAS675X_PAGE_SIZE,
},
};
static void tas675x_regmap_lock(void *lock_arg)
{
struct tas675x_priv *tas = lock_arg;
mutex_lock(&tas->io_lock);
}
static void tas675x_regmap_unlock(void *lock_arg)
{
struct tas675x_priv *tas = lock_arg;
mutex_unlock(&tas->io_lock);
}
static const struct regmap_config tas675x_regmap_config = {
.reg_bits = 8,
.val_bits = 8,
.max_register = TAS675X_PAGE_SIZE * TAS675X_PAGE_SIZE - 1,
.ranges = tas675x_ranges,
.num_ranges = ARRAY_SIZE(tas675x_ranges),
.cache_type = REGCACHE_MAPLE,
.reg_defaults = tas675x_reg_defaults,
.num_reg_defaults = ARRAY_SIZE(tas675x_reg_defaults),
.readable_reg = tas675x_is_readable_register,
.volatile_reg = tas675x_is_volatile_register,
};
static int tas675x_i2c_probe(struct i2c_client *client)
{
struct regmap_config cfg = tas675x_regmap_config;
struct tas675x_priv *tas;
u32 val;
int i, ret;
tas = devm_kzalloc(&client->dev, sizeof(*tas), GFP_KERNEL);
if (!tas)
return -ENOMEM;
tas->dev = &client->dev;
i2c_set_clientdata(client, tas);
mutex_init(&tas->io_lock);
cfg.lock = tas675x_regmap_lock;
cfg.unlock = tas675x_regmap_unlock;
cfg.lock_arg = tas;
memcpy(tas->dsp_params, tas675x_dsp_defaults, sizeof(tas->dsp_params));
INIT_DELAYED_WORK(&tas->fault_check_work, tas675x_fault_check_work);
tas->regmap = devm_regmap_init_i2c(client, &cfg);
if (IS_ERR(tas->regmap))
return PTR_ERR(tas->regmap);
/* Keep regmap cache-only until hardware is powered on */
regcache_cache_only(tas->regmap, true);
tas->dev_type = (enum tas675x_type)(unsigned long)device_get_match_data(tas->dev);
tas->fast_boot = device_property_read_bool(tas->dev, "ti,fast-boot");
tas->audio_slot = -1;
tas->llp_slot = -1;
tas->vpredict_slot = -1;
tas->isense_slot = -1;
if (!device_property_read_u32(tas->dev, "ti,audio-slot-no", &val))
tas->audio_slot = val;
if (!device_property_read_u32(tas->dev, "ti,llp-slot-no", &val))
tas->llp_slot = val;
if (!device_property_read_u32(tas->dev, "ti,vpredict-slot-no", &val))
tas->vpredict_slot = val;
if (!device_property_read_u32(tas->dev, "ti,isense-slot-no", &val))
tas->isense_slot = val;
tas->gpio1_func = tas675x_gpio_func_parse(tas->dev, "ti,gpio1-function");
tas->gpio2_func = tas675x_gpio_func_parse(tas->dev, "ti,gpio2-function");
for (i = 0; i < ARRAY_SIZE(tas675x_supply_names); i++)
tas->supplies[i].supply = tas675x_supply_names[i];
ret = devm_regulator_bulk_get(tas->dev, ARRAY_SIZE(tas->supplies), tas->supplies);
if (ret)
return dev_err_probe(tas->dev, ret, "Failed to request supplies\n");
tas->vbat = devm_regulator_get_optional(tas->dev, "vbat");
if (IS_ERR(tas->vbat) && PTR_ERR(tas->vbat) != -ENODEV)
return dev_err_probe(tas->dev, PTR_ERR(tas->vbat),
"Failed to get vbat supply\n");
tas->pd_gpio = devm_gpiod_get_optional(tas->dev, "powerdown", GPIOD_OUT_HIGH);
if (IS_ERR(tas->pd_gpio))
return dev_err_probe(tas->dev, PTR_ERR(tas->pd_gpio), "Failed powerdown-gpios\n");
tas->stby_gpio = devm_gpiod_get_optional(tas->dev, "standby", GPIOD_OUT_HIGH);
if (IS_ERR(tas->stby_gpio))
return dev_err_probe(tas->dev, PTR_ERR(tas->stby_gpio), "Failed standby-gpios\n");
if (!tas->pd_gpio && !tas->stby_gpio)
return dev_err_probe(tas->dev, -EINVAL,
"At least one of powerdown-gpios or standby-gpios is required\n");
ret = tas675x_power_on(tas);
if (ret)
return ret;
if (client->irq) {
ret = devm_request_threaded_irq(tas->dev, client->irq, NULL,
tas675x_irq_handler,
IRQF_ONESHOT | IRQF_TRIGGER_FALLING,
"tas675x-fault", tas);
if (ret) {
tas675x_power_off(tas);
return dev_err_probe(tas->dev, ret, "Failed to request IRQ\n");
}
} else {
/* Schedule delayed work for fault checking at probe and runtime resume */
schedule_delayed_work(&tas->fault_check_work,
msecs_to_jiffies(TAS675X_FAULT_CHECK_INTERVAL_MS));
}
/* Enable runtime PM with 2s autosuspend */
pm_runtime_set_autosuspend_delay(tas->dev, 2000);
pm_runtime_use_autosuspend(tas->dev);
pm_runtime_set_active(tas->dev);
pm_runtime_mark_last_busy(tas->dev);
pm_runtime_enable(tas->dev);
ret = devm_snd_soc_register_component(tas->dev, &soc_codec_dev_tas675x,
tas675x_dais, ARRAY_SIZE(tas675x_dais));
if (ret)
goto err_pm_disable;
return 0;
err_pm_disable:
pm_runtime_force_suspend(tas->dev);
pm_runtime_disable(tas->dev);
tas675x_power_off(tas);
return ret;
}
static void tas675x_i2c_remove(struct i2c_client *client)
{
struct tas675x_priv *tas = dev_get_drvdata(&client->dev);
disable_delayed_work_sync(&tas->fault_check_work);
if (client->irq)
disable_irq(client->irq);
pm_runtime_force_suspend(&client->dev);
pm_runtime_disable(&client->dev);
tas675x_power_off(tas);
}
static const struct dev_pm_ops tas675x_pm_ops = {
SYSTEM_SLEEP_PM_OPS(tas675x_system_suspend, tas675x_system_resume)
RUNTIME_PM_OPS(tas675x_runtime_suspend, tas675x_runtime_resume, NULL)
};
static const struct of_device_id tas675x_of_match[] = {
{ .compatible = "ti,tas67524", .data = (void *)TAS67524 },
{ }
};
MODULE_DEVICE_TABLE(of, tas675x_of_match);
static const struct i2c_device_id tas675x_i2c_id[] = {
{ "tas67524", TAS67524 },
{ }
};
MODULE_DEVICE_TABLE(i2c, tas675x_i2c_id);
static struct i2c_driver tas675x_i2c_driver = {
.driver = {
.name = "tas675x",
.of_match_table = tas675x_of_match,
.pm = pm_ptr(&tas675x_pm_ops),
},
.probe = tas675x_i2c_probe,
.remove = tas675x_i2c_remove,
.id_table = tas675x_i2c_id,
};
module_i2c_driver(tas675x_i2c_driver);
MODULE_AUTHOR("Sen Wang <sen@ti.com>");
MODULE_DESCRIPTION("ASoC TAS675x Audio Amplifier Driver");
MODULE_LICENSE("GPL");