All of lore.kernel.org
 help / color / mirror / Atom feed
From: Lu Guanqun <guanqun.lu@intel.com>
To: Rajeev Kumar <rajeev-dlh.kumar@st.com>
Cc: "tiwai@suse.de" <tiwai@suse.de>,
	"alsa-devel@alsa-project.org" <alsa-devel@alsa-project.org>,
	"broonie@opensource.wolfsonmicro.com"
	<broonie@opensource.wolfsonmicro.com>,
	"lrg@slimlogic.co.uk" <lrg@slimlogic.co.uk>
Subject: Re: [PATCH V3 1/5] sound: asoc: Adding support for STA529 Audio Codec
Date: Mon, 11 Apr 2011 20:53:18 +0800	[thread overview]
Message-ID: <20110411125318.GA11072@qtel.sh.intel.com> (raw)
In-Reply-To: <1302499804-24386-2-git-send-email-rajeev-dlh.kumar@st.com>

On Mon, Apr 11, 2011 at 01:30:00PM +0800, Rajeev Kumar wrote:
> This patch adds the support for STA529 audio codec.
> Details of the audio codec can be seen here:
> http://www.st.com/internet/imag_video/product/159187.jsp
> 
> Signed-off-by: Rajeev Kumar <rajeev-dlh.kumar@st.com>
> ---
>  sound/soc/codecs/Kconfig  |    5 +
>  sound/soc/codecs/Makefile |    2 +
>  sound/soc/codecs/sta529.c |  383 +++++++++++++++++++++++++++++++++++++++++++++
>  3 files changed, 390 insertions(+), 0 deletions(-)
>  create mode 100644 sound/soc/codecs/sta529.c
> 
> diff --git a/sound/soc/codecs/Kconfig b/sound/soc/codecs/Kconfig
> index b814ed0..d536740 100644
> --- a/sound/soc/codecs/Kconfig
> +++ b/sound/soc/codecs/Kconfig
> @@ -40,6 +40,7 @@ config SND_SOC_ALL_CODECS
>         select SND_SOC_SN95031 if INTEL_SCU_IPC
>         select SND_SOC_SPDIF
>         select SND_SOC_SSM2602 if I2C
> +       select SND_SOC_STA529 if I2C
>         select SND_SOC_STAC9766 if SND_SOC_AC97_BUS
>         select SND_SOC_TLV320AIC23 if I2C
>         select SND_SOC_TLV320AIC26 if SPI_MASTER
> @@ -206,6 +207,10 @@ config SND_SOC_SPDIF
>  config SND_SOC_SSM2602
>         tristate
> 
> +config SND_SOC_STA529
> +       tristate
> +       depends on I2C

I see many drivers depend on I2C, but they don't add the line above. Any
reasons for it?

> +
>  config SND_SOC_STAC9766
>         tristate
> 
> diff --git a/sound/soc/codecs/Makefile b/sound/soc/codecs/Makefile
> index 49121ad..f889ef5 100644
> --- a/sound/soc/codecs/Makefile
> +++ b/sound/soc/codecs/Makefile
> @@ -26,6 +26,7 @@ snd-soc-alc5623-objs := alc5623.o
>  snd-soc-sn95031-objs := sn95031.o
>  snd-soc-spdif-objs := spdif_transciever.o
>  snd-soc-ssm2602-objs := ssm2602.o
> +snd-soc-sta529-objs := sta529.o
>  snd-soc-stac9766-objs := stac9766.o
>  snd-soc-tlv320aic23-objs := tlv320aic23.o
>  snd-soc-tlv320aic26-objs := tlv320aic26.o
> @@ -114,6 +115,7 @@ obj-$(CONFIG_SND_SOC_SGTL5000)  += snd-soc-sgtl5000.o
>  obj-$(CONFIG_SND_SOC_SN95031)  +=snd-soc-sn95031.o
>  obj-$(CONFIG_SND_SOC_SPDIF)    += snd-soc-spdif.o
>  obj-$(CONFIG_SND_SOC_SSM2602)  += snd-soc-ssm2602.o
> +obj-$(CONFIG_SND_SOC_STA529)   += snd-soc-sta529.o
>  obj-$(CONFIG_SND_SOC_STAC9766) += snd-soc-stac9766.o
>  obj-$(CONFIG_SND_SOC_TLV320AIC23)      += snd-soc-tlv320aic23.o
>  obj-$(CONFIG_SND_SOC_TLV320AIC26)      += snd-soc-tlv320aic26.o
> diff --git a/sound/soc/codecs/sta529.c b/sound/soc/codecs/sta529.c
> new file mode 100644
> index 0000000..8dda5cd
> --- /dev/null
> +++ b/sound/soc/codecs/sta529.c
> @@ -0,0 +1,383 @@
> +/*
> + * ASoC codec driver for spear platform
> + *
> + * sound/soc/codecs/sta529.c -- spear ALSA Soc codec driver
> + *
> + * Copyright (C) 2011 ST Microelectronics
> + * Rajeev Kumar <rajeev-dlh.kumar@st.com>
> + *
> + * This file is licensed under the terms of the GNU General Public
> + * License version 2. This program is licensed "as is" without any
> + * warranty of any kind, whether express or implied.
> + */
> +
> +#include <linux/clk.h>
> +#include <linux/delay.h>
> +#include <linux/init.h>
> +#include <linux/i2c.h>
> +#include <linux/io.h>
> +#include <linux/module.h>
> +#include <linux/moduleparam.h>
> +#include <linux/platform_device.h>
> +#include <linux/pm.h>
> +#include <linux/slab.h>
> +
> +#include <sound/core.h>
> +#include <sound/initval.h>
> +#include <sound/pcm.h>
> +#include <sound/pcm_params.h>
> +#include <sound/soc.h>
> +#include <sound/soc-dapm.h>
> +#include <sound/tlv.h>
> +
> +/* STA529 Register offsets */
> +#define         STA529_FFXCFG0         0x00
> +#define         STA529_FFXCFG1         0x01
> +#define         STA529_MVOL            0x02
> +#define         STA529_LVOL            0x03
> +#define         STA529_RVOL            0x04
> +#define         STA529_TTF0            0x05
> +#define         STA529_TTF1            0x06
> +#define         STA529_TTP0            0x07
> +#define         STA529_TTP1            0x08
> +#define         STA529_S2PCFG0         0x0A
> +#define         STA529_S2PCFG1         0x0B
> +#define         STA529_P2SCFG0         0x0C
> +#define         STA529_P2SCFG1         0x0D
> +#define         STA529_PLLCFG0         0x14
> +#define         STA529_PLLCFG1         0x15
> +#define         STA529_PLLCFG2         0x16
> +#define         STA529_PLLCFG3         0x17
> +#define         STA529_PLLPFE          0x18
> +#define         STA529_PLLST           0x19
> +#define         STA529_ADCCFG          0x1E /*mic_select*/
> +#define         STA529_CKOCFG          0x1F
> +#define         STA529_MISC            0x20
> +#define         STA529_PADST0          0x21
> +#define         STA529_PADST1          0x22
> +#define         STA529_FFXST           0x23
> +#define         STA529_PWMIN1          0x2D
> +#define         STA529_PWMIN2          0x2E
> +#define         STA529_POWST           0x32
> +
> +#define STA529_CACHEREGNUM     0x33 /*total num of reg. in sta529*/
> +
> +#define STA529_RATES           SNDRV_PCM_RATE_48000
> +#define STA529_FORMAT          SNDRV_PCM_FMTBIT_S16_LE
> +#define        S2PC_VALUE              0x98
> +#define CLOCK_OUT              0x60
> +#define LEFT_J_DATA_FORMAT     0x10
> +#define I2S_DATA_FORMAT                0x12
> +#define RIGHT_J_DATA_FORMAT    0x14
> +#define CODEC_MUTE_VAL         0x80
> +
> +#define POWER_CNTLMSAK         0x40
> +#define POWER_STDBY            0x40
> +#define FFX_MASK               0x80
> +#define FFX_OFF                        0x80
> +#define POWER_UP               0x00
> +
> +static const u8 sta529_reg[STA529_CACHEREGNUM] = {
> +       0x75, 0xf8, 0x50, 0x00,
> +       0x00, 0x00, 0x02, 0x00,
> +       0x02, 0x02, 0xD2, 0x91,
> +       0xD3, 0x91, 0x00, 0x00,
> +       0x00, 0x00, 0x00, 0x00,
> +       0x00, 0x00, 0x00, 0x00,
> +       0x80, 0x00, 0x00, 0x00,
> +       0x00, 0x00, 0x52, 0x40,
> +       0x21, 0xef, 0x04, 0x06,
> +       0x41, 0x00, 0x00, 0x00,
> +       0x00, 0x00, 0x00, 0x00,
> +       0x00, 0x00, 0x00, 0x00,
> +       0x00, 0x00,
> +};
> +
> +struct sta529 {
> +       unsigned int sysclk;
> +       enum snd_soc_control_type control_type;
> +       void *control_data;
> +};
> +
> +static const char *pwm_mode_text[] = { "binary", "headphone", "ternary",
> +       "phase-shift"};
> +static const char *op_mode_text[] = { "slave", "master"};
> +
> +static const struct soc_enum pwm_src_enum =
> +SOC_ENUM_SINGLE(STA529_FFXCFG1, 4, 4, pwm_mode_text);
> +
> +static const struct soc_enum mode_src_enum =
> +SOC_ENUM_SINGLE(STA529_P2SCFG0, 0, 2, op_mode_text);
> +
> +static const struct snd_kcontrol_new sta529_new_snd_controls[] = {
> +       SOC_ENUM("pwm select", pwm_src_enum),
> +       SOC_ENUM("mode select", mode_src_enum),
> +};
> +
> +static const DECLARE_TLV_DB_SCALE(out_gain_tlv, -9150, 50, 0);
> +static const DECLARE_TLV_DB_SCALE(master_vol_tlv, -12750, 50, 0);
> +
> +static const struct snd_kcontrol_new sta529_snd_controls[] = {
> +       SOC_DOUBLE_R_TLV("Digital Playback Volume", STA529_LVOL, STA529_RVOL, 0,
> +                       127, 0, out_gain_tlv),
> +       SOC_SINGLE_TLV("Master Playback Volume", STA529_MVOL, 0, 127, 1,
> +                       master_vol_tlv),
> +};
> +
> +static int sta529_hw_params(struct snd_pcm_substream *substream,
> +               struct snd_pcm_hw_params *params,
> +               struct snd_soc_dai *dai)
> +{
> +       struct snd_soc_pcm_runtime *rtd = substream->private_data;
> +       struct snd_soc_codec *codec = rtd->codec;
> +       int pdata = 0;
> +
> +       switch (params_format(params)) {
> +       case SNDRV_PCM_FORMAT_S16_LE:
> +               pdata = 1;
> +               break;
> +       case SNDRV_PCM_FORMAT_S24_LE:
> +               pdata = 2;
> +               break;
> +       case SNDRV_PCM_FORMAT_S32_LE:
> +               pdata = 3;
> +               break;
> +       }
> +       if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
> +               snd_soc_update_bits(codec, STA529_S2PCFG1, 0xB0, pdata);
> +       else
> +               snd_soc_update_bits(codec, STA529_P2SCFG1, 0xB0, pdata);
> +
> +       return 0;
> +}
> +
> +static int sta529_set_dai_fmt(struct snd_soc_dai *codec_dai, u32 fmt)
> +{
> +       struct snd_soc_codec *codec = codec_dai->codec;
> +       u8 mode = 0;
> +
> +       /* interface format */
> +       switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
> +       case SND_SOC_DAIFMT_LEFT_J:
> +               mode = LEFT_J_DATA_FORMAT;
> +               break;
> +       case SND_SOC_DAIFMT_I2S:
> +               mode = I2S_DATA_FORMAT;
> +               break;
> +       case SND_SOC_DAIFMT_RIGHT_J:
> +               mode = RIGHT_J_DATA_FORMAT;
> +               break;
> +       default:
> +               return -EINVAL;
> +       }
> +       mode |= 0x20;
> +       snd_soc_update_bits(codec, STA529_S2PCFG0, 0xE0, mode);
> +
> +       return 0;
> +}
> +
> +static int sta529_mute(struct snd_soc_dai *dai, int mute)
> +{
> +       struct snd_soc_codec *codec = dai->codec;
> +
> +       u8 mute_reg = snd_soc_read(codec, STA529_FFXCFG0) & ~CODEC_MUTE_VAL;
> +
> +       if (mute)
> +               mute_reg |= CODEC_MUTE_VAL;
> +
> +       snd_soc_update_bits(codec, STA529_FFXCFG0, 0x80, 00);
> +
> +       return 0;
> +}
> +
> +static int
> +sta529_set_bias_level(struct snd_soc_codec *codec,
> +               enum snd_soc_bias_level level)
> +{
> +       switch (level) {
> +       case SND_SOC_BIAS_ON:
> +       case SND_SOC_BIAS_PREPARE:
> +               snd_soc_update_bits(codec, STA529_FFXCFG0, POWER_CNTLMSAK,
> +                               POWER_UP);
> +               snd_soc_update_bits(codec, STA529_MISC, 1, 0x01);
> +               break;
> +       case SND_SOC_BIAS_STANDBY:
> +       case SND_SOC_BIAS_OFF:
> +               snd_soc_update_bits(codec, STA529_FFXCFG0, POWER_CNTLMSAK,
> +                               POWER_STDBY);
> +               /* Making FFX output to zero */
> +               snd_soc_update_bits(codec, STA529_FFXCFG0, FFX_MASK,
> +                               FFX_OFF);
> +               snd_soc_update_bits(codec, STA529_MISC, 1, 0x00);
> +
> +               break;
> +       }
> +
> +       /*store the label for powers down audio subsystem for suspend.This is
> +        ** used by soc core layer*/
> +       codec->bias_level = level;
> +       return 0;
> +
> +}
> +
> +static struct snd_soc_dai_ops sta529_dai_ops = {
> +       .hw_params      =       sta529_hw_params,
> +       .set_fmt        =       sta529_set_dai_fmt,
> +       .digital_mute   =       sta529_mute,
> +};
> +
> +static struct snd_soc_dai_driver sta529_dai = {
> +       .name = "sta529-audio",
> +       .playback = {
> +               .stream_name = "Playback",
> +               .channels_min = 2,
> +               .channels_max = 2,
> +               .rates = STA529_RATES,
> +               .formats = STA529_FORMAT,
> +       },
> +       .capture = {
> +               .stream_name = "Capture",
> +               .channels_min = 2,
> +               .channels_max = 2,
> +               .rates = STA529_RATES,
> +               .formats = STA529_FORMAT,
> +       },
> +       .ops    = &sta529_dai_ops,
> +};
> +
> +static int sta529_probe(struct snd_soc_codec *codec)
> +{
> +       struct sta529 *sta529 = snd_soc_codec_get_drvdata(codec);
> +       int ret;
> +
> +       codec->hw_write = (hw_write_t)i2c_master_send;
> +       codec->hw_read = NULL;
> +       ret = snd_soc_codec_set_cache_io(codec, 8, 8, sta529->control_type);
> +       if (ret < 0) {
> +               dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
> +               return ret;
> +       }
> +
> +       sta529_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
> +
> +       snd_soc_add_controls(codec, sta529_snd_controls,
> +                       ARRAY_SIZE(sta529_snd_controls));
> +
> +       snd_soc_add_controls(codec, sta529_new_snd_controls,
> +                       ARRAY_SIZE(sta529_new_snd_controls));
> +       return 0;
> +}
> +
> +/* power down chip */
> +static int sta529_remove(struct snd_soc_codec *codec)
> +{
> +       sta529_set_bias_level(codec, SND_SOC_BIAS_OFF);
> +
> +       return 0;
> +}
> +
> +static int sta529_suspend(struct snd_soc_codec *codec, pm_message_t state)
> +{
> +       sta529_set_bias_level(codec, SND_SOC_BIAS_OFF);
> +
> +       return 0;
> +}
> +
> +static int sta529_resume(struct snd_soc_codec *codec)
> +{
> +       int i;
> +       u8 data[2];
> +       u8 *cache = codec->reg_cache;
> +
> +       for (i = 0; i < ARRAY_SIZE(sta529_reg); i++) {
> +               data[0] = i;
> +               data[1] = cache[i];
> +               codec->hw_write(codec->control_data, data, 2);
> +       }
> +
> +       sta529_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
> +       sta529_set_bias_level(codec, codec->suspend_bias_level);
> +       return 0;
> +}
> +
> +struct snd_soc_codec_driver soc_codec_dev_sta529 = {
> +       .probe = sta529_probe,
> +       .remove = sta529_remove,
> +       .set_bias_level = sta529_set_bias_level,
> +       .suspend = sta529_suspend,
> +       .resume = sta529_resume,
> +       .reg_cache_size = STA529_CACHEREGNUM,
> +       .reg_word_size = sizeof(u8),
> +       .reg_cache_default = sta529_reg,
> +
> +};
> +
> +static __devinit int sta529_i2c_probe(struct i2c_client *i2c,
> +               const struct i2c_device_id *id)
> +{
> +       struct sta529 *sta529;
> +       int ret;
> +
> +       if (!i2c_check_functionality(i2c->adapter, I2C_FUNC_SMBUS_BYTE_DATA))
> +               return -EINVAL;
> +
> +       sta529 = kzalloc(sizeof(struct sta529), GFP_KERNEL);
> +       if (sta529 == NULL)
> +               return -ENOMEM;
> +
> +       i2c_set_clientdata(i2c, sta529);
> +       sta529->control_data = i2c;
> +       sta529->control_type = SND_SOC_I2C;
> +
> +       ret = snd_soc_register_codec(&i2c->dev,
> +                       &soc_codec_dev_sta529, &sta529_dai, 1);
> +       if (ret < 0)
> +               kfree(sta529);
> +       return ret;
> +}
> +
> +static int sta529_i2c_remove(struct i2c_client *client)
> +{
> +       snd_soc_unregister_codec(&client->dev);
> +       kfree(i2c_get_clientdata(client));
> +       return 0;
> +}
> +
> +static const struct i2c_device_id sta529_i2c_id[] = {
> +       { "sta529", 0 },
> +       { }
> +};
> +MODULE_DEVICE_TABLE(i2c, sta529_i2c_id);
> +
> +static struct i2c_driver sta529_i2c_driver = {
> +       .driver = {
> +               .name = "sta529",
> +               .owner = THIS_MODULE,
> +       },
> +       .probe          = sta529_i2c_probe,
> +       .remove         = __devexit_p(sta529_i2c_remove),
> +       .id_table       = sta529_i2c_id,
> +};
> +
> +static int __init sta529_modinit(void)
> +{
> +       int ret = 0;
> +
> +       ret = i2c_add_driver(&sta529_i2c_driver);

As an idiom, I often see the above code surrounded with an #ifdef, see
below code snippet from sound/soc/codecs/wm8974.c

static int __init wm8974_modinit(void)
{
        int ret = 0;
#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
        ret = i2c_add_driver(&wm8974_i2c_driver);
        if (ret != 0) {
                printk(KERN_ERR "Failed to register wm8974 I2C driver: %d\n",
                       ret);
        }
#endif
        return ret;
}

Maybe you should add the simlar structure as well.

> +       if (ret != 0)
> +               printk(KERN_ERR "Failed to reg sta529 I2C driver: %d\n", ret);
> +
> +       return ret;
> +
> +}
> +module_init(sta529_modinit);
> +
> +static void __exit sta529_exit(void)
> +{
> +       i2c_del_driver(&sta529_i2c_driver);
> +}
> +module_exit(sta529_exit);
> +
> +MODULE_DESCRIPTION("ASoC STA529 codec driver");
> +MODULE_AUTHOR("Rajeev Kumar <rajeev-dlh.kumar@st.com>");
> +MODULE_LICENSE("GPL");
> --
> 1.6.0.2
> 
> _______________________________________________
> Alsa-devel mailing list
> Alsa-devel@alsa-project.org
> http://mailman.alsa-project.org/mailman/listinfo/alsa-devel

-- 
guanqun

  parent reply	other threads:[~2011-04-11 12:54 UTC|newest]

Thread overview: 27+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2011-04-11  5:29 [PATCH V3 0/5] Adding ASoC drivers for SPEAr13XX platform Rajeev Kumar
2011-04-11  5:30 ` [PATCH V3 1/5] sound: asoc: Adding support for STA529 Audio Codec Rajeev Kumar
2011-04-11  5:30   ` [PATCH V3 2/5] sound: asoc: Adding support for SPEAr13XX ASoC platform driver Rajeev Kumar
2011-04-11  5:30     ` [PATCH V3 3/5] sound: asoc: Adding support for SPEAr13XX ASoC machine driver Rajeev Kumar
2011-04-11  5:30       ` [PATCH V3 4/5] sound: asoc: Adding Kconfig and Makefile to support SPEAr13XX ASoC driver Rajeev Kumar
2011-04-11  5:30         ` [PATCH V3 5/5] sound: asoc: Adding support for SPEAr13XX in soc Rajeev Kumar
2011-04-11 13:00     ` [PATCH V3 2/5] sound: asoc: Adding support for SPEAr13XX ASoC platform driver Lu Guanqun
2011-04-18  5:00       ` rajeev
2011-04-11 12:53   ` Lu Guanqun [this message]
2011-04-11 13:35     ` [PATCH V3 1/5] sound: asoc: Adding support for STA529 Audio Codec Mark Brown
2011-04-11 15:09       ` Lu Guanqun
2011-04-11 23:54         ` Mark Brown
2011-04-12  0:06           ` Lu Guanqun
2011-04-11 14:56   ` Mark Brown
2011-04-14  9:54     ` Takashi Iwai
2011-04-14 12:18       ` Mark Brown
2011-04-14 12:25         ` Takashi Iwai
2011-04-14 14:19       ` Lu Guanqun
2011-04-14 14:28         ` Lu Guanqun
2011-04-15  9:27           ` Takashi Iwai
2011-04-15 14:10             ` Lu Guanqun
2011-04-15  9:23         ` Takashi Iwai
2011-04-20  4:24     ` rajeev
2011-04-20 10:56       ` Mark Brown
2011-04-20 11:41         ` rajeev
2011-04-20 11:42         ` rajeev
2011-04-11 15:45   ` Lu Guanqun

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=20110411125318.GA11072@qtel.sh.intel.com \
    --to=guanqun.lu@intel.com \
    --cc=alsa-devel@alsa-project.org \
    --cc=broonie@opensource.wolfsonmicro.com \
    --cc=lrg@slimlogic.co.uk \
    --cc=rajeev-dlh.kumar@st.com \
    --cc=tiwai@suse.de \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.