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
|
/*
* 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/w2cbw003-bt.h"
static int gta04_headset_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_IB_IF | // positive sync pulse, driven on rising, sampled on falling clock
SND_SOC_DAIFMT_CBM_CFM; // clocks come from bluetooth modem - but this can be configured in the Modem chip
/* 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;
}
ret = snd_soc_dai_set_sysclk(cpu_dai, OMAP_MCBSP_SYSCLK_CLKX_EXT, 0,
SND_SOC_CLOCK_IN);
// FIXME: set clock divisor
if (ret < 0) {
printk(KERN_ERR "can't set cpu system clock\n");
return ret;
}
return 0;
}
static int gta04_headset_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_headset_startup(struct snd_pcm_substream *substream)
{
/* enable clock used by codec */
return 0;
}
static void gta04_headset_shutdown(struct snd_pcm_substream *substream)
{
/* disable clock used by codec */
}
static struct snd_soc_ops gta04_headset_ops = {
.startup = gta04_headset_startup,
.hw_params = gta04_headset_hw_params,
.shutdown = gta04_headset_shutdown,
};
/* digital headset interface glue - connects codec <--> cpu */
static struct snd_soc_dai_link gta04_headset_dai = {
.name = "W2CBW003",
.stream_name = "W2CBW003",
.cpu_dai_name = "omap-mcbsp.3",
.platform_name = "omap-pcm-audio",
.codec_dai_name = "W2CBW003",
.codec_name = "w2cbw003_codec_audio",
.init = gta04_headset_init,
.ops = >a04_headset_ops,
};
/* headset machine driver */
static struct snd_soc_card gta04_headset_card = {
.name = "gta04-headset",
.dai_link = >a04_headset_dai,
.num_links = 1,
};
static struct platform_device *gta04_headset_snd_device;
static int __init gta04_headset_soc_init(void)
{
struct device *dev;
int ret;
pr_info("gta04-headset SoC init\n");
gta04_headset_snd_device = platform_device_alloc("soc-audio", 2);
if (!gta04_headset_snd_device) {
printk(KERN_ERR "platform device allocation failed\n");
return -ENOMEM;
}
dev = >a04_headset_snd_device->dev;
platform_set_drvdata(gta04_headset_snd_device, >a04_headset_card);
ret = platform_device_add(gta04_headset_snd_device);
if (ret) {
printk(KERN_ERR "unable to add platform device\n");
platform_device_put(gta04_headset_snd_device);
}
return ret;
}
static void __exit gta04_headset_soc_exit(void)
{
platform_device_unregister(gta04_headset_snd_device);
}
module_init(gta04_headset_soc_init);
module_exit(gta04_headset_soc_exit);
MODULE_AUTHOR("John Ogness <john.ogness@linutronix.de>");
MODULE_DESCRIPTION("ALSA SoC GTA04 Headset");
MODULE_LICENSE("GPL v2");
|