linux/sound/soc/samsung/smdk_wm8580pcm.c
<<
>>
Prefs
   1/*
   2 *  sound/soc/samsung/smdk_wm8580pcm.c
   3 *
   4 *  Copyright (c) 2011 Samsung Electronics Co. Ltd
   5 *
   6 *  This program is free software; you can redistribute  it and/or  modify it
   7 *  under  the terms of  the GNU General  Public License as published by the
   8 *  Free Software Foundation;  either version 2 of the  License, or (at your
   9 *  option) any later version.
  10 */
  11#include <linux/module.h>
  12#include <sound/soc.h>
  13#include <sound/pcm_params.h>
  14#include <sound/pcm.h>
  15
  16#include <asm/mach-types.h>
  17
  18#include "../codecs/wm8580.h"
  19#include "dma.h"
  20#include "pcm.h"
  21
  22/*
  23 * Board Settings:
  24 *  o '1' means 'ON'
  25 *  o '0' means 'OFF'
  26 *  o 'X' means 'Don't care'
  27 *
  28 * SMDK6410 Base B/D: CFG1-0000, CFG2-1111
  29 * SMDKC110, SMDKV210: CFGB11-100100, CFGB12-0000
  30 */
  31
  32#define SMDK_WM8580_EXT_OSC 12000000
  33#define SMDK_WM8580_EXT_MCLK 4096000
  34#define SMDK_WM8580_EXT_VOICE 2048000
  35
  36static unsigned long mclk_freq;
  37static unsigned long xtal_freq;
  38
  39/*
  40 * If MCLK clock directly gets from XTAL, we don't have to use PLL
  41 * to make MCLK, but if XTAL clock source connects with other codec
  42 * pin (like XTI), we should have to set codec's PLL to make MCLK.
  43 * Because Samsung SoC does not support pcmcdclk output like I2S.
  44 */
  45
  46static int smdk_wm8580_pcm_hw_params(struct snd_pcm_substream *substream,
  47                              struct snd_pcm_hw_params *params)
  48{
  49        struct snd_soc_pcm_runtime *rtd = substream->private_data;
  50        struct snd_soc_dai *codec_dai = rtd->codec_dai;
  51        struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
  52        int rfs, ret;
  53
  54        switch (params_rate(params)) {
  55        case 8000:
  56                break;
  57        default:
  58                printk(KERN_ERR "%s:%d Sampling Rate %u not supported!\n",
  59                __func__, __LINE__, params_rate(params));
  60                return -EINVAL;
  61        }
  62
  63        rfs = mclk_freq / params_rate(params) / 2;
  64
  65        if (mclk_freq == xtal_freq) {
  66                ret = snd_soc_dai_set_sysclk(codec_dai, WM8580_CLKSRC_MCLK,
  67                                                mclk_freq, SND_SOC_CLOCK_IN);
  68                if (ret < 0)
  69                        return ret;
  70
  71                ret = snd_soc_dai_set_clkdiv(codec_dai, WM8580_MCLK,
  72                                                WM8580_CLKSRC_MCLK);
  73                if (ret < 0)
  74                        return ret;
  75        } else {
  76                ret = snd_soc_dai_set_sysclk(codec_dai, WM8580_CLKSRC_PLLA,
  77                                                mclk_freq, SND_SOC_CLOCK_IN);
  78                if (ret < 0)
  79                        return ret;
  80
  81                ret = snd_soc_dai_set_clkdiv(codec_dai, WM8580_MCLK,
  82                                                WM8580_CLKSRC_PLLA);
  83                if (ret < 0)
  84                        return ret;
  85
  86                ret = snd_soc_dai_set_pll(codec_dai, WM8580_PLLA, 0,
  87                                                xtal_freq, mclk_freq);
  88                if (ret < 0)
  89                        return ret;
  90        }
  91
  92        /* Set PCM source clock on CPU */
  93        ret = snd_soc_dai_set_sysclk(cpu_dai, S3C_PCM_CLKSRC_MUX,
  94                                        mclk_freq, SND_SOC_CLOCK_IN);
  95        if (ret < 0)
  96                return ret;
  97
  98        /* Set SCLK_DIV for making bclk */
  99        ret = snd_soc_dai_set_clkdiv(cpu_dai, S3C_PCM_SCLK_PER_FS, rfs);
 100        if (ret < 0)
 101                return ret;
 102
 103        return 0;
 104}
 105
 106static struct snd_soc_ops smdk_wm8580_pcm_ops = {
 107        .hw_params = smdk_wm8580_pcm_hw_params,
 108};
 109
 110#define SMDK_DAI_FMT (SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_IB_NF | \
 111        SND_SOC_DAIFMT_CBS_CFS)
 112
 113static struct snd_soc_dai_link smdk_dai[] = {
 114        {
 115                .name = "WM8580 PAIF PCM RX",
 116                .stream_name = "Playback",
 117                .cpu_dai_name = "samsung-pcm.0",
 118                .codec_dai_name = "wm8580-hifi-playback",
 119                .platform_name = "samsung-audio",
 120                .codec_name = "wm8580.0-001b",
 121                .dai_fmt = SMDK_DAI_FMT,
 122                .ops = &smdk_wm8580_pcm_ops,
 123        }, {
 124                .name = "WM8580 PAIF PCM TX",
 125                .stream_name = "Capture",
 126                .cpu_dai_name = "samsung-pcm.0",
 127                .codec_dai_name = "wm8580-hifi-capture",
 128                .platform_name = "samsung-pcm.0",
 129                .codec_name = "wm8580.0-001b",
 130                .dai_fmt = SMDK_DAI_FMT,
 131                .ops = &smdk_wm8580_pcm_ops,
 132        },
 133};
 134
 135static struct snd_soc_card smdk_pcm = {
 136        .name = "SMDK-PCM",
 137        .owner = THIS_MODULE,
 138        .dai_link = smdk_dai,
 139        .num_links = 2,
 140};
 141
 142/*
 143 * After SMDKC110 Base Board's Rev is '0.1', 12MHz External OSC(X1)
 144 * is absent (or not connected), so we connect EXT_VOICE_CLK(OSC4),
 145 * 2.0484Mhz, directly with MCLK both Codec and SoC.
 146 */
 147static int snd_smdk_probe(struct platform_device *pdev)
 148{
 149        int ret = 0;
 150
 151        xtal_freq = SMDK_WM8580_EXT_OSC;
 152        mclk_freq = SMDK_WM8580_EXT_MCLK;
 153
 154        if (machine_is_smdkc110() || machine_is_smdkv210())
 155                xtal_freq = mclk_freq = SMDK_WM8580_EXT_VOICE;
 156
 157        smdk_pcm.dev = &pdev->dev;
 158        ret = devm_snd_soc_register_card(&pdev->dev, &smdk_pcm);
 159        if (ret)
 160                dev_err(&pdev->dev, "snd_soc_register_card failed %d\n", ret);
 161
 162        return ret;
 163}
 164
 165static struct platform_driver snd_smdk_driver = {
 166        .driver = {
 167                .name = "samsung-smdk-pcm",
 168        },
 169        .probe = snd_smdk_probe,
 170};
 171
 172module_platform_driver(snd_smdk_driver);
 173
 174MODULE_AUTHOR("Sangbeom Kim, <sbkim73@samsung.com>");
 175MODULE_DESCRIPTION("ALSA SoC SMDK WM8580 for PCM");
 176MODULE_LICENSE("GPL");
 177