Commit 45749c91 authored by John Keeping's avatar John Keeping Committed by Mark Brown

ASoC: es8328: Support more sample rates

Signed-off-by: default avatarJohn Keeping <john@metanate.com>
Signed-off-by: default avatarMark Brown <broonie@kernel.org>
parent 779e86a3
...@@ -26,18 +26,30 @@ ...@@ -26,18 +26,30 @@
#include <sound/tlv.h> #include <sound/tlv.h>
#include "es8328.h" #include "es8328.h"
#define ES8328_SYSCLK_RATE_1X 11289600 static const unsigned int rates_12288[] = {
#define ES8328_SYSCLK_RATE_2X 22579200 8000, 12000, 16000, 24000, 32000, 48000, 96000,
};
/* Run the codec at 22.5792 or 11.2896 MHz to support these rates */ static const int ratios_12288[] = {
static struct { 10, 7, 6, 4, 3, 2, 0,
int rate; };
u8 ratio;
} mclk_ratios[] = { static const struct snd_pcm_hw_constraint_list constraints_12288 = {
{ 8000, 9 }, .count = ARRAY_SIZE(rates_12288),
{11025, 7 }, .list = rates_12288,
{22050, 4 }, };
{44100, 2 },
static const unsigned int rates_11289[] = {
8018, 11025, 22050, 44100, 88200,
};
static const int ratios_11289[] = {
9, 7, 4, 2, 0,
};
static const struct snd_pcm_hw_constraint_list constraints_11289 = {
.count = ARRAY_SIZE(rates_11289),
.list = rates_11289,
}; };
/* regulator supplies for sgtl5000, VDDD is an optional external supply */ /* regulator supplies for sgtl5000, VDDD is an optional external supply */
...@@ -57,9 +69,14 @@ static const char * const supply_names[ES8328_SUPPLY_NUM] = { ...@@ -57,9 +69,14 @@ static const char * const supply_names[ES8328_SUPPLY_NUM] = {
"HPVDD", "HPVDD",
}; };
#define ES8328_RATES (SNDRV_PCM_RATE_44100 | \ #define ES8328_RATES (SNDRV_PCM_RATE_96000 | \
SNDRV_PCM_RATE_48000 | \
SNDRV_PCM_RATE_44100 | \
SNDRV_PCM_RATE_32000 | \
SNDRV_PCM_RATE_22050 | \ SNDRV_PCM_RATE_22050 | \
SNDRV_PCM_RATE_11025) SNDRV_PCM_RATE_16000 | \
SNDRV_PCM_RATE_11025 | \
SNDRV_PCM_RATE_8000)
#define ES8328_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | \ #define ES8328_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | \
SNDRV_PCM_FMTBIT_S18_3LE | \ SNDRV_PCM_FMTBIT_S18_3LE | \
SNDRV_PCM_FMTBIT_S20_3LE | \ SNDRV_PCM_FMTBIT_S20_3LE | \
...@@ -71,6 +88,9 @@ struct es8328_priv { ...@@ -71,6 +88,9 @@ struct es8328_priv {
struct clk *clk; struct clk *clk;
int playback_fs; int playback_fs;
bool deemph; bool deemph;
int mclkdiv2;
const struct snd_pcm_hw_constraint_list *sysclk_constraints;
const int *mclk_ratios;
struct regulator_bulk_data supplies[ES8328_SUPPLY_NUM]; struct regulator_bulk_data supplies[ES8328_SUPPLY_NUM];
}; };
...@@ -443,40 +463,55 @@ static int es8328_mute(struct snd_soc_dai *dai, int mute) ...@@ -443,40 +463,55 @@ static int es8328_mute(struct snd_soc_dai *dai, int mute)
mute ? ES8328_DACCONTROL3_DACMUTE : 0); mute ? ES8328_DACCONTROL3_DACMUTE : 0);
} }
static int es8328_startup(struct snd_pcm_substream *substream,
struct snd_soc_dai *dai)
{
struct snd_soc_codec *codec = dai->codec;
struct es8328_priv *es8328 = snd_soc_codec_get_drvdata(codec);
if (es8328->sysclk_constraints)
snd_pcm_hw_constraint_list(substream->runtime, 0,
SNDRV_PCM_HW_PARAM_RATE,
es8328->sysclk_constraints);
return 0;
}
static int es8328_hw_params(struct snd_pcm_substream *substream, static int es8328_hw_params(struct snd_pcm_substream *substream,
struct snd_pcm_hw_params *params, struct snd_pcm_hw_params *params,
struct snd_soc_dai *dai) struct snd_soc_dai *dai)
{ {
struct snd_soc_codec *codec = dai->codec; struct snd_soc_codec *codec = dai->codec;
struct es8328_priv *es8328 = snd_soc_codec_get_drvdata(codec); struct es8328_priv *es8328 = snd_soc_codec_get_drvdata(codec);
int clk_rate = clk_get_rate(es8328->clk);
int i; int i;
int reg; int reg;
int val;
int wl; int wl;
u8 ratio; int ratio;
if (!es8328->sysclk_constraints) {
dev_err(codec->dev, "No MCLK configured\n");
return -EINVAL;
}
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
reg = ES8328_DACCONTROL2; reg = ES8328_DACCONTROL2;
else else
reg = ES8328_ADCCONTROL5; reg = ES8328_ADCCONTROL5;
switch (clk_rate) { for (i = 0; i < es8328->sysclk_constraints->count; i++)
case ES8328_SYSCLK_RATE_1X: if (es8328->sysclk_constraints->list[i] == params_rate(params))
val = 0; break;
break;
case ES8328_SYSCLK_RATE_2X: if (i == es8328->sysclk_constraints->count) {
val = ES8328_MASTERMODE_MCLKDIV2; dev_err(codec->dev, "LRCLK %d unsupported with current clock\n",
break; params_rate(params));
default:
dev_err(codec->dev,
"%s: clock is running at %d Hz, not %d or %d Hz\n",
__func__, clk_rate,
ES8328_SYSCLK_RATE_1X, ES8328_SYSCLK_RATE_2X);
return -EINVAL; return -EINVAL;
} }
ratio = es8328->mclk_ratios[i];
snd_soc_update_bits(codec, ES8328_MASTERMODE, snd_soc_update_bits(codec, ES8328_MASTERMODE,
ES8328_MASTERMODE_MCLKDIV2, val); ES8328_MASTERMODE_MCLKDIV2,
es8328->mclkdiv2 ? ES8328_MASTERMODE_MCLKDIV2 : 0);
switch (params_width(params)) { switch (params_width(params)) {
case 16: case 16:
...@@ -498,12 +533,6 @@ static int es8328_hw_params(struct snd_pcm_substream *substream, ...@@ -498,12 +533,6 @@ static int es8328_hw_params(struct snd_pcm_substream *substream,
return -EINVAL; return -EINVAL;
} }
/* find master mode MCLK to sampling frequency ratio */
ratio = mclk_ratios[0].rate;
for (i = 1; i < ARRAY_SIZE(mclk_ratios); i++)
if (params_rate(params) <= mclk_ratios[i].rate)
ratio = mclk_ratios[i].ratio;
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
snd_soc_update_bits(codec, ES8328_DACCONTROL1, snd_soc_update_bits(codec, ES8328_DACCONTROL1,
ES8328_DACCONTROL1_DACWL_MASK, ES8328_DACCONTROL1_DACWL_MASK,
...@@ -519,6 +548,40 @@ static int es8328_hw_params(struct snd_pcm_substream *substream, ...@@ -519,6 +548,40 @@ static int es8328_hw_params(struct snd_pcm_substream *substream,
return snd_soc_update_bits(codec, reg, ES8328_RATEMASK, ratio); return snd_soc_update_bits(codec, reg, ES8328_RATEMASK, ratio);
} }
static int es8328_set_sysclk(struct snd_soc_dai *codec_dai,
int clk_id, unsigned int freq, int dir)
{
struct snd_soc_codec *codec = codec_dai->codec;
struct es8328_priv *es8328 = snd_soc_codec_get_drvdata(codec);
int mclkdiv2 = 0;
switch (freq) {
case 0:
es8328->sysclk_constraints = NULL;
es8328->mclk_ratios = NULL;
break;
case 22579200:
mclkdiv2 = 1;
/* fallthru */
case 11289600:
es8328->sysclk_constraints = &constraints_11289;
es8328->mclk_ratios = ratios_11289;
break;
case 24576000:
mclkdiv2 = 1;
/* fallthru */
case 12288000:
es8328->sysclk_constraints = &constraints_12288;
es8328->mclk_ratios = ratios_12288;
break;
default:
return -EINVAL;
}
es8328->mclkdiv2 = mclkdiv2;
return 0;
}
static int es8328_set_dai_fmt(struct snd_soc_dai *codec_dai, static int es8328_set_dai_fmt(struct snd_soc_dai *codec_dai,
unsigned int fmt) unsigned int fmt)
{ {
...@@ -616,8 +679,10 @@ static int es8328_set_bias_level(struct snd_soc_codec *codec, ...@@ -616,8 +679,10 @@ static int es8328_set_bias_level(struct snd_soc_codec *codec,
} }
static const struct snd_soc_dai_ops es8328_dai_ops = { static const struct snd_soc_dai_ops es8328_dai_ops = {
.startup = es8328_startup,
.hw_params = es8328_hw_params, .hw_params = es8328_hw_params,
.digital_mute = es8328_mute, .digital_mute = es8328_mute,
.set_sysclk = es8328_set_sysclk,
.set_fmt = es8328_set_dai_fmt, .set_fmt = es8328_set_dai_fmt,
}; };
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment