On Thu, Mar 27, 2014 at 05:18:02PM +0100, Daniel Mack wrote:
From: Sven Brandau brandau@gmx.de
The TI STA350 is an integrated 2.1-channel power amplifier that is controllable over I2C. This patch adds an ASoC driver for it.
At a glance, this chip is very similar to the STA320 for which a driver already exists. In details, however, the register maps contain subtle differences which made a whole new driver easier to write and maintain.
A few issues below, some may already be present in the sta320 driver, generally newer idioms and so on, but that's no reason to not fix them here!
[daniel@zonque.org: cleanups, DT property rework, rebased on asoc-next] Signed-off-by: Sven Brandau brandau@gmx.de
I'm sending this patch on behalf of Sven Brandau, who wrote the driver, based on earlier bits for the STA320.
You need to add a Signed-off-by if you're the one sending it.
+config SND_SOC_STA350
- tristate
DT capable drivers should be visible in Kconfig so they can be used with simple-audio.
+/* regulator power supply names */ +static const char const *sta350_supply_names[] = {
- "vdd-dig", /* digital supply, 3.3V */
- "vdd-pll", /* pll supply, 3.3V */
- "vcc" /* power amp supply, 5V - 26V */
+};
These should be documented as mandatory properties in the DT binding but weren't mentioned at all.
+static const char const *sta350_noise_shaper_type[] = {
- "Third order", "Forth order"
+};
Fourth.
+static const struct soc_enum sta350_limiter2_release_rate_enum =
- SOC_ENUM_SINGLE(STA350_L2AR, STA350_LxR_SHIFT,
16, sta350_limiter_release_rate);
Use SOC_ENUM_SINGLE_DECL if you can, it makes things a bit less error prone.
+static int sta350_coefficient_get(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
+{
- struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
- struct sta350_priv *sta350 = snd_soc_codec_get_drvdata(codec);
- int numcoef = kcontrol->private_value >> 16;
- int index = kcontrol->private_value & 0xffff;
- unsigned int cfud, val;
- int i;
- /* preserve reserved bits in STA350_CFUD */
- regmap_read(sta350->regmap, STA350_CFUD, &cfud);
- cfud &= 0xf0;
- /*
* chip documentation does not say if the bits are self clearing,
* so do it explicitly
*/
- regmap_write(sta350->regmap, STA350_CFUD, cfud);
- regmap_write(sta350->regmap, STA350_CFADDR2, index);
- if (numcoef == 1)
regmap_write(sta350->regmap, STA350_CFUD, cfud | 0x04);
- else if (numcoef == 5)
regmap_write(sta350->regmap, STA350_CFUD, cfud | 0x08);
- else
return -EINVAL;
- for (i = 0; i < 3 * numcoef; i++) {
regmap_read(sta350->regmap, STA350_B1CF1 + i, &val);
ucontrol->value.bytes.data[i] = val;
- }
You need some sort of locking around these windows I think?
+SND_SOC_DAPM_DAC("DAC", "Playback", SND_SOC_NOPM, 0, 0),
Don't specify a stream on the DAC, use DAPM to connect the DAI to the DAC (the DAI name is a DAPM widget name).
+static int sta350_set_dai_sysclk(struct snd_soc_dai *codec_dai,
int clk_id, unsigned int freq, int dir)
+{
- struct snd_soc_codec *codec = codec_dai->codec;
- struct sta350_priv *sta350 = snd_soc_codec_get_drvdata(codec);
- pr_debug("mclk=%u\n", freq);
- sta350->mclk = freq;
dev_dbg() here and for some of the rest of the code too.
- regmap_read(sta350->regmap, STA350_CONFB, &confb);
Or just use _update_bits() at the end?
- for (i = 0; i < ARRAY_SIZE(interpolation_ratios); i++)
if (interpolation_ratios[i].fs == rate) {
ir = interpolation_ratios[i].ir;
break;
}
- if (ir < 0) {
Can we have some braces around the for () loop please - it'd be easier to read (similarly for others).
- regmap_read(sta350->regmap, STA350_CONFB, &confb);
- confb &= ~(STA350_CONFB_SAI_MASK | STA350_CONFB_SAIFB);
- switch (params_format(params)) {
- case SNDRV_PCM_FORMAT_S24_LE:
- case SNDRV_PCM_FORMAT_S24_BE:
- case SNDRV_PCM_FORMAT_S24_3LE:
- case SNDRV_PCM_FORMAT_S24_3BE:
pr_debug("24bit\n");
/* fall through */
Use params_width() and this all gets shorter.
+static int sta350_startup_sequence(struct sta350_priv *sta350) +{
- if (gpio_is_valid(sta350->gpio_power_down))
gpio_set_value(sta350->gpio_power_down, 1);
- if (gpio_is_valid(sta350->gpio_nreset)) {
gpio_set_value(sta350->gpio_nreset, 1);
mdelay(1);
gpio_set_value(sta350->gpio_nreset, 0);
mdelay(1);
gpio_set_value(sta350->gpio_nreset, 1);
mdelay(1);
- }
That sequence looks odd - we bring the device out of reset, reset it again and then bring it back to reset. It'd be more normal to leave the device in reset when idle and then just bring it out of reset rather than something like this. Why the odd sequence?
+static int sta350_probe(struct snd_soc_codec *codec) +{
- struct sta350_priv *sta350 = snd_soc_codec_get_drvdata(codec);
- struct sta350_platform_data *pdata = sta350->pdata;
- int i, ret = 0, thermal = 0;
- if (gpio_is_valid(sta350->gpio_nreset))
if (devm_gpio_request_one(codec->dev, sta350->gpio_nreset,
GPIOF_OUT_INIT_HIGH,
"ST350 Reset"))
sta350->gpio_nreset = -EINVAL;
- if (gpio_is_valid(sta350->gpio_power_down))
if (devm_gpio_request_one(codec->dev, sta350->gpio_power_down,
GPIOF_OUT_INIT_HIGH,
"ST350 Power-Down"))
sta350->gpio_power_down = -EINVAL;
This doesn't handle probe deferral - it needs to special case -EPROBE_DEFER (or just pass back the error it gets). The driver should also be doing this at the I2C level probe not the ASoC level one.
- ret = sta350_startup_sequence(sta350);
- if (ret < 0) {
dev_err(codec->dev, "Failed to startup device\n");
return ret;
- }
- ret = regulator_bulk_enable(ARRAY_SIZE(sta350->supplies),
sta350->supplies);
- if (ret < 0) {
dev_err(codec->dev, "Failed to enable supplies: %d\n", ret);
return ret;
- }
I would have expected to run through the startup sequence after applying power rather than before otherwise the startup sequence may not be observed.
- /*
* Tell ASoC what kind of I/O to use to read the registers. ASoC will
* then do the I2C transactions itself.
*/
- codec->control_data = sta350->regmap;
Shouldn't be required any more, dev_get_regmap() will DTRT.
- /*
* Chip documentation explicitly requires that the reset values
* of reserved register bits are left untouched.
* Write the register default value to cache for reserved registers,
* so the write to the these registers are suppressed by the cache
* restore code when it skips writes of default registers.
*/
- regcache_cache_only(sta350->regmap, true);
- regmap_write(sta350->regmap, STA350_CONFC, 0xc2);
- regmap_write(sta350->regmap, STA350_CONFE, 0xc2);
- regmap_write(sta350->regmap, STA350_CONFF, 0x5c);
- regmap_write(sta350->regmap, STA350_MMUTE, 0x10);
- regmap_write(sta350->regmap, STA350_AUTO1, 0x60);
- regmap_write(sta350->regmap, STA350_AUTO3, 0x00);
- regmap_write(sta350->regmap, STA350_C3CFG, 0x40);
- regcache_cache_only(sta350->regmap, false);
I don't understand this? If defaults are provided (and they are) we will already default to them so this should have no effect.
+static int sta350_remove(struct snd_soc_codec *codec) +{
- struct sta350_priv *sta350 = snd_soc_codec_get_drvdata(codec);
- if (gpio_is_valid(sta350->gpio_nreset))
gpio_set_value(sta350->gpio_nreset, 0);
- if (gpio_is_valid(sta350->gpio_power_down))
gpio_set_value(sta350->gpio_power_down, 0);
- sta350_set_bias_level(codec, SND_SOC_BIAS_OFF);
- regulator_bulk_disable(ARRAY_SIZE(sta350->supplies), sta350->supplies);
I'd expect the GPIO and regulator stuff to be happening over suspend too. _BIAS_OFF does the power down but not the reset.
+#ifdef CONFIG_OF
- if (of_match_device(st350_dt_ids, &i2c->dev)) {
ret = sta350_probe_dt(&i2c->dev, sta350);
if (ret < 0)
return ret;
- }
+#endif
This is weird - normally we just go and try to parse the DT?