e069642059
A bunch of small fixes that come in during the merge window, mainly fixing issues from some core refactoring around dummy components that weren't detected until things reached mainline. The TAS driver changes are a little larger than normal for a device ID addition due to some shuffling around of where things are registered and DT updates but aren't really any more substantial than normal. -----BEGIN PGP SIGNATURE----- iQEzBAABCgAdFiEEreZoqmdXGLWf4p/qJNaLcl1Uh9AFAmWmnFEACgkQJNaLcl1U h9Dzvgf/RfM1g44TeC/hSX0+qdrL6QYg5wPeMWPtV4xgLqx2oqftU9eIcBLHPz03 N1hdutFFqNTR3W5GiTmyV5mLdgJpZBgkYU5dkhXCrlI7xJ6tV7Wsx86Q9ClSmEBH Z4ZR2oAmCQIt3swshOBvwYUI6MOcG/zSqDBcRQ6t6d98TOkp2m3PPrjEYqEA2ja9 tyr73WXsdammYXZVYPu9oYoNE2Ef6Io5nkyds1F4irCcx1EXiS95cZjF3/OZ9OPP iGpzVkXvgwb0Jbq+NsBogydsIsSAu4pyNNcu3P9Xcr1rkNqVwz1X1KYvvnSHDqdw ZbmU6r83dCUpizpgFA4wv8irNn+aKw== =Rr9L -----END PGP SIGNATURE----- Merge tag 'asoc-fix-v6.8-merge-window' of https://git.kernel.org/pub/scm/linux/kernel/git/broonie/sound into for-linus ASoC: Fixes for v6.8 A bunch of small fixes that come in during the merge window, mainly fixing issues from some core refactoring around dummy components that weren't detected until things reached mainline. The TAS driver changes are a little larger than normal for a device ID addition due to some shuffling around of where things are registered and DT updates but aren't really any more substantial than normal.
767 lines
21 KiB
C
767 lines
21 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
//
|
|
// ALSA SoC Texas Instruments TAS2563/TAS2781 Audio Smart Amplifier
|
|
//
|
|
// Copyright (C) 2022 - 2023 Texas Instruments Incorporated
|
|
// https://www.ti.com
|
|
//
|
|
// The TAS2563/TAS2781 driver implements a flexible and configurable
|
|
// algo coefficient setting for one, two, or even multiple
|
|
// TAS2563/TAS2781 chips.
|
|
//
|
|
// Author: Shenghao Ding <shenghao-ding@ti.com>
|
|
// Author: Kevin Lu <kevin-lu@ti.com>
|
|
//
|
|
|
|
#include <linux/crc8.h>
|
|
#include <linux/firmware.h>
|
|
#include <linux/gpio/consumer.h>
|
|
#include <linux/i2c.h>
|
|
#include <linux/init.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/module.h>
|
|
#include <linux/of.h>
|
|
#include <linux/of_gpio.h>
|
|
#include <linux/of_irq.h>
|
|
#include <linux/regmap.h>
|
|
#include <linux/slab.h>
|
|
#include <sound/pcm_params.h>
|
|
#include <sound/soc.h>
|
|
#include <sound/tas2781.h>
|
|
#include <sound/tlv.h>
|
|
#include <sound/tas2781-tlv.h>
|
|
|
|
static const struct i2c_device_id tasdevice_id[] = {
|
|
{ "tas2563", TAS2563 },
|
|
{ "tas2781", TAS2781 },
|
|
{}
|
|
};
|
|
MODULE_DEVICE_TABLE(i2c, tasdevice_id);
|
|
|
|
#ifdef CONFIG_OF
|
|
static const struct of_device_id tasdevice_of_match[] = {
|
|
{ .compatible = "ti,tas2563" },
|
|
{ .compatible = "ti,tas2781" },
|
|
{},
|
|
};
|
|
MODULE_DEVICE_TABLE(of, tasdevice_of_match);
|
|
#endif
|
|
|
|
/**
|
|
* tas2781_digital_getvol - get the volum control
|
|
* @kcontrol: control pointer
|
|
* @ucontrol: User data
|
|
* Customer Kcontrol for tas2781 is primarily for regmap booking, paging
|
|
* depends on internal regmap mechanism.
|
|
* tas2781 contains book and page two-level register map, especially
|
|
* book switching will set the register BXXP00R7F, after switching to the
|
|
* correct book, then leverage the mechanism for paging to access the
|
|
* register.
|
|
*/
|
|
static int tas2781_digital_getvol(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
struct soc_mixer_control *mc =
|
|
(struct soc_mixer_control *)kcontrol->private_value;
|
|
|
|
return tasdevice_digital_getvol(tas_priv, ucontrol, mc);
|
|
}
|
|
|
|
static int tas2781_digital_putvol(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
struct soc_mixer_control *mc =
|
|
(struct soc_mixer_control *)kcontrol->private_value;
|
|
|
|
return tasdevice_digital_putvol(tas_priv, ucontrol, mc);
|
|
}
|
|
|
|
static int tas2781_amp_getvol(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
struct soc_mixer_control *mc =
|
|
(struct soc_mixer_control *)kcontrol->private_value;
|
|
|
|
return tasdevice_amp_getvol(tas_priv, ucontrol, mc);
|
|
}
|
|
|
|
static int tas2781_amp_putvol(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv =
|
|
snd_soc_component_get_drvdata(codec);
|
|
struct soc_mixer_control *mc =
|
|
(struct soc_mixer_control *)kcontrol->private_value;
|
|
|
|
return tasdevice_amp_putvol(tas_priv, ucontrol, mc);
|
|
}
|
|
|
|
static int tas2781_force_fwload_get(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *component =
|
|
snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv =
|
|
snd_soc_component_get_drvdata(component);
|
|
|
|
ucontrol->value.integer.value[0] = (int)tas_priv->force_fwload_status;
|
|
dev_dbg(tas_priv->dev, "%s : Force FWload %s\n", __func__,
|
|
tas_priv->force_fwload_status ? "ON" : "OFF");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tas2781_force_fwload_put(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *component =
|
|
snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv =
|
|
snd_soc_component_get_drvdata(component);
|
|
bool change, val = (bool)ucontrol->value.integer.value[0];
|
|
|
|
if (tas_priv->force_fwload_status == val)
|
|
change = false;
|
|
else {
|
|
change = true;
|
|
tas_priv->force_fwload_status = val;
|
|
}
|
|
dev_dbg(tas_priv->dev, "%s : Force FWload %s\n", __func__,
|
|
tas_priv->force_fwload_status ? "ON" : "OFF");
|
|
|
|
return change;
|
|
}
|
|
|
|
static const struct snd_kcontrol_new tas2781_snd_controls[] = {
|
|
SOC_SINGLE_RANGE_EXT_TLV("Speaker Analog Gain", TAS2781_AMP_LEVEL,
|
|
1, 0, 20, 0, tas2781_amp_getvol,
|
|
tas2781_amp_putvol, amp_vol_tlv),
|
|
SOC_SINGLE_RANGE_EXT_TLV("Speaker Digital Gain", TAS2781_DVC_LVL,
|
|
0, 0, 200, 1, tas2781_digital_getvol,
|
|
tas2781_digital_putvol, dvc_tlv),
|
|
SOC_SINGLE_BOOL_EXT("Speaker Force Firmware Load", 0,
|
|
tas2781_force_fwload_get, tas2781_force_fwload_put),
|
|
};
|
|
|
|
static int tasdevice_set_profile_id(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
int ret = 0;
|
|
|
|
if (tas_priv->rcabin.profile_cfg_id !=
|
|
ucontrol->value.integer.value[0]) {
|
|
tas_priv->rcabin.profile_cfg_id =
|
|
ucontrol->value.integer.value[0];
|
|
ret = 1;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int tasdevice_info_programs(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_info *uinfo)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
struct tasdevice_fw *tas_fw = tas_priv->fmw;
|
|
|
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
|
|
uinfo->count = 1;
|
|
uinfo->value.integer.min = 0;
|
|
uinfo->value.integer.max = (int)tas_fw->nr_programs;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tasdevice_info_configurations(
|
|
struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
|
|
{
|
|
struct snd_soc_component *codec =
|
|
snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
struct tasdevice_fw *tas_fw = tas_priv->fmw;
|
|
|
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
|
|
uinfo->count = 1;
|
|
uinfo->value.integer.min = 0;
|
|
uinfo->value.integer.max = (int)tas_fw->nr_configurations - 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tasdevice_info_profile(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_info *uinfo)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
|
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
|
|
uinfo->count = 1;
|
|
uinfo->value.integer.min = 0;
|
|
uinfo->value.integer.max = tas_priv->rcabin.ncfgs - 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tasdevice_get_profile_id(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
|
|
ucontrol->value.integer.value[0] = tas_priv->rcabin.profile_cfg_id;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tasdevice_create_control(struct tasdevice_priv *tas_priv)
|
|
{
|
|
struct snd_kcontrol_new *prof_ctrls;
|
|
int nr_controls = 1;
|
|
int mix_index = 0;
|
|
int ret;
|
|
char *name;
|
|
|
|
prof_ctrls = devm_kcalloc(tas_priv->dev, nr_controls,
|
|
sizeof(prof_ctrls[0]), GFP_KERNEL);
|
|
if (!prof_ctrls) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
/* Create a mixer item for selecting the active profile */
|
|
name = devm_kzalloc(tas_priv->dev, SNDRV_CTL_ELEM_ID_NAME_MAXLEN,
|
|
GFP_KERNEL);
|
|
if (!name) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
scnprintf(name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN, "Speaker Profile Id");
|
|
prof_ctrls[mix_index].name = name;
|
|
prof_ctrls[mix_index].iface = SNDRV_CTL_ELEM_IFACE_MIXER;
|
|
prof_ctrls[mix_index].info = tasdevice_info_profile;
|
|
prof_ctrls[mix_index].get = tasdevice_get_profile_id;
|
|
prof_ctrls[mix_index].put = tasdevice_set_profile_id;
|
|
mix_index++;
|
|
|
|
ret = snd_soc_add_component_controls(tas_priv->codec,
|
|
prof_ctrls, nr_controls < mix_index ? nr_controls : mix_index);
|
|
|
|
out:
|
|
return ret;
|
|
}
|
|
|
|
static int tasdevice_program_get(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
|
|
ucontrol->value.integer.value[0] = tas_priv->cur_prog;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tasdevice_program_put(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
unsigned int nr_program = ucontrol->value.integer.value[0];
|
|
int ret = 0;
|
|
|
|
if (tas_priv->cur_prog != nr_program) {
|
|
tas_priv->cur_prog = nr_program;
|
|
ret = 1;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int tasdevice_configuration_get(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
|
|
ucontrol->value.integer.value[0] = tas_priv->cur_conf;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tasdevice_configuration_put(
|
|
struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_kcontrol_component(kcontrol);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
unsigned int nr_configuration = ucontrol->value.integer.value[0];
|
|
int ret = 0;
|
|
|
|
if (tas_priv->cur_conf != nr_configuration) {
|
|
tas_priv->cur_conf = nr_configuration;
|
|
ret = 1;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int tasdevice_dsp_create_ctrls(
|
|
struct tasdevice_priv *tas_priv)
|
|
{
|
|
struct snd_kcontrol_new *dsp_ctrls;
|
|
char *prog_name, *conf_name;
|
|
int nr_controls = 2;
|
|
int mix_index = 0;
|
|
int ret;
|
|
|
|
/* Alloc kcontrol via devm_kzalloc, which don't manually
|
|
* free the kcontrol
|
|
*/
|
|
dsp_ctrls = devm_kcalloc(tas_priv->dev, nr_controls,
|
|
sizeof(dsp_ctrls[0]), GFP_KERNEL);
|
|
if (!dsp_ctrls) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
/* Create a mixer item for selecting the active profile */
|
|
prog_name = devm_kzalloc(tas_priv->dev,
|
|
SNDRV_CTL_ELEM_ID_NAME_MAXLEN, GFP_KERNEL);
|
|
conf_name = devm_kzalloc(tas_priv->dev, SNDRV_CTL_ELEM_ID_NAME_MAXLEN,
|
|
GFP_KERNEL);
|
|
if (!prog_name || !conf_name) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
scnprintf(prog_name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN,
|
|
"Speaker Program Id");
|
|
dsp_ctrls[mix_index].name = prog_name;
|
|
dsp_ctrls[mix_index].iface = SNDRV_CTL_ELEM_IFACE_MIXER;
|
|
dsp_ctrls[mix_index].info = tasdevice_info_programs;
|
|
dsp_ctrls[mix_index].get = tasdevice_program_get;
|
|
dsp_ctrls[mix_index].put = tasdevice_program_put;
|
|
mix_index++;
|
|
|
|
scnprintf(conf_name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN,
|
|
"Speaker Config Id");
|
|
dsp_ctrls[mix_index].name = conf_name;
|
|
dsp_ctrls[mix_index].iface = SNDRV_CTL_ELEM_IFACE_MIXER;
|
|
dsp_ctrls[mix_index].info = tasdevice_info_configurations;
|
|
dsp_ctrls[mix_index].get = tasdevice_configuration_get;
|
|
dsp_ctrls[mix_index].put = tasdevice_configuration_put;
|
|
mix_index++;
|
|
|
|
ret = snd_soc_add_component_controls(tas_priv->codec, dsp_ctrls,
|
|
nr_controls < mix_index ? nr_controls : mix_index);
|
|
|
|
out:
|
|
return ret;
|
|
}
|
|
|
|
static void tasdevice_fw_ready(const struct firmware *fmw,
|
|
void *context)
|
|
{
|
|
struct tasdevice_priv *tas_priv = context;
|
|
int ret = 0;
|
|
int i;
|
|
|
|
mutex_lock(&tas_priv->codec_lock);
|
|
|
|
ret = tasdevice_rca_parser(tas_priv, fmw);
|
|
if (ret)
|
|
goto out;
|
|
tasdevice_create_control(tas_priv);
|
|
|
|
tasdevice_dsp_remove(tas_priv);
|
|
tasdevice_calbin_remove(tas_priv);
|
|
tas_priv->fw_state = TASDEVICE_DSP_FW_PENDING;
|
|
scnprintf(tas_priv->coef_binaryname, 64, "%s_coef.bin",
|
|
tas_priv->dev_name);
|
|
ret = tasdevice_dsp_parser(tas_priv);
|
|
if (ret) {
|
|
dev_err(tas_priv->dev, "dspfw load %s error\n",
|
|
tas_priv->coef_binaryname);
|
|
tas_priv->fw_state = TASDEVICE_DSP_FW_FAIL;
|
|
goto out;
|
|
}
|
|
tasdevice_dsp_create_ctrls(tas_priv);
|
|
|
|
tas_priv->fw_state = TASDEVICE_DSP_FW_ALL_OK;
|
|
|
|
/* If calibrated data occurs error, dsp will still works with default
|
|
* calibrated data inside algo.
|
|
*/
|
|
for (i = 0; i < tas_priv->ndev; i++) {
|
|
scnprintf(tas_priv->cal_binaryname[i], 64, "%s_cal_0x%02x.bin",
|
|
tas_priv->dev_name, tas_priv->tasdevice[i].dev_addr);
|
|
ret = tas2781_load_calibration(tas_priv,
|
|
tas_priv->cal_binaryname[i], i);
|
|
if (ret != 0)
|
|
dev_err(tas_priv->dev,
|
|
"%s: load %s error, default will effect\n",
|
|
__func__, tas_priv->cal_binaryname[i]);
|
|
}
|
|
|
|
tasdevice_prmg_calibdata_load(tas_priv, 0);
|
|
tas_priv->cur_prog = 0;
|
|
out:
|
|
if (tas_priv->fw_state == TASDEVICE_DSP_FW_FAIL) {
|
|
/*If DSP FW fail, kcontrol won't be created */
|
|
tasdevice_config_info_remove(tas_priv);
|
|
tasdevice_dsp_remove(tas_priv);
|
|
}
|
|
mutex_unlock(&tas_priv->codec_lock);
|
|
if (fmw)
|
|
release_firmware(fmw);
|
|
}
|
|
|
|
static int tasdevice_dapm_event(struct snd_soc_dapm_widget *w,
|
|
struct snd_kcontrol *kcontrol, int event)
|
|
{
|
|
struct snd_soc_component *codec = snd_soc_dapm_to_component(w->dapm);
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
int state = 0;
|
|
|
|
/* Codec Lock Hold */
|
|
mutex_lock(&tas_priv->codec_lock);
|
|
if (event == SND_SOC_DAPM_PRE_PMD)
|
|
state = 1;
|
|
tasdevice_tuning_switch(tas_priv, state);
|
|
/* Codec Lock Release*/
|
|
mutex_unlock(&tas_priv->codec_lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct snd_soc_dapm_widget tasdevice_dapm_widgets[] = {
|
|
SND_SOC_DAPM_AIF_IN("ASI", "ASI Playback", 0, SND_SOC_NOPM, 0, 0),
|
|
SND_SOC_DAPM_AIF_OUT_E("ASI OUT", "ASI Capture", 0, SND_SOC_NOPM,
|
|
0, 0, tasdevice_dapm_event,
|
|
SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
|
|
SND_SOC_DAPM_SPK("SPK", tasdevice_dapm_event),
|
|
SND_SOC_DAPM_OUTPUT("OUT"),
|
|
SND_SOC_DAPM_INPUT("DMIC")
|
|
};
|
|
|
|
static const struct snd_soc_dapm_route tasdevice_audio_map[] = {
|
|
{"SPK", NULL, "ASI"},
|
|
{"OUT", NULL, "SPK"},
|
|
{"ASI OUT", NULL, "DMIC"}
|
|
};
|
|
|
|
static int tasdevice_startup(struct snd_pcm_substream *substream,
|
|
struct snd_soc_dai *dai)
|
|
{
|
|
struct snd_soc_component *codec = dai->component;
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
int ret = 0;
|
|
|
|
if (tas_priv->fw_state != TASDEVICE_DSP_FW_ALL_OK) {
|
|
dev_err(tas_priv->dev, "DSP bin file not loaded\n");
|
|
ret = -EINVAL;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int tasdevice_hw_params(struct snd_pcm_substream *substream,
|
|
struct snd_pcm_hw_params *params, struct snd_soc_dai *dai)
|
|
{
|
|
struct tasdevice_priv *tas_priv = snd_soc_dai_get_drvdata(dai);
|
|
unsigned int slot_width;
|
|
unsigned int fsrate;
|
|
int bclk_rate;
|
|
int rc = 0;
|
|
|
|
fsrate = params_rate(params);
|
|
switch (fsrate) {
|
|
case 48000:
|
|
case 44100:
|
|
break;
|
|
default:
|
|
dev_err(tas_priv->dev, "%s: incorrect sample rate = %u\n",
|
|
__func__, fsrate);
|
|
rc = -EINVAL;
|
|
goto out;
|
|
}
|
|
|
|
slot_width = params_width(params);
|
|
switch (slot_width) {
|
|
case 16:
|
|
case 20:
|
|
case 24:
|
|
case 32:
|
|
break;
|
|
default:
|
|
dev_err(tas_priv->dev, "%s: incorrect slot width = %u\n",
|
|
__func__, slot_width);
|
|
rc = -EINVAL;
|
|
goto out;
|
|
}
|
|
|
|
bclk_rate = snd_soc_params_to_bclk(params);
|
|
if (bclk_rate < 0) {
|
|
dev_err(tas_priv->dev, "%s: incorrect bclk rate = %d\n",
|
|
__func__, bclk_rate);
|
|
rc = bclk_rate;
|
|
goto out;
|
|
}
|
|
|
|
out:
|
|
return rc;
|
|
}
|
|
|
|
static int tasdevice_set_dai_sysclk(struct snd_soc_dai *codec_dai,
|
|
int clk_id, unsigned int freq, int dir)
|
|
{
|
|
struct tasdevice_priv *tas_priv = snd_soc_dai_get_drvdata(codec_dai);
|
|
|
|
tas_priv->sysclk = freq;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct snd_soc_dai_ops tasdevice_dai_ops = {
|
|
.startup = tasdevice_startup,
|
|
.hw_params = tasdevice_hw_params,
|
|
.set_sysclk = tasdevice_set_dai_sysclk,
|
|
};
|
|
|
|
static struct snd_soc_dai_driver tasdevice_dai_driver[] = {
|
|
{
|
|
.name = "tas2781_codec",
|
|
.id = 0,
|
|
.playback = {
|
|
.stream_name = "Playback",
|
|
.channels_min = 1,
|
|
.channels_max = 4,
|
|
.rates = TASDEVICE_RATES,
|
|
.formats = TASDEVICE_FORMATS,
|
|
},
|
|
.capture = {
|
|
.stream_name = "Capture",
|
|
.channels_min = 1,
|
|
.channels_max = 4,
|
|
.rates = TASDEVICE_RATES,
|
|
.formats = TASDEVICE_FORMATS,
|
|
},
|
|
.ops = &tasdevice_dai_ops,
|
|
.symmetric_rate = 1,
|
|
},
|
|
};
|
|
|
|
static int tasdevice_codec_probe(struct snd_soc_component *codec)
|
|
{
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
|
|
return tascodec_init(tas_priv, codec, tasdevice_fw_ready);
|
|
}
|
|
|
|
static void tasdevice_deinit(void *context)
|
|
{
|
|
struct tasdevice_priv *tas_priv = (struct tasdevice_priv *) context;
|
|
|
|
tasdevice_config_info_remove(tas_priv);
|
|
tasdevice_dsp_remove(tas_priv);
|
|
tasdevice_calbin_remove(tas_priv);
|
|
tas_priv->fw_state = TASDEVICE_DSP_FW_PENDING;
|
|
}
|
|
|
|
static void tasdevice_codec_remove(
|
|
struct snd_soc_component *codec)
|
|
{
|
|
struct tasdevice_priv *tas_priv = snd_soc_component_get_drvdata(codec);
|
|
|
|
tasdevice_deinit(tas_priv);
|
|
}
|
|
|
|
static const struct snd_soc_component_driver
|
|
soc_codec_driver_tasdevice = {
|
|
.probe = tasdevice_codec_probe,
|
|
.remove = tasdevice_codec_remove,
|
|
.controls = tas2781_snd_controls,
|
|
.num_controls = ARRAY_SIZE(tas2781_snd_controls),
|
|
.dapm_widgets = tasdevice_dapm_widgets,
|
|
.num_dapm_widgets = ARRAY_SIZE(tasdevice_dapm_widgets),
|
|
.dapm_routes = tasdevice_audio_map,
|
|
.num_dapm_routes = ARRAY_SIZE(tasdevice_audio_map),
|
|
.idle_bias_on = 1,
|
|
.endianness = 1,
|
|
};
|
|
|
|
static void tasdevice_parse_dt(struct tasdevice_priv *tas_priv)
|
|
{
|
|
struct i2c_client *client = (struct i2c_client *)tas_priv->client;
|
|
unsigned int dev_addrs[TASDEVICE_MAX_CHANNELS];
|
|
int rc, i, ndev = 0;
|
|
|
|
if (tas_priv->isacpi) {
|
|
ndev = device_property_read_u32_array(&client->dev,
|
|
"ti,audio-slots", NULL, 0);
|
|
if (ndev <= 0) {
|
|
ndev = 1;
|
|
dev_addrs[0] = client->addr;
|
|
} else {
|
|
ndev = (ndev < ARRAY_SIZE(dev_addrs))
|
|
? ndev : ARRAY_SIZE(dev_addrs);
|
|
ndev = device_property_read_u32_array(&client->dev,
|
|
"ti,audio-slots", dev_addrs, ndev);
|
|
}
|
|
|
|
tas_priv->irq_info.irq_gpio =
|
|
acpi_dev_gpio_irq_get(ACPI_COMPANION(&client->dev), 0);
|
|
} else {
|
|
struct device_node *np = tas_priv->dev->of_node;
|
|
#ifdef CONFIG_OF
|
|
const __be32 *reg, *reg_end;
|
|
int len, sw, aw;
|
|
|
|
aw = of_n_addr_cells(np);
|
|
sw = of_n_size_cells(np);
|
|
if (sw == 0) {
|
|
reg = (const __be32 *)of_get_property(np,
|
|
"reg", &len);
|
|
reg_end = reg + len/sizeof(*reg);
|
|
ndev = 0;
|
|
do {
|
|
dev_addrs[ndev] = of_read_number(reg, aw);
|
|
reg += aw;
|
|
ndev++;
|
|
} while (reg < reg_end);
|
|
} else {
|
|
ndev = 1;
|
|
dev_addrs[0] = client->addr;
|
|
}
|
|
#else
|
|
ndev = 1;
|
|
dev_addrs[0] = client->addr;
|
|
#endif
|
|
tas_priv->irq_info.irq_gpio = of_irq_get(np, 0);
|
|
}
|
|
tas_priv->ndev = ndev;
|
|
for (i = 0; i < ndev; i++)
|
|
tas_priv->tasdevice[i].dev_addr = dev_addrs[i];
|
|
|
|
tas_priv->reset = devm_gpiod_get_optional(&client->dev,
|
|
"reset-gpios", GPIOD_OUT_HIGH);
|
|
if (IS_ERR(tas_priv->reset))
|
|
dev_err(tas_priv->dev, "%s Can't get reset GPIO\n",
|
|
__func__);
|
|
|
|
strcpy(tas_priv->dev_name, tasdevice_id[tas_priv->chip_id].name);
|
|
|
|
if (gpio_is_valid(tas_priv->irq_info.irq_gpio)) {
|
|
rc = gpio_request(tas_priv->irq_info.irq_gpio,
|
|
"AUDEV-IRQ");
|
|
if (!rc) {
|
|
gpio_direction_input(
|
|
tas_priv->irq_info.irq_gpio);
|
|
|
|
tas_priv->irq_info.irq =
|
|
gpio_to_irq(tas_priv->irq_info.irq_gpio);
|
|
} else
|
|
dev_err(tas_priv->dev, "%s: GPIO %d request error\n",
|
|
__func__, tas_priv->irq_info.irq_gpio);
|
|
} else
|
|
dev_err(tas_priv->dev,
|
|
"Looking up irq-gpio property failed %d\n",
|
|
tas_priv->irq_info.irq_gpio);
|
|
}
|
|
|
|
static int tasdevice_i2c_probe(struct i2c_client *i2c)
|
|
{
|
|
const struct i2c_device_id *id = i2c_match_id(tasdevice_id, i2c);
|
|
const struct acpi_device_id *acpi_id;
|
|
struct tasdevice_priv *tas_priv;
|
|
int ret;
|
|
|
|
tas_priv = tasdevice_kzalloc(i2c);
|
|
if (!tas_priv)
|
|
return -ENOMEM;
|
|
|
|
dev_set_drvdata(&i2c->dev, tas_priv);
|
|
|
|
if (ACPI_HANDLE(&i2c->dev)) {
|
|
acpi_id = acpi_match_device(i2c->dev.driver->acpi_match_table,
|
|
&i2c->dev);
|
|
if (!acpi_id) {
|
|
dev_err(&i2c->dev, "No driver data\n");
|
|
ret = -EINVAL;
|
|
goto err;
|
|
}
|
|
tas_priv->chip_id = acpi_id->driver_data;
|
|
tas_priv->isacpi = true;
|
|
} else {
|
|
tas_priv->chip_id = id ? id->driver_data : 0;
|
|
tas_priv->isacpi = false;
|
|
}
|
|
|
|
tasdevice_parse_dt(tas_priv);
|
|
|
|
ret = tasdevice_init(tas_priv);
|
|
if (ret)
|
|
goto err;
|
|
|
|
ret = devm_snd_soc_register_component(tas_priv->dev,
|
|
&soc_codec_driver_tasdevice,
|
|
tasdevice_dai_driver, ARRAY_SIZE(tasdevice_dai_driver));
|
|
if (ret) {
|
|
dev_err(tas_priv->dev, "%s: codec register error:0x%08x\n",
|
|
__func__, ret);
|
|
goto err;
|
|
}
|
|
err:
|
|
if (ret < 0)
|
|
tasdevice_remove(tas_priv);
|
|
return ret;
|
|
}
|
|
|
|
static void tasdevice_i2c_remove(struct i2c_client *client)
|
|
{
|
|
struct tasdevice_priv *tas_priv = i2c_get_clientdata(client);
|
|
|
|
tasdevice_remove(tas_priv);
|
|
}
|
|
|
|
#ifdef CONFIG_ACPI
|
|
static const struct acpi_device_id tasdevice_acpi_match[] = {
|
|
{ "TAS2781", TAS2781 },
|
|
{},
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(acpi, tasdevice_acpi_match);
|
|
#endif
|
|
|
|
static struct i2c_driver tasdevice_i2c_driver = {
|
|
.driver = {
|
|
.name = "tas2781-codec",
|
|
.of_match_table = of_match_ptr(tasdevice_of_match),
|
|
#ifdef CONFIG_ACPI
|
|
.acpi_match_table = ACPI_PTR(tasdevice_acpi_match),
|
|
#endif
|
|
},
|
|
.probe = tasdevice_i2c_probe,
|
|
.remove = tasdevice_i2c_remove,
|
|
.id_table = tasdevice_id,
|
|
};
|
|
|
|
module_i2c_driver(tasdevice_i2c_driver);
|
|
|
|
MODULE_AUTHOR("Shenghao Ding <shenghao-ding@ti.com>");
|
|
MODULE_AUTHOR("Kevin Lu <kevin-lu@ti.com>");
|
|
MODULE_DESCRIPTION("ASoC TAS2781 Driver");
|
|
MODULE_LICENSE("GPL");
|
|
MODULE_IMPORT_NS(SND_SOC_TAS2781_FMWLIB);
|