blob: c6cbd81fdbceb996da1cba50d058c1dc9978c07c [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Simon Glass279fc162014-02-27 13:26:18 -07002/*
3 * Copyright (c) 2013 Google, Inc
Simon Glass279fc162014-02-27 13:26:18 -07004 */
5
Simon Glass4d5814c2019-02-16 20:24:56 -07006#define LOG_CATEGORY UCLASS_SOUND
7
Tom Riniabb9a042024-05-18 20:20:43 -06008#include <common.h>
Simon Glassed96cde2018-12-10 10:37:33 -07009#include <audio_codec.h>
Simon Glassc953aaf2018-12-10 10:37:34 -070010#include <dm.h>
11#include <i2s.h>
Simon Glass0f2af882020-05-10 11:40:05 -060012#include <log.h>
Simon Glass76072ac2018-12-10 10:37:36 -070013#include <sound.h>
Simon Glass279fc162014-02-27 13:26:18 -070014#include <asm/sdl.h>
15
Simon Glassed96cde2018-12-10 10:37:33 -070016struct sandbox_codec_priv {
17 int interface;
18 int rate;
19 int mclk_freq;
20 int bits_per_sample;
21 uint channels;
22};
23
Simon Glassc953aaf2018-12-10 10:37:34 -070024struct sandbox_i2s_priv {
25 int sum; /* Use to sum the provided audio data */
Simon Glass4d5814c2019-02-16 20:24:56 -070026 bool silent; /* Sound is silent, don't use SDL */
Simon Glassc953aaf2018-12-10 10:37:34 -070027};
28
Simon Glass76072ac2018-12-10 10:37:36 -070029struct sandbox_sound_priv {
Simon Glass70bc14b2020-02-03 07:36:06 -070030 int setup_called; /* Incremented when setup() method is called */
31 bool active; /* TX data is being sent */
Heinrich Schuchardt707a6fc2022-12-04 17:11:41 +010032 int count; /* Use to count the provided audio data */
Simon Glassecd02e72019-02-16 20:24:54 -070033 int sum; /* Use to sum the provided audio data */
34 bool allow_beep; /* true to allow the start_beep() interface */
35 int frequency_hz; /* Beep frequency if active, else 0 */
Simon Glass76072ac2018-12-10 10:37:36 -070036};
37
Simon Glassed96cde2018-12-10 10:37:33 -070038void sandbox_get_codec_params(struct udevice *dev, int *interfacep, int *ratep,
39 int *mclk_freqp, int *bits_per_samplep,
40 uint *channelsp)
41{
42 struct sandbox_codec_priv *priv = dev_get_priv(dev);
43
44 *interfacep = priv->interface;
45 *ratep = priv->rate;
46 *mclk_freqp = priv->mclk_freq;
47 *bits_per_samplep = priv->bits_per_sample;
48 *channelsp = priv->channels;
49}
50
Simon Glassc953aaf2018-12-10 10:37:34 -070051int sandbox_get_i2s_sum(struct udevice *dev)
52{
53 struct sandbox_i2s_priv *priv = dev_get_priv(dev);
54
55 return priv->sum;
56}
57
Simon Glass76072ac2018-12-10 10:37:36 -070058int sandbox_get_setup_called(struct udevice *dev)
59{
60 struct sandbox_sound_priv *priv = dev_get_priv(dev);
61
62 return priv->setup_called;
63}
64
Simon Glass70bc14b2020-02-03 07:36:06 -070065int sandbox_get_sound_active(struct udevice *dev)
66{
67 struct sandbox_sound_priv *priv = dev_get_priv(dev);
68
69 return priv->active;
70}
71
Heinrich Schuchardt707a6fc2022-12-04 17:11:41 +010072int sandbox_get_sound_count(struct udevice *dev)
73{
74 struct sandbox_sound_priv *priv = dev_get_priv(dev);
75
76 return priv->count;
77}
78
Simon Glass76072ac2018-12-10 10:37:36 -070079int sandbox_get_sound_sum(struct udevice *dev)
80{
81 struct sandbox_sound_priv *priv = dev_get_priv(dev);
82
83 return priv->sum;
84}
85
Simon Glassecd02e72019-02-16 20:24:54 -070086void sandbox_set_allow_beep(struct udevice *dev, bool allow)
87{
88 struct sandbox_sound_priv *priv = dev_get_priv(dev);
89
90 priv->allow_beep = allow;
91}
92
93int sandbox_get_beep_frequency(struct udevice *dev)
94{
95 struct sandbox_sound_priv *priv = dev_get_priv(dev);
96
97 return priv->frequency_hz;
98}
99
Simon Glassed96cde2018-12-10 10:37:33 -0700100static int sandbox_codec_set_params(struct udevice *dev, int interface,
101 int rate, int mclk_freq,
102 int bits_per_sample, uint channels)
103{
104 struct sandbox_codec_priv *priv = dev_get_priv(dev);
105
106 priv->interface = interface;
107 priv->rate = rate;
108 priv->mclk_freq = mclk_freq;
109 priv->bits_per_sample = bits_per_sample;
110 priv->channels = channels;
111
112 return 0;
113}
114
Simon Glassc953aaf2018-12-10 10:37:34 -0700115static int sandbox_i2s_tx_data(struct udevice *dev, void *data,
116 uint data_size)
117{
118 struct sandbox_i2s_priv *priv = dev_get_priv(dev);
119 int i;
120
121 for (i = 0; i < data_size; i++)
122 priv->sum += ((uint8_t *)data)[i];
123
Simon Glass4d5814c2019-02-16 20:24:56 -0700124 if (!priv->silent) {
125 int ret;
126
127 ret = sandbox_sdl_sound_play(data, data_size);
128 if (ret)
129 return ret;
130 }
131
132 return 0;
Simon Glassc953aaf2018-12-10 10:37:34 -0700133}
134
135static int sandbox_i2s_probe(struct udevice *dev)
136{
137 struct i2s_uc_priv *uc_priv = dev_get_uclass_priv(dev);
Simon Glass4d5814c2019-02-16 20:24:56 -0700138 struct sandbox_i2s_priv *priv = dev_get_priv(dev);
Simon Glassc953aaf2018-12-10 10:37:34 -0700139
140 /* Use hard-coded values here */
141 uc_priv->rfs = 256;
142 uc_priv->bfs = 32;
143 uc_priv->audio_pll_clk = 192000000;
144 uc_priv->samplingrate = 48000;
145 uc_priv->bitspersample = 16;
146 uc_priv->channels = 2;
147 uc_priv->id = 1;
148
Simon Glass4d5814c2019-02-16 20:24:56 -0700149 priv->silent = dev_read_bool(dev, "sandbox,silent");
150
151 if (priv->silent) {
152 log_warning("Sound is silenced\n");
153 } else if (sandbox_sdl_sound_init(uc_priv->samplingrate,
154 uc_priv->channels)) {
155 /* Ignore any error here - we'll just have no sound */
156 priv->silent = true;
157 }
Simon Glass4070ba62018-12-10 10:37:39 -0700158
159 return 0;
Simon Glass76072ac2018-12-10 10:37:36 -0700160}
161
162static int sandbox_sound_setup(struct udevice *dev)
163{
164 struct sandbox_sound_priv *priv = dev_get_priv(dev);
165
166 priv->setup_called++;
167
Simon Glassc953aaf2018-12-10 10:37:34 -0700168 return 0;
169}
170
Simon Glass76072ac2018-12-10 10:37:36 -0700171static int sandbox_sound_play(struct udevice *dev, void *data, uint data_size)
172{
173 struct sound_uc_priv *uc_priv = dev_get_uclass_priv(dev);
174 struct sandbox_sound_priv *priv = dev_get_priv(dev);
175 int i;
176
177 for (i = 0; i < data_size; i++)
178 priv->sum += ((uint8_t *)data)[i];
Heinrich Schuchardt707a6fc2022-12-04 17:11:41 +0100179 priv->count += data_size;
Simon Glass76072ac2018-12-10 10:37:36 -0700180
181 return i2s_tx_data(uc_priv->i2s, data, data_size);
182}
183
Simon Glass70bc14b2020-02-03 07:36:06 -0700184static int sandbox_sound_stop_play(struct udevice *dev)
185{
186 struct sandbox_sound_priv *priv = dev_get_priv(dev);
187
188 sandbox_sdl_sound_stop();
189 priv->active = false;
190
191 return 0;
192}
193
Simon Glassecd02e72019-02-16 20:24:54 -0700194int sandbox_sound_start_beep(struct udevice *dev, int frequency_hz)
195{
196 struct sandbox_sound_priv *priv = dev_get_priv(dev);
197
198 if (!priv->allow_beep)
199 return -ENOSYS;
200 priv->frequency_hz = frequency_hz;
201
202 return 0;
203}
204
205int sandbox_sound_stop_beep(struct udevice *dev)
206{
207 struct sandbox_sound_priv *priv = dev_get_priv(dev);
208
209 if (!priv->allow_beep)
210 return -ENOSYS;
211 priv->frequency_hz = 0;
212
213 return 0;
214}
215
Simon Glass76072ac2018-12-10 10:37:36 -0700216static int sandbox_sound_probe(struct udevice *dev)
217{
218 return sound_find_codec_i2s(dev);
219}
220
Simon Glassed96cde2018-12-10 10:37:33 -0700221static const struct audio_codec_ops sandbox_codec_ops = {
222 .set_params = sandbox_codec_set_params,
223};
224
225static const struct udevice_id sandbox_codec_ids[] = {
226 { .compatible = "sandbox,audio-codec" },
227 { }
228};
229
230U_BOOT_DRIVER(sandbox_codec) = {
231 .name = "sandbox_codec",
232 .id = UCLASS_AUDIO_CODEC,
233 .of_match = sandbox_codec_ids,
234 .ops = &sandbox_codec_ops,
Simon Glass8a2b47f2020-12-03 16:55:17 -0700235 .priv_auto = sizeof(struct sandbox_codec_priv),
Simon Glassed96cde2018-12-10 10:37:33 -0700236};
Simon Glassc953aaf2018-12-10 10:37:34 -0700237
238static const struct i2s_ops sandbox_i2s_ops = {
239 .tx_data = sandbox_i2s_tx_data,
240};
241
242static const struct udevice_id sandbox_i2s_ids[] = {
243 { .compatible = "sandbox,i2s" },
244 { }
245};
246
247U_BOOT_DRIVER(sandbox_i2s) = {
248 .name = "sandbox_i2s",
249 .id = UCLASS_I2S,
250 .of_match = sandbox_i2s_ids,
251 .ops = &sandbox_i2s_ops,
252 .probe = sandbox_i2s_probe,
Simon Glass8a2b47f2020-12-03 16:55:17 -0700253 .priv_auto = sizeof(struct sandbox_i2s_priv),
Simon Glassc953aaf2018-12-10 10:37:34 -0700254};
Simon Glass76072ac2018-12-10 10:37:36 -0700255
256static const struct sound_ops sandbox_sound_ops = {
Simon Glassecd02e72019-02-16 20:24:54 -0700257 .setup = sandbox_sound_setup,
258 .play = sandbox_sound_play,
Simon Glass70bc14b2020-02-03 07:36:06 -0700259 .stop_play = sandbox_sound_stop_play,
Simon Glassecd02e72019-02-16 20:24:54 -0700260 .start_beep = sandbox_sound_start_beep,
261 .stop_beep = sandbox_sound_stop_beep,
Simon Glass76072ac2018-12-10 10:37:36 -0700262};
263
264static const struct udevice_id sandbox_sound_ids[] = {
265 { .compatible = "sandbox,sound" },
266 { }
267};
268
269U_BOOT_DRIVER(sandbox_sound) = {
270 .name = "sandbox_sound",
271 .id = UCLASS_SOUND,
272 .of_match = sandbox_sound_ids,
273 .ops = &sandbox_sound_ops,
Simon Glass8a2b47f2020-12-03 16:55:17 -0700274 .priv_auto = sizeof(struct sandbox_sound_priv),
Simon Glass76072ac2018-12-10 10:37:36 -0700275 .probe = sandbox_sound_probe,
276};