aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/omap/gta04-voice.c
blob: dcc81cf6935c0adfaf23dc04cb1492c830022ae8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
/*
 * Copyright (C) 2011 John Ogness
 *   Author: John Ogness <john.ogness@linutronix.de>
 *
 * based on sound/soc/omap/omap3beagle.c by
 *   Steve Sakoman <steve@sakoman.com>
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the
 * Free Software Foundation version 2.
 *
 * This program is distributed "as is" WITHOUT ANY WARRANTY of any kind,
 * whether express or implied; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
 * General Public License for more details.
 */

#include <linux/platform_device.h>
#include <linux/module.h>

#include <sound/core.h>
#include <sound/pcm.h>
#include <sound/soc.h>
#include <sound/soc-dapm.h>

#include "omap-mcbsp.h"
#include "omap-pcm.h"
#include "../codecs/gtm601.h"

static int gta04_voice_hw_params(struct snd_pcm_substream *substream,
				 struct snd_pcm_hw_params *params)
{
	/* setup codec dai and cpu dai hardware params */
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
//	struct snd_soc_dai *codec_dai = rtd->dai->codec_dai;
	struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
	unsigned int fmt;
	int ret;
	
	fmt =	SND_SOC_DAIFMT_I2S |	// I2S
		//	SND_SOC_DAIFMT_GATED |	// try to power down if not needed
			SND_SOC_DAIFMT_IB_IF |	// positive sync pulse, driven on rising, sampled on falling clock
			SND_SOC_DAIFMT_CBM_CFM;	// clocks come from GSM modem
	
#if 0	// set clocks to be outputs

	fmt =	SND_SOC_DAIFMT_I2S |
			SND_SOC_DAIFMT_IB_IF |
			SND_SOC_DAIFMT_CBS_CFS;

#endif	
	/* Set cpu DAI configuration */
	ret = snd_soc_dai_set_fmt(cpu_dai, fmt);
	if (ret < 0) {
		printk(KERN_ERR "can't set cpu DAI configuration\n");
		return ret;
	}
	
#if 0	// for testing we switch to output - don't use on a board with GTM601 installed!!!
	ret = snd_soc_dai_set_sysclk(cpu_dai, OMAP_MCBSP_SYSCLK_CLK, 26000000,
								 SND_SOC_CLOCK_OUT);
	// set divider...
#else
	ret = snd_soc_dai_set_sysclk(cpu_dai, OMAP_MCBSP_SYSCLK_CLKX_EXT, 0,
								 SND_SOC_CLOCK_IN);
	if(cpu_dai == 0) {
		/*
		 * could be necessary if we connect to McBSP1 - but we are on McBSP4
		 * see http://mailman.alsa-project.org/pipermail/alsa-devel/2009-August/020771.html
		 */
		if(ret >= 0)
			ret = snd_soc_dai_set_sysclk(cpu_dai, OMAP_MCBSP_CLKR_SRC_CLKX, 0,
												SND_SOC_CLOCK_IN);
		if(ret >= 0)
			ret = snd_soc_dai_set_sysclk(cpu_dai, OMAP_MCBSP_FSR_SRC_FSX, 0,
												SND_SOC_CLOCK_IN);
	}
#endif
	if (ret < 0) {
		printk(KERN_ERR "can't set cpu system clock\n");
		return ret;
	}	
		
	return 0;
}

static int gta04_voice_init(struct snd_soc_pcm_runtime *runtime)
{
	/* add controls */
	/* add routes */
	/* setup pins */
	struct snd_soc_codec *codec = runtime->codec;
	struct snd_soc_dapm_context *dapm = &codec->dapm;
	snd_soc_dapm_sync(dapm);
	return 0;
}

static int gta04_voice_startup(struct snd_pcm_substream *substream)
{
	/* enable clock used by codec */
	/* clock is provided by the GTM601 */
	return 0;
}

static void gta04_voice_shutdown(struct snd_pcm_substream *substream)
{
	/* disable clock used by codec */
	/* clock is provided by the GTM601 */
}

static struct snd_soc_ops gta04_voice_ops = {
	.startup	= gta04_voice_startup,
	.hw_params	= gta04_voice_hw_params,
	.shutdown	= gta04_voice_shutdown,
};

/* digital voice interface glue - connects codec <--> cpu */
static struct snd_soc_dai_link gta04_voice_dai = {
	.name 		= "GTM601",
	.stream_name 	= "GTM601",
// 	.cpu_dai 	= &omap_mcbsp_dai[1],
	.cpu_dai_name	= "omap-mcpdm-dai.1",
// 	.codec_dai 	= &gtm601_dai,
	.codec_dai_name = "gtm601_codec_audio",
	.init		= gta04_voice_init,
	.ops 		= &gta04_voice_ops,
};

/* voice machine driver */
static struct snd_soc_card gta04_voice_card = {
	.name		= "gta04-voice",
// 	.platform	= &omap_soc_platform,
	.dai_link	= &gta04_voice_dai,
	.num_links	= 1,
};

/* voice subsystem */
/*static struct snd_soc_device gta04_voice_devdata = {
	.card		= &gta04_voice_card,
	.codec_dev	= &soc_codec_dev_gtm601,
};*/

static struct platform_device *gta04_voice_snd_device;

static int __init gta04_voice_soc_init(void)
{
	struct device *dev;
	int ret;

	pr_info("gta04-voice SoC init\n");

	gta04_voice_snd_device = platform_device_alloc("soc-audio", 1);
	if (!gta04_voice_snd_device) {
		printk(KERN_ERR "platform device allocation failed\n");
		return -ENOMEM;
	}

	dev = &gta04_voice_snd_device->dev;

	platform_set_drvdata(gta04_voice_snd_device, &gta04_voice_card);
// 	gta04_voice_devdata.dev = &gta04_voice_snd_device->dev;
// 	*(unsigned int *)gta04_voice_dai.cpu_dai->private_data = 3;	// McBSP4

	ret = platform_device_add(gta04_voice_snd_device);
	if (ret) {
		printk(KERN_ERR "unable to add platform device\n");
		platform_device_put(gta04_voice_snd_device);
	}

	return ret;
}

static void __exit gta04_voice_soc_exit(void)
{
	platform_device_unregister(gta04_voice_snd_device);
}

module_init(gta04_voice_soc_init);
module_exit(gta04_voice_soc_exit);

MODULE_AUTHOR("John Ogness <john.ogness@linutronix.de>");
MODULE_DESCRIPTION("ALSA SoC GTA04 Voice");
MODULE_LICENSE("GPL v2");