172 lines
4.0 KiB
C
172 lines
4.0 KiB
C
// SPDX-License-Identifier: (GPL-2.0+ OR MIT)
|
|
/*
|
|
* Copyright (c) 2023 Rockchip Electronics Co., Ltd.
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/i2c.h>
|
|
#include <linux/init.h>
|
|
#include <linux/regmap.h>
|
|
#include <sound/core.h>
|
|
#include <sound/control.h>
|
|
#include <sound/soc.h>
|
|
|
|
#include "tda7803.h"
|
|
|
|
#define TDA7803_SAMPLE_RATE \
|
|
(SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000 | \
|
|
SNDRV_PCM_RATE_96000 | SNDRV_PCM_RATE_192000)
|
|
|
|
struct tda7803_priv {
|
|
struct regmap *regmap;
|
|
u32 input_format;
|
|
};
|
|
|
|
static int tda7803_startup(struct snd_pcm_substream *substream,
|
|
struct snd_soc_dai *dai)
|
|
{
|
|
struct snd_soc_component *component = dai->component;
|
|
struct tda7803_priv *tda7803 = snd_soc_component_get_drvdata(component);
|
|
struct snd_soc_pcm_runtime *rtd = asoc_substream_to_rtd(substream);
|
|
struct snd_soc_dai *codec_dai = asoc_rtd_to_codec(rtd, 0);
|
|
int val = 0;
|
|
|
|
snd_soc_component_write(component, TDA7803_REG2, DIGITAL_MUTE_OFF |
|
|
CH2_4_UMUTE | CH1_3_UMUTE |
|
|
MUTE_TIME_SETTING_1_45MS);
|
|
snd_soc_component_write(component, TDA7803_REG7, AMPLIEFIR_SWITCH_ON);
|
|
|
|
switch (tda7803->input_format) {
|
|
case 0:
|
|
val = INPUT_FORMAT_TDM_8CH_MODEL1;
|
|
break;
|
|
case 1:
|
|
val = INPUT_FORMAT_TDM_8CH_MODEL2;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
snd_soc_dai_set_fmt(codec_dai, val);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tda7803_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 val = 0;
|
|
|
|
switch (params_rate(params)) {
|
|
case 44100:
|
|
val = SAMPLE_FREQUENCY_RANGE_44100HZ;
|
|
break;
|
|
case 48000:
|
|
val = SAMPLE_FREQUENCY_RANGE_48000HZ;
|
|
break;
|
|
case 96000:
|
|
val = SAMPLE_FREQUENCY_RANGE_96000HZ;
|
|
break;
|
|
case 192000:
|
|
val = SAMPLE_FREQUENCY_RANGE_192000HZ;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
snd_soc_component_write(component, TDA7803_REG3, val);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tda7803_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
|
|
{
|
|
struct snd_soc_component *component = dai->component;
|
|
|
|
snd_soc_component_write(component, TDA7803_REG3, fmt);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct snd_soc_dai_ops tda7803_ops = {
|
|
.startup = tda7803_startup,
|
|
.hw_params = tda7803_hw_params,
|
|
.set_fmt = tda7803_set_fmt,
|
|
};
|
|
|
|
static struct snd_soc_dai_driver tda7803_dai = {
|
|
.name = "tda7803-hifi",
|
|
.playback = {
|
|
.stream_name = "Playback",
|
|
.channels_min = 2,
|
|
.channels_max = 8,
|
|
.rates = TDA7803_SAMPLE_RATE,
|
|
.formats = SNDRV_PCM_FMTBIT_S32_LE |
|
|
SNDRV_PCM_FMTBIT_S24_LE |
|
|
SNDRV_PCM_FMTBIT_S16_LE,
|
|
},
|
|
.ops = &tda7803_ops,
|
|
};
|
|
|
|
static const struct snd_soc_component_driver soc_codec_dev_tda7803 = {
|
|
.name = "tda7803",
|
|
};
|
|
|
|
static const struct regmap_config tda7803_i2c_regmap = {
|
|
.reg_bits = 8,
|
|
.val_bits = 8,
|
|
.max_register = TDA7803_REGMAX,
|
|
.cache_type = REGCACHE_RBTREE,
|
|
};
|
|
|
|
static int tda7803_i2c_probe(struct i2c_client *i2c,
|
|
const struct i2c_device_id *id)
|
|
{
|
|
struct tda7803_priv *tda7803;
|
|
int val;
|
|
|
|
tda7803 = devm_kzalloc(&i2c->dev, sizeof(*tda7803), GFP_KERNEL);
|
|
if (!tda7803)
|
|
return -ENOMEM;
|
|
|
|
i2c_set_clientdata(i2c, tda7803);
|
|
|
|
tda7803->regmap = devm_regmap_init_i2c(i2c, &tda7803_i2c_regmap);
|
|
if (IS_ERR(tda7803->regmap))
|
|
return PTR_ERR(tda7803->regmap);
|
|
|
|
if (!device_property_read_u32(&i2c->dev, "st,tda7803-format", &val))
|
|
tda7803->input_format = val;
|
|
|
|
return devm_snd_soc_register_component(&i2c->dev,
|
|
&soc_codec_dev_tda7803,
|
|
&tda7803_dai, 1);
|
|
}
|
|
|
|
static const struct i2c_device_id tda7803_i2c_id[] = {
|
|
{ "tda7803", 0 },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(i2c, tda7803_i2c_id);
|
|
|
|
static const struct of_device_id tda7803_of_match[] = {
|
|
{ .compatible = "st,tda7803" },
|
|
{ },
|
|
};
|
|
|
|
static struct i2c_driver tda7803_i2c_driver = {
|
|
.driver = {
|
|
.name = "tda7803",
|
|
.of_match_table = of_match_ptr(tda7803_of_match),
|
|
},
|
|
.probe = tda7803_i2c_probe,
|
|
.id_table = tda7803_i2c_id,
|
|
};
|
|
module_i2c_driver(tda7803_i2c_driver);
|
|
|
|
MODULE_AUTHOR("Jun Zeng <jun.zeng@rock-chips.com>");
|
|
MODULE_DESCRIPTION("TDA7803 audio processor driver");
|
|
MODULE_LICENSE("GPL");
|