xref: /linux/sound/soc/generic/test-component.c (revision 078f644cb81b78afdfbc42b9cc2c11959f2ed65c)
1 // SPDX-License-Identifier: GPL-2.0
2 //
3 // test-component.c  --  Test Audio Component driver
4 //
5 // Copyright (C) 2020 Renesas Electronics Corporation
6 // Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
7 
8 #include <linux/slab.h>
9 #include <linux/of.h>
10 #include <linux/of_graph.h>
11 #include <linux/module.h>
12 #include <linux/workqueue.h>
13 #include <sound/pcm.h>
14 #include <sound/soc.h>
15 
16 #define TEST_NAME_LEN 32
17 struct test_dai_name {
18 	char name[TEST_NAME_LEN];
19 	char name_playback[TEST_NAME_LEN];
20 	char name_capture[TEST_NAME_LEN];
21 };
22 
23 struct test_priv {
24 	struct device *dev;
25 	struct snd_pcm_substream *substream;
26 	struct delayed_work dwork;
27 	struct snd_soc_component_driver *component_driver;
28 	struct snd_soc_dai_driver *dai_driver;
29 	struct test_dai_name *name;
30 };
31 
32 struct test_adata {
33 	u32 is_cpu:1;
34 	u32 cmp_v:1;
35 	u32 dai_v:1;
36 };
37 
38 #define mile_stone(d)		dev_info((d)->dev, "%s() : %s", __func__, (d)->driver->name)
39 #define mile_stone_x(dev)	dev_info(dev, "%s()", __func__)
40 
41 static int test_dai_set_sysclk(struct snd_soc_dai *dai,
42 			       int clk_id, unsigned int freq, int dir)
43 {
44 	mile_stone(dai);
45 
46 	return 0;
47 }
48 
49 static int test_dai_set_pll(struct snd_soc_dai *dai, int pll_id, int source,
50 			    unsigned int freq_in, unsigned int freq_out)
51 {
52 	mile_stone(dai);
53 
54 	return 0;
55 }
56 
57 static int test_dai_set_clkdiv(struct snd_soc_dai *dai, int div_id, int div)
58 {
59 	mile_stone(dai);
60 
61 	return 0;
62 }
63 
64 static int test_dai_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
65 {
66 	unsigned int format = fmt & SND_SOC_DAIFMT_FORMAT_MASK;
67 	unsigned int clock  = fmt & SND_SOC_DAIFMT_CLOCK_MASK;
68 	unsigned int inv    = fmt & SND_SOC_DAIFMT_INV_MASK;
69 	unsigned int master = fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK;
70 	char *str;
71 
72 	dev_info(dai->dev, "name   : %s", dai->name);
73 
74 	str = "unknown";
75 	switch (format) {
76 	case SND_SOC_DAIFMT_I2S:
77 		str = "i2s";
78 		break;
79 	case SND_SOC_DAIFMT_RIGHT_J:
80 		str = "right_j";
81 		break;
82 	case SND_SOC_DAIFMT_LEFT_J:
83 		str = "left_j";
84 		break;
85 	case SND_SOC_DAIFMT_DSP_A:
86 		str = "dsp_a";
87 		break;
88 	case SND_SOC_DAIFMT_DSP_B:
89 		str = "dsp_b";
90 		break;
91 	case SND_SOC_DAIFMT_AC97:
92 		str = "ac97";
93 		break;
94 	case SND_SOC_DAIFMT_PDM:
95 		str = "pdm";
96 		break;
97 	}
98 	dev_info(dai->dev, "format : %s", str);
99 
100 	if (clock == SND_SOC_DAIFMT_CONT)
101 		str = "continuous";
102 	else
103 		str = "gated";
104 	dev_info(dai->dev, "clock  : %s", str);
105 
106 	str = "unknown";
107 	switch (master) {
108 	case SND_SOC_DAIFMT_BP_FP:
109 		str = "clk provider, frame provider";
110 		break;
111 	case SND_SOC_DAIFMT_BC_FP:
112 		str = "clk consumer, frame provider";
113 		break;
114 	case SND_SOC_DAIFMT_BP_FC:
115 		str = "clk provider, frame consumer";
116 		break;
117 	case SND_SOC_DAIFMT_BC_FC:
118 		str = "clk consumer, frame consumer";
119 		break;
120 	}
121 	dev_info(dai->dev, "clock  : codec is %s", str);
122 
123 	str = "unknown";
124 	switch (inv) {
125 	case SND_SOC_DAIFMT_NB_NF:
126 		str = "normal bit, normal frame";
127 		break;
128 	case SND_SOC_DAIFMT_NB_IF:
129 		str = "normal bit, invert frame";
130 		break;
131 	case SND_SOC_DAIFMT_IB_NF:
132 		str = "invert bit, normal frame";
133 		break;
134 	case SND_SOC_DAIFMT_IB_IF:
135 		str = "invert bit, invert frame";
136 		break;
137 	}
138 	dev_info(dai->dev, "signal : %s", str);
139 
140 	return 0;
141 }
142 
143 static int test_dai_mute_stream(struct snd_soc_dai *dai, int mute, int stream)
144 {
145 	mile_stone(dai);
146 
147 	return 0;
148 }
149 
150 static int test_dai_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
151 {
152 	mile_stone(dai);
153 
154 	return 0;
155 }
156 
157 static void test_dai_shutdown(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
158 {
159 	mile_stone(dai);
160 }
161 
162 static int test_dai_hw_params(struct snd_pcm_substream *substream,
163 			      struct snd_pcm_hw_params *params, struct snd_soc_dai *dai)
164 {
165 	mile_stone(dai);
166 
167 	return 0;
168 }
169 
170 static int test_dai_hw_free(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
171 {
172 	mile_stone(dai);
173 
174 	return 0;
175 }
176 
177 static int test_dai_trigger(struct snd_pcm_substream *substream, int cmd, struct snd_soc_dai *dai)
178 {
179 	mile_stone(dai);
180 
181 	return 0;
182 }
183 
184 static const u64 test_dai_formats =
185 	/*
186 	 * Select below from Sound Card, not auto
187 	 *	SND_SOC_POSSIBLE_DAIFMT_BP_FP
188 	 *	SND_SOC_POSSIBLE_DAIFMT_BC_FP
189 	 *	SND_SOC_POSSIBLE_DAIFMT_BP_FC
190 	 *	SND_SOC_POSSIBLE_DAIFMT_BC_FC
191 	 */
192 	SND_SOC_POSSIBLE_DAIFMT_I2S	|
193 	SND_SOC_POSSIBLE_DAIFMT_RIGHT_J	|
194 	SND_SOC_POSSIBLE_DAIFMT_LEFT_J	|
195 	SND_SOC_POSSIBLE_DAIFMT_DSP_A	|
196 	SND_SOC_POSSIBLE_DAIFMT_DSP_B	|
197 	SND_SOC_POSSIBLE_DAIFMT_AC97	|
198 	SND_SOC_POSSIBLE_DAIFMT_PDM	|
199 	SND_SOC_POSSIBLE_DAIFMT_NB_NF	|
200 	SND_SOC_POSSIBLE_DAIFMT_NB_IF	|
201 	SND_SOC_POSSIBLE_DAIFMT_IB_NF	|
202 	SND_SOC_POSSIBLE_DAIFMT_IB_IF;
203 
204 static const struct snd_soc_dai_ops test_ops = {
205 	.set_fmt		= test_dai_set_fmt,
206 	.startup		= test_dai_startup,
207 	.shutdown		= test_dai_shutdown,
208 	.auto_selectable_formats	= &test_dai_formats,
209 	.num_auto_selectable_formats	= 1,
210 };
211 
212 static const struct snd_soc_dai_ops test_verbose_ops = {
213 	.set_sysclk		= test_dai_set_sysclk,
214 	.set_pll		= test_dai_set_pll,
215 	.set_clkdiv		= test_dai_set_clkdiv,
216 	.set_fmt		= test_dai_set_fmt,
217 	.mute_stream		= test_dai_mute_stream,
218 	.startup		= test_dai_startup,
219 	.shutdown		= test_dai_shutdown,
220 	.hw_params		= test_dai_hw_params,
221 	.hw_free		= test_dai_hw_free,
222 	.trigger		= test_dai_trigger,
223 	.auto_selectable_formats	= &test_dai_formats,
224 	.num_auto_selectable_formats	= 1,
225 };
226 
227 #define STUB_RATES	SNDRV_PCM_RATE_8000_384000
228 #define STUB_FORMATS	(SNDRV_PCM_FMTBIT_S8		| \
229 			 SNDRV_PCM_FMTBIT_U8		| \
230 			 SNDRV_PCM_FMTBIT_S16_LE	| \
231 			 SNDRV_PCM_FMTBIT_U16_LE	| \
232 			 SNDRV_PCM_FMTBIT_S24_LE	| \
233 			 SNDRV_PCM_FMTBIT_S24_3LE	| \
234 			 SNDRV_PCM_FMTBIT_U24_LE	| \
235 			 SNDRV_PCM_FMTBIT_S32_LE	| \
236 			 SNDRV_PCM_FMTBIT_U32_LE)
237 
238 static int test_component_probe(struct snd_soc_component *component)
239 {
240 	mile_stone(component);
241 
242 	return 0;
243 }
244 
245 static void test_component_remove(struct snd_soc_component *component)
246 {
247 	mile_stone(component);
248 }
249 
250 static int test_component_suspend(struct snd_soc_component *component)
251 {
252 	mile_stone(component);
253 
254 	return 0;
255 }
256 
257 static int test_component_resume(struct snd_soc_component *component)
258 {
259 	mile_stone(component);
260 
261 	return 0;
262 }
263 
264 #define PREALLOC_BUFFER		(32 * 1024)
265 static int test_component_pcm_construct(struct snd_soc_component *component,
266 					struct snd_soc_pcm_runtime *rtd)
267 {
268 	mile_stone(component);
269 
270 	snd_pcm_set_managed_buffer_all(
271 		rtd->pcm,
272 		SNDRV_DMA_TYPE_DEV,
273 		rtd->card->snd_card->dev,
274 		PREALLOC_BUFFER, PREALLOC_BUFFER);
275 
276 	return 0;
277 }
278 
279 static void test_component_pcm_destruct(struct snd_soc_component *component,
280 					struct snd_pcm *pcm)
281 {
282 	mile_stone(component);
283 }
284 
285 static int test_component_set_sysclk(struct snd_soc_component *component,
286 				     int clk_id, int source, unsigned int freq, int dir)
287 {
288 	mile_stone(component);
289 
290 	return 0;
291 }
292 
293 static int test_component_set_pll(struct snd_soc_component *component, int pll_id,
294 				  int source, unsigned int freq_in, unsigned int freq_out)
295 {
296 	mile_stone(component);
297 
298 	return 0;
299 }
300 
301 static int test_component_set_jack(struct snd_soc_component *component,
302 				   struct snd_soc_jack *jack,  void *data)
303 {
304 	mile_stone(component);
305 
306 	return 0;
307 }
308 
309 static void test_component_seq_notifier(struct snd_soc_component *component,
310 					enum snd_soc_dapm_type type, int subseq)
311 {
312 	mile_stone(component);
313 }
314 
315 static int test_component_stream_event(struct snd_soc_component *component, int event)
316 {
317 	mile_stone(component);
318 
319 	return 0;
320 }
321 
322 static int test_component_set_bias_level(struct snd_soc_component *component,
323 					 enum snd_soc_bias_level level)
324 {
325 	mile_stone(component);
326 
327 	return 0;
328 }
329 
330 static const struct snd_pcm_hardware test_component_hardware = {
331 	/* Random values to keep userspace happy when checking constraints */
332 	.info			= SNDRV_PCM_INFO_INTERLEAVED	|
333 				  SNDRV_PCM_INFO_MMAP		|
334 				  SNDRV_PCM_INFO_MMAP_VALID,
335 	.buffer_bytes_max	= 32 * 1024,
336 	.period_bytes_min	= 32,
337 	.period_bytes_max	= 8192,
338 	.periods_min		= 1,
339 	.periods_max		= 128,
340 	.fifo_size		= 256,
341 };
342 
343 static int test_component_open(struct snd_soc_component *component,
344 			       struct snd_pcm_substream *substream)
345 {
346 	struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);
347 
348 	mile_stone(component);
349 
350 	/* BE's dont need dummy params */
351 	if (!rtd->dai_link->no_pcm)
352 		snd_soc_set_runtime_hwparams(substream, &test_component_hardware);
353 
354 	return 0;
355 }
356 
357 static int test_component_close(struct snd_soc_component *component,
358 				struct snd_pcm_substream *substream)
359 {
360 	mile_stone(component);
361 
362 	return 0;
363 }
364 
365 static int test_component_ioctl(struct snd_soc_component *component,
366 				struct snd_pcm_substream *substream,
367 				unsigned int cmd, void *arg)
368 {
369 	mile_stone(component);
370 
371 	return 0;
372 }
373 
374 static int test_component_hw_params(struct snd_soc_component *component,
375 				    struct snd_pcm_substream *substream,
376 				    struct snd_pcm_hw_params *params)
377 {
378 	mile_stone(component);
379 
380 	return 0;
381 }
382 
383 static int test_component_hw_free(struct snd_soc_component *component,
384 				  struct snd_pcm_substream *substream)
385 {
386 	mile_stone(component);
387 
388 	return 0;
389 }
390 
391 static int test_component_prepare(struct snd_soc_component *component,
392 				  struct snd_pcm_substream *substream)
393 {
394 	mile_stone(component);
395 
396 	return 0;
397 }
398 
399 static void test_component_timer_stop(struct test_priv *priv)
400 {
401 	cancel_delayed_work(&priv->dwork);
402 }
403 
404 static void test_component_timer_start(struct test_priv *priv)
405 {
406 	schedule_delayed_work(&priv->dwork, msecs_to_jiffies(10));
407 }
408 
409 static void test_component_dwork(struct work_struct *work)
410 {
411 	struct test_priv *priv = container_of(work, struct test_priv, dwork.work);
412 
413 	if (priv->substream)
414 		snd_pcm_period_elapsed(priv->substream);
415 
416 	test_component_timer_start(priv);
417 }
418 
419 static int test_component_trigger(struct snd_soc_component *component,
420 				  struct snd_pcm_substream *substream, int cmd)
421 {
422 	struct test_priv *priv = dev_get_drvdata(component->dev);
423 
424 	mile_stone(component);
425 
426 	switch (cmd) {
427 	case SNDRV_PCM_TRIGGER_START:
428 		test_component_timer_start(priv);
429 		priv->substream = substream; /* set substream later */
430 		break;
431 	case SNDRV_PCM_TRIGGER_STOP:
432 		priv->substream = NULL;
433 		test_component_timer_stop(priv);
434 	}
435 
436 	return 0;
437 }
438 
439 static int test_component_sync_stop(struct snd_soc_component *component,
440 				    struct snd_pcm_substream *substream)
441 {
442 	mile_stone(component);
443 
444 	return 0;
445 }
446 
447 static snd_pcm_uframes_t test_component_pointer(struct snd_soc_component *component,
448 						struct snd_pcm_substream *substream)
449 {
450 	struct snd_pcm_runtime *runtime = substream->runtime;
451 	static int pointer;
452 
453 	if (!runtime)
454 		return 0;
455 
456 	pointer += 10;
457 	if (pointer > PREALLOC_BUFFER)
458 		pointer = 0;
459 
460 	/* mile_stone(component); */
461 
462 	return bytes_to_frames(runtime, pointer);
463 }
464 
465 static int test_component_get_time_info(struct snd_soc_component *component,
466 					struct snd_pcm_substream *substream,
467 					struct timespec64 *system_ts,
468 					struct timespec64 *audio_ts,
469 					struct snd_pcm_audio_tstamp_config *audio_tstamp_config,
470 					struct snd_pcm_audio_tstamp_report *audio_tstamp_report)
471 {
472 	mile_stone(component);
473 
474 	return 0;
475 }
476 
477 static int test_component_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
478 					     struct snd_pcm_hw_params *params)
479 {
480 	mile_stone_x(rtd->dev);
481 
482 	return 0;
483 }
484 
485 /* CPU */
486 static const struct test_adata test_cpu		= { .is_cpu = 1, .cmp_v = 0, .dai_v = 0, };
487 static const struct test_adata test_cpu_vv	= { .is_cpu = 1, .cmp_v = 1, .dai_v = 1, };
488 static const struct test_adata test_cpu_nv	= { .is_cpu = 1, .cmp_v = 0, .dai_v = 1, };
489 static const struct test_adata test_cpu_vn	= { .is_cpu = 1, .cmp_v = 1, .dai_v = 0, };
490 /* Codec */
491 static const struct test_adata test_codec	= { .is_cpu = 0, .cmp_v = 0, .dai_v = 0, };
492 static const struct test_adata test_codec_vv	= { .is_cpu = 0, .cmp_v = 1, .dai_v = 1, };
493 static const struct test_adata test_codec_nv	= { .is_cpu = 0, .cmp_v = 0, .dai_v = 1, };
494 static const struct test_adata test_codec_vn	= { .is_cpu = 0, .cmp_v = 1, .dai_v = 0, };
495 
496 static const struct of_device_id test_of_match[] = {
497 	{ .compatible = "test-cpu",			.data = (void *)&test_cpu,    },
498 	{ .compatible = "test-cpu-verbose",		.data = (void *)&test_cpu_vv, },
499 	{ .compatible = "test-cpu-verbose-dai",		.data = (void *)&test_cpu_nv, },
500 	{ .compatible = "test-cpu-verbose-component",	.data = (void *)&test_cpu_vn, },
501 	{ .compatible = "test-codec",			.data = (void *)&test_codec,    },
502 	{ .compatible = "test-codec-verbose",		.data = (void *)&test_codec_vv, },
503 	{ .compatible = "test-codec-verbose-dai",	.data = (void *)&test_codec_nv, },
504 	{ .compatible = "test-codec-verbose-component",	.data = (void *)&test_codec_vn, },
505 	{},
506 };
507 MODULE_DEVICE_TABLE(of, test_of_match);
508 
509 static const struct snd_soc_dapm_widget widgets[] = {
510 	/*
511 	 * FIXME
512 	 *
513 	 * Just IN/OUT is OK for now,
514 	 * but need to be updated ?
515 	 */
516 	SND_SOC_DAPM_INPUT("IN"),
517 	SND_SOC_DAPM_OUTPUT("OUT"),
518 };
519 
520 static int test_driver_probe(struct platform_device *pdev)
521 {
522 	struct device *dev = &pdev->dev;
523 	struct device_node *node = dev->of_node;
524 	const struct test_adata *adata = of_device_get_match_data(&pdev->dev);
525 	struct snd_soc_component_driver *cdriv;
526 	struct snd_soc_dai_driver *ddriv;
527 	struct test_dai_name *dname;
528 	struct test_priv *priv;
529 	int num, ret, i;
530 
531 	num = of_graph_get_endpoint_count(node);
532 	if (!num) {
533 		dev_err(dev, "no port exits\n");
534 		return -EINVAL;
535 	}
536 
537 	priv	= devm_kzalloc(dev, sizeof(*priv),		GFP_KERNEL);
538 	cdriv	= devm_kzalloc(dev, sizeof(*cdriv),		GFP_KERNEL);
539 	ddriv	= devm_kzalloc(dev, sizeof(*ddriv) * num,	GFP_KERNEL);
540 	dname	= devm_kzalloc(dev, sizeof(*dname) * num,	GFP_KERNEL);
541 	if (!priv || !cdriv || !ddriv || !dname || !adata)
542 		return -EINVAL;
543 
544 	priv->dev		= dev;
545 	priv->component_driver	= cdriv;
546 	priv->dai_driver	= ddriv;
547 	priv->name		= dname;
548 
549 	INIT_DELAYED_WORK(&priv->dwork, test_component_dwork);
550 	dev_set_drvdata(dev, priv);
551 
552 	if (adata->is_cpu) {
553 		cdriv->name			= "test_cpu";
554 		cdriv->pcm_construct		= test_component_pcm_construct;
555 		cdriv->pointer			= test_component_pointer;
556 		cdriv->trigger			= test_component_trigger;
557 		cdriv->legacy_dai_naming	= 1;
558 	} else {
559 		cdriv->name			= "test_codec";
560 		cdriv->idle_bias_on		= 1;
561 		cdriv->endianness		= 1;
562 	}
563 
564 	cdriv->open		= test_component_open;
565 	cdriv->dapm_widgets	= widgets;
566 	cdriv->num_dapm_widgets	= ARRAY_SIZE(widgets);
567 
568 	if (adata->cmp_v) {
569 		cdriv->probe			= test_component_probe;
570 		cdriv->remove			= test_component_remove;
571 		cdriv->suspend			= test_component_suspend;
572 		cdriv->resume			= test_component_resume;
573 		cdriv->set_sysclk		= test_component_set_sysclk;
574 		cdriv->set_pll			= test_component_set_pll;
575 		cdriv->set_jack			= test_component_set_jack;
576 		cdriv->seq_notifier		= test_component_seq_notifier;
577 		cdriv->stream_event		= test_component_stream_event;
578 		cdriv->set_bias_level		= test_component_set_bias_level;
579 		cdriv->close			= test_component_close;
580 		cdriv->ioctl			= test_component_ioctl;
581 		cdriv->hw_params		= test_component_hw_params;
582 		cdriv->hw_free			= test_component_hw_free;
583 		cdriv->prepare			= test_component_prepare;
584 		cdriv->sync_stop		= test_component_sync_stop;
585 		cdriv->get_time_info		= test_component_get_time_info;
586 		cdriv->be_hw_params_fixup	= test_component_be_hw_params_fixup;
587 
588 		if (adata->is_cpu)
589 			cdriv->pcm_destruct	= test_component_pcm_destruct;
590 	}
591 
592 	i = 0;
593 	for_each_of_graph_port(node, port) {
594 		snprintf(dname[i].name, TEST_NAME_LEN, "%s.%d", node->name, i);
595 		ddriv[i].name = dname[i].name;
596 
597 		snprintf(dname[i].name_playback, TEST_NAME_LEN, "DAI%d Playback", i);
598 		ddriv[i].playback.stream_name	= dname[i].name_playback;
599 		ddriv[i].playback.channels_min	= 1;
600 		ddriv[i].playback.channels_max	= 384;
601 		ddriv[i].playback.rates		= STUB_RATES;
602 		ddriv[i].playback.formats	= STUB_FORMATS;
603 
604 		snprintf(dname[i].name_capture, TEST_NAME_LEN, "DAI%d Capture", i);
605 		ddriv[i].capture.stream_name	= dname[i].name_capture;
606 		ddriv[i].capture.channels_min	= 1;
607 		ddriv[i].capture.channels_max	= 384;
608 		ddriv[i].capture.rates		= STUB_RATES;
609 		ddriv[i].capture.formats	= STUB_FORMATS;
610 
611 		if (adata->dai_v)
612 			ddriv[i].ops = &test_verbose_ops;
613 		else
614 			ddriv[i].ops = &test_ops;
615 
616 		i++;
617 	}
618 
619 	ret = devm_snd_soc_register_component(dev, cdriv, ddriv, num);
620 	if (ret < 0)
621 		return ret;
622 
623 	mile_stone_x(dev);
624 
625 	return 0;
626 }
627 
628 static void test_driver_remove(struct platform_device *pdev)
629 {
630 	mile_stone_x(&pdev->dev);
631 }
632 
633 static struct platform_driver test_driver = {
634 	.driver = {
635 		.name = "test-component",
636 		.of_match_table = test_of_match,
637 	},
638 	.probe  = test_driver_probe,
639 	.remove = test_driver_remove,
640 };
641 module_platform_driver(test_driver);
642 
643 MODULE_ALIAS("platform:asoc-test-component");
644 MODULE_AUTHOR("Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>");
645 MODULE_DESCRIPTION("ASoC Test Component");
646 MODULE_LICENSE("GPL v2");
647