blob: da9c2a264d939d47c3ab4f6f5f65b406f7423d69 [file] [log] [blame]
Sangbeom Kimd2ec3ab2011-06-20 17:29:29 +09001/*
2 * sound/soc/samsung/smdk_wm8994pcm.c
3 *
4 * Copyright (c) 2011 Samsung Electronics Co., Ltd
5 * http://www.samsung.com
6 *
7 * This program is free software; you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License as published by the
9 * Free Software Foundation; either version 2 of the License, or (at your
10 * option) any later version.
11 */
Paul Gortmakerda155d52011-07-15 12:38:28 -040012#include <linux/module.h>
Sangbeom Kimd2ec3ab2011-06-20 17:29:29 +090013#include <sound/soc.h>
14#include <sound/pcm.h>
15#include <sound/pcm_params.h>
16
17#include "../codecs/wm8994.h"
18#include "dma.h"
19#include "pcm.h"
20
21/*
22 * Board Settings:
23 * o '1' means 'ON'
24 * o '0' means 'OFF'
25 * o 'X' means 'Don't care'
26 *
27 * SMDKC210, SMDKV310: CFG3- 1001, CFG5-1000, CFG7-111111
28 */
29
30/*
31 * Configure audio route as :-
32 * $ amixer sset 'DAC1' on,on
33 * $ amixer sset 'Right Headphone Mux' 'DAC'
34 * $ amixer sset 'Left Headphone Mux' 'DAC'
35 * $ amixer sset 'DAC1R Mixer AIF1.1' on
36 * $ amixer sset 'DAC1L Mixer AIF1.1' on
37 * $ amixer sset 'IN2L' on
38 * $ amixer sset 'IN2L PGA IN2LN' on
39 * $ amixer sset 'MIXINL IN2L' on
40 * $ amixer sset 'AIF1ADC1L Mixer ADC/DMIC' on
41 * $ amixer sset 'IN2R' on
42 * $ amixer sset 'IN2R PGA IN2RN' on
43 * $ amixer sset 'MIXINR IN2R' on
44 * $ amixer sset 'AIF1ADC1R Mixer ADC/DMIC' on
45 */
46
47/* SMDK has a 16.9344MHZ crystal attached to WM8994 */
48#define SMDK_WM8994_FREQ 16934400
49
50static int smdk_wm8994_pcm_hw_params(struct snd_pcm_substream *substream,
51 struct snd_pcm_hw_params *params)
52{
53 struct snd_soc_pcm_runtime *rtd = substream->private_data;
54 struct snd_soc_dai *codec_dai = rtd->codec_dai;
55 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
56 unsigned long mclk_freq;
57 int rfs, ret;
58
59 switch(params_rate(params)) {
60 case 8000:
61 rfs = 512;
62 break;
63 default:
64 dev_err(cpu_dai->dev, "%s:%d Sampling Rate %u not supported!\n",
65 __func__, __LINE__, params_rate(params));
66 return -EINVAL;
67 }
68
69 mclk_freq = params_rate(params) * rfs;
70
71 /* Set the codec DAI configuration */
72 ret = snd_soc_dai_set_fmt(codec_dai, SND_SOC_DAIFMT_DSP_B
73 | SND_SOC_DAIFMT_IB_NF
74 | SND_SOC_DAIFMT_CBS_CFS);
75 if (ret < 0)
76 return ret;
77
78 /* Set the cpu DAI configuration */
79 ret = snd_soc_dai_set_fmt(cpu_dai, SND_SOC_DAIFMT_DSP_B
80 | SND_SOC_DAIFMT_IB_NF
81 | SND_SOC_DAIFMT_CBS_CFS);
82 if (ret < 0)
83 return ret;
84
85 ret = snd_soc_dai_set_sysclk(codec_dai, WM8994_SYSCLK_FLL1,
86 mclk_freq, SND_SOC_CLOCK_IN);
87 if (ret < 0)
88 return ret;
89
90 ret = snd_soc_dai_set_pll(codec_dai, WM8994_FLL1, WM8994_FLL_SRC_MCLK1,
91 SMDK_WM8994_FREQ, mclk_freq);
92 if (ret < 0)
93 return ret;
94
95 /* Set PCM source clock on CPU */
96 ret = snd_soc_dai_set_sysclk(cpu_dai, S3C_PCM_CLKSRC_MUX,
97 mclk_freq, SND_SOC_CLOCK_IN);
98 if (ret < 0)
99 return ret;
100
101 /* Set SCLK_DIV for making bclk */
102 ret = snd_soc_dai_set_clkdiv(cpu_dai, S3C_PCM_SCLK_PER_FS, rfs);
103 if (ret < 0)
104 return ret;
105
106 return 0;
107}
108
109static struct snd_soc_ops smdk_wm8994_pcm_ops = {
110 .hw_params = smdk_wm8994_pcm_hw_params,
111};
112
113static struct snd_soc_dai_link smdk_dai[] = {
114 {
115 .name = "WM8994 PAIF PCM",
116 .stream_name = "Primary PCM",
117 .cpu_dai_name = "samsung-pcm.0",
118 .codec_dai_name = "wm8994-aif1",
119 .platform_name = "samsung-audio",
120 .codec_name = "wm8994-codec",
121 .ops = &smdk_wm8994_pcm_ops,
122 },
123};
124
125static struct snd_soc_card smdk_pcm = {
126 .name = "SMDK-PCM",
127 .dai_link = smdk_dai,
128 .num_links = 1,
129};
130
131static int __devinit snd_smdk_probe(struct platform_device *pdev)
132{
133 int ret = 0;
134
135 smdk_pcm.dev = &pdev->dev;
136 ret = snd_soc_register_card(&smdk_pcm);
137 if (ret) {
138 dev_err(&pdev->dev, "snd_soc_register_card failed %d\n", ret);
139 return ret;
140 }
141
142 return 0;
143}
144
145static int __devexit snd_smdk_remove(struct platform_device *pdev)
146{
147 snd_soc_unregister_card(&smdk_pcm);
148 platform_set_drvdata(pdev, NULL);
149 return 0;
150}
151
152static struct platform_driver snd_smdk_driver = {
153 .driver = {
154 .owner = THIS_MODULE,
155 .name = "samsung-smdk-pcm",
156 },
157 .probe = snd_smdk_probe,
158 .remove = __devexit_p(snd_smdk_remove),
159};
160
161static int __init smdk_audio_init(void)
162{
163 return platform_driver_register(&snd_smdk_driver);
164}
165
166module_init(smdk_audio_init);
167
168static void __exit smdk_audio_exit(void)
169{
170 platform_driver_unregister(&snd_smdk_driver);
171}
172
173module_exit(smdk_audio_exit);
174
175MODULE_AUTHOR("Sangbeom Kim, <sbkim73@samsung.com>");
176MODULE_DESCRIPTION("ALSA SoC SMDK WM8994 for PCM");
177MODULE_LICENSE("GPL");