blob: 363c687bafd11d2ebcf16ba3dda3c69157352c35 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Simon Glassc34c0242014-02-27 13:26:18 -07002/*
3 * Copyright (c) 2013 Google, Inc
Simon Glassc34c0242014-02-27 13:26:18 -07004 */
5
Simon Glassecc79732019-02-16 20:24:56 -07006#define LOG_CATEGORY UCLASS_SOUND
7
Simon Glassc34c0242014-02-27 13:26:18 -07008#include <common.h>
Simon Glassce6d99a2018-12-10 10:37:33 -07009#include <audio_codec.h>
Simon Glasse96fa6c2018-12-10 10:37:34 -070010#include <dm.h>
11#include <i2s.h>
Simon Glassd4901892018-12-10 10:37:36 -070012#include <sound.h>
Simon Glassc34c0242014-02-27 13:26:18 -070013#include <asm/sdl.h>
14
Simon Glassce6d99a2018-12-10 10:37:33 -070015struct sandbox_codec_priv {
16 int interface;
17 int rate;
18 int mclk_freq;
19 int bits_per_sample;
20 uint channels;
21};
22
Simon Glasse96fa6c2018-12-10 10:37:34 -070023struct sandbox_i2s_priv {
24 int sum; /* Use to sum the provided audio data */
Simon Glassecc79732019-02-16 20:24:56 -070025 bool silent; /* Sound is silent, don't use SDL */
Simon Glasse96fa6c2018-12-10 10:37:34 -070026};
27
Simon Glassd4901892018-12-10 10:37:36 -070028struct sandbox_sound_priv {
29 int setup_called;
Simon Glass28502662019-02-16 20:24:54 -070030 int sum; /* Use to sum the provided audio data */
31 bool allow_beep; /* true to allow the start_beep() interface */
32 int frequency_hz; /* Beep frequency if active, else 0 */
Simon Glassd4901892018-12-10 10:37:36 -070033};
34
Simon Glassce6d99a2018-12-10 10:37:33 -070035void sandbox_get_codec_params(struct udevice *dev, int *interfacep, int *ratep,
36 int *mclk_freqp, int *bits_per_samplep,
37 uint *channelsp)
38{
39 struct sandbox_codec_priv *priv = dev_get_priv(dev);
40
41 *interfacep = priv->interface;
42 *ratep = priv->rate;
43 *mclk_freqp = priv->mclk_freq;
44 *bits_per_samplep = priv->bits_per_sample;
45 *channelsp = priv->channels;
46}
47
Simon Glasse96fa6c2018-12-10 10:37:34 -070048int sandbox_get_i2s_sum(struct udevice *dev)
49{
50 struct sandbox_i2s_priv *priv = dev_get_priv(dev);
51
52 return priv->sum;
53}
54
Simon Glassd4901892018-12-10 10:37:36 -070055int sandbox_get_setup_called(struct udevice *dev)
56{
57 struct sandbox_sound_priv *priv = dev_get_priv(dev);
58
59 return priv->setup_called;
60}
61
62int sandbox_get_sound_sum(struct udevice *dev)
63{
64 struct sandbox_sound_priv *priv = dev_get_priv(dev);
65
66 return priv->sum;
67}
68
Simon Glass28502662019-02-16 20:24:54 -070069void sandbox_set_allow_beep(struct udevice *dev, bool allow)
70{
71 struct sandbox_sound_priv *priv = dev_get_priv(dev);
72
73 priv->allow_beep = allow;
74}
75
76int sandbox_get_beep_frequency(struct udevice *dev)
77{
78 struct sandbox_sound_priv *priv = dev_get_priv(dev);
79
80 return priv->frequency_hz;
81}
82
Simon Glassce6d99a2018-12-10 10:37:33 -070083static int sandbox_codec_set_params(struct udevice *dev, int interface,
84 int rate, int mclk_freq,
85 int bits_per_sample, uint channels)
86{
87 struct sandbox_codec_priv *priv = dev_get_priv(dev);
88
89 priv->interface = interface;
90 priv->rate = rate;
91 priv->mclk_freq = mclk_freq;
92 priv->bits_per_sample = bits_per_sample;
93 priv->channels = channels;
94
95 return 0;
96}
97
Simon Glasse96fa6c2018-12-10 10:37:34 -070098static int sandbox_i2s_tx_data(struct udevice *dev, void *data,
99 uint data_size)
100{
101 struct sandbox_i2s_priv *priv = dev_get_priv(dev);
102 int i;
103
104 for (i = 0; i < data_size; i++)
105 priv->sum += ((uint8_t *)data)[i];
106
Simon Glassecc79732019-02-16 20:24:56 -0700107 if (!priv->silent) {
108 int ret;
109
110 ret = sandbox_sdl_sound_play(data, data_size);
111 if (ret)
112 return ret;
113 }
114
115 return 0;
Simon Glasse96fa6c2018-12-10 10:37:34 -0700116}
117
118static int sandbox_i2s_probe(struct udevice *dev)
119{
120 struct i2s_uc_priv *uc_priv = dev_get_uclass_priv(dev);
Simon Glassecc79732019-02-16 20:24:56 -0700121 struct sandbox_i2s_priv *priv = dev_get_priv(dev);
Simon Glasse96fa6c2018-12-10 10:37:34 -0700122
123 /* Use hard-coded values here */
124 uc_priv->rfs = 256;
125 uc_priv->bfs = 32;
126 uc_priv->audio_pll_clk = 192000000;
127 uc_priv->samplingrate = 48000;
128 uc_priv->bitspersample = 16;
129 uc_priv->channels = 2;
130 uc_priv->id = 1;
131
Simon Glassecc79732019-02-16 20:24:56 -0700132 priv->silent = dev_read_bool(dev, "sandbox,silent");
133
134 if (priv->silent) {
135 log_warning("Sound is silenced\n");
136 } else if (sandbox_sdl_sound_init(uc_priv->samplingrate,
137 uc_priv->channels)) {
138 /* Ignore any error here - we'll just have no sound */
139 priv->silent = true;
140 }
Simon Glassd6cadd52018-12-10 10:37:39 -0700141
142 return 0;
Simon Glassd4901892018-12-10 10:37:36 -0700143}
144
145static int sandbox_sound_setup(struct udevice *dev)
146{
147 struct sandbox_sound_priv *priv = dev_get_priv(dev);
148
149 priv->setup_called++;
150
Simon Glasse96fa6c2018-12-10 10:37:34 -0700151 return 0;
152}
153
Simon Glassd4901892018-12-10 10:37:36 -0700154static int sandbox_sound_play(struct udevice *dev, void *data, uint data_size)
155{
156 struct sound_uc_priv *uc_priv = dev_get_uclass_priv(dev);
157 struct sandbox_sound_priv *priv = dev_get_priv(dev);
158 int i;
159
160 for (i = 0; i < data_size; i++)
161 priv->sum += ((uint8_t *)data)[i];
162
163 return i2s_tx_data(uc_priv->i2s, data, data_size);
164}
165
Simon Glass28502662019-02-16 20:24:54 -0700166int sandbox_sound_start_beep(struct udevice *dev, int frequency_hz)
167{
168 struct sandbox_sound_priv *priv = dev_get_priv(dev);
169
170 if (!priv->allow_beep)
171 return -ENOSYS;
172 priv->frequency_hz = frequency_hz;
173
174 return 0;
175}
176
177int sandbox_sound_stop_beep(struct udevice *dev)
178{
179 struct sandbox_sound_priv *priv = dev_get_priv(dev);
180
181 if (!priv->allow_beep)
182 return -ENOSYS;
183 priv->frequency_hz = 0;
184
185 return 0;
186}
187
Simon Glassd4901892018-12-10 10:37:36 -0700188static int sandbox_sound_probe(struct udevice *dev)
189{
190 return sound_find_codec_i2s(dev);
191}
192
Simon Glassce6d99a2018-12-10 10:37:33 -0700193static const struct audio_codec_ops sandbox_codec_ops = {
194 .set_params = sandbox_codec_set_params,
195};
196
197static const struct udevice_id sandbox_codec_ids[] = {
198 { .compatible = "sandbox,audio-codec" },
199 { }
200};
201
202U_BOOT_DRIVER(sandbox_codec) = {
203 .name = "sandbox_codec",
204 .id = UCLASS_AUDIO_CODEC,
205 .of_match = sandbox_codec_ids,
206 .ops = &sandbox_codec_ops,
207 .priv_auto_alloc_size = sizeof(struct sandbox_codec_priv),
208};
Simon Glasse96fa6c2018-12-10 10:37:34 -0700209
210static const struct i2s_ops sandbox_i2s_ops = {
211 .tx_data = sandbox_i2s_tx_data,
212};
213
214static const struct udevice_id sandbox_i2s_ids[] = {
215 { .compatible = "sandbox,i2s" },
216 { }
217};
218
219U_BOOT_DRIVER(sandbox_i2s) = {
220 .name = "sandbox_i2s",
221 .id = UCLASS_I2S,
222 .of_match = sandbox_i2s_ids,
223 .ops = &sandbox_i2s_ops,
224 .probe = sandbox_i2s_probe,
225 .priv_auto_alloc_size = sizeof(struct sandbox_i2s_priv),
226};
Simon Glassd4901892018-12-10 10:37:36 -0700227
228static const struct sound_ops sandbox_sound_ops = {
Simon Glass28502662019-02-16 20:24:54 -0700229 .setup = sandbox_sound_setup,
230 .play = sandbox_sound_play,
231 .start_beep = sandbox_sound_start_beep,
232 .stop_beep = sandbox_sound_stop_beep,
Simon Glassd4901892018-12-10 10:37:36 -0700233};
234
235static const struct udevice_id sandbox_sound_ids[] = {
236 { .compatible = "sandbox,sound" },
237 { }
238};
239
240U_BOOT_DRIVER(sandbox_sound) = {
241 .name = "sandbox_sound",
242 .id = UCLASS_SOUND,
243 .of_match = sandbox_sound_ids,
244 .ops = &sandbox_sound_ops,
245 .priv_auto_alloc_size = sizeof(struct sandbox_sound_priv),
246 .probe = sandbox_sound_probe,
247};