1// SPDX-License-Identifier: GPL-2.0-only
2//
3// sdw-mockup.c -- a mockup SoundWire codec for tests where only the host
4// drives the bus.
5//
6// Copyright(c) 2021 Intel Corporation
7//
8//
9
10#include <linux/device.h>
11#include <linux/mod_devicetable.h>
12#include <linux/module.h>
13#include <linux/soundwire/sdw.h>
14#include <linux/soundwire/sdw_type.h>
15#include <linux/soundwire/sdw_registers.h>
16#include <sound/core.h>
17#include <sound/pcm.h>
18#include <sound/pcm_params.h>
19#include <sound/sdw.h>
20#include <sound/soc.h>
21
22struct  sdw_mockup_priv {
23	struct sdw_slave *slave;
24};
25
26static int sdw_mockup_component_probe(struct snd_soc_component *component)
27{
28	return 0;
29}
30
31static void sdw_mockup_component_remove(struct snd_soc_component *component)
32{
33}
34
35static const struct snd_soc_component_driver snd_soc_sdw_mockup_component = {
36	.probe = sdw_mockup_component_probe,
37	.remove = sdw_mockup_component_remove,
38	.endianness = 1,
39};
40
41static int sdw_mockup_set_sdw_stream(struct snd_soc_dai *dai, void *sdw_stream,
42				     int direction)
43{
44	snd_soc_dai_dma_data_set(dai, direction, sdw_stream);
45
46	return 0;
47}
48
49static void sdw_mockup_shutdown(struct snd_pcm_substream *substream,
50				struct snd_soc_dai *dai)
51{
52	snd_soc_dai_set_dma_data(dai, substream, NULL);
53}
54
55static int sdw_mockup_pcm_hw_params(struct snd_pcm_substream *substream,
56				    struct snd_pcm_hw_params *params,
57				    struct snd_soc_dai *dai)
58{
59	struct snd_soc_component *component = dai->component;
60	struct sdw_mockup_priv *sdw_mockup = snd_soc_component_get_drvdata(component);
61	struct sdw_stream_config stream_config = {0};
62	struct sdw_port_config port_config = {0};
63	struct sdw_stream_runtime *sdw_stream = snd_soc_dai_get_dma_data(dai, substream);
64	int ret;
65
66	if (!sdw_stream)
67		return -EINVAL;
68
69	if (!sdw_mockup->slave)
70		return -EINVAL;
71
72	/* SoundWire specific configuration */
73	snd_sdw_params_to_config(substream, params, &stream_config, &port_config);
74
75	if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
76		port_config.num = 1;
77	else
78		port_config.num = 8;
79
80	ret = sdw_stream_add_slave(sdw_mockup->slave, &stream_config,
81				   &port_config, 1, sdw_stream);
82	if (ret)
83		dev_err(dai->dev, "Unable to configure port\n");
84
85	return ret;
86}
87
88static int sdw_mockup_pcm_hw_free(struct snd_pcm_substream *substream,
89				  struct snd_soc_dai *dai)
90{
91	struct snd_soc_component *component = dai->component;
92	struct sdw_mockup_priv *sdw_mockup = snd_soc_component_get_drvdata(component);
93	struct sdw_stream_runtime *sdw_stream = snd_soc_dai_get_dma_data(dai, substream);
94
95	if (!sdw_mockup->slave)
96		return -EINVAL;
97
98	sdw_stream_remove_slave(sdw_mockup->slave, sdw_stream);
99	return 0;
100}
101
102static const struct snd_soc_dai_ops sdw_mockup_ops = {
103	.hw_params	= sdw_mockup_pcm_hw_params,
104	.hw_free	= sdw_mockup_pcm_hw_free,
105	.set_stream	= sdw_mockup_set_sdw_stream,
106	.shutdown	= sdw_mockup_shutdown,
107};
108
109static struct snd_soc_dai_driver sdw_mockup_dai[] = {
110	{
111		.name = "sdw-mockup-aif1",
112		.id = 1,
113		.playback = {
114			.stream_name = "DP1 Playback",
115			.channels_min = 1,
116			.channels_max = 2,
117		},
118		.capture = {
119			.stream_name = "DP8 Capture",
120			.channels_min = 1,
121			.channels_max = 2,
122		},
123		.ops = &sdw_mockup_ops,
124	},
125};
126
127static int sdw_mockup_update_status(struct sdw_slave *slave,
128				    enum sdw_slave_status status)
129{
130	return 0;
131}
132
133static int sdw_mockup_read_prop(struct sdw_slave *slave)
134{
135	struct sdw_slave_prop *prop = &slave->prop;
136	int nval;
137	int i, j;
138	u32 bit;
139	unsigned long addr;
140	struct sdw_dpn_prop *dpn;
141
142	prop->paging_support = false;
143
144	/*
145	 * first we need to allocate memory for set bits in port lists
146	 * the port allocation is completely arbitrary:
147	 * DP0 is not supported
148	 * DP1 is sink
149	 * DP8 is source
150	 */
151	prop->source_ports = BIT(8);
152	prop->sink_ports = BIT(1);
153
154	nval = hweight32(prop->source_ports);
155	prop->src_dpn_prop = devm_kcalloc(&slave->dev, nval,
156					  sizeof(*prop->src_dpn_prop),
157					  GFP_KERNEL);
158	if (!prop->src_dpn_prop)
159		return -ENOMEM;
160
161	i = 0;
162	dpn = prop->src_dpn_prop;
163	addr = prop->source_ports;
164	for_each_set_bit(bit, &addr, 32) {
165		dpn[i].num = bit;
166		dpn[i].type = SDW_DPN_FULL;
167		dpn[i].simple_ch_prep_sm = true;
168		i++;
169	}
170
171	/* do this again for sink now */
172	nval = hweight32(prop->sink_ports);
173	prop->sink_dpn_prop = devm_kcalloc(&slave->dev, nval,
174					   sizeof(*prop->sink_dpn_prop),
175					   GFP_KERNEL);
176	if (!prop->sink_dpn_prop)
177		return -ENOMEM;
178
179	j = 0;
180	dpn = prop->sink_dpn_prop;
181	addr = prop->sink_ports;
182	for_each_set_bit(bit, &addr, 32) {
183		dpn[j].num = bit;
184		dpn[j].type = SDW_DPN_FULL;
185		dpn[j].simple_ch_prep_sm = true;
186		j++;
187	}
188
189	prop->simple_clk_stop_capable = true;
190
191	/* wake-up event */
192	prop->wake_capable = 0;
193
194	return 0;
195}
196
197static int sdw_mockup_bus_config(struct sdw_slave *slave,
198				 struct sdw_bus_params *params)
199{
200	return 0;
201}
202
203static int sdw_mockup_interrupt_callback(struct sdw_slave *slave,
204					 struct sdw_slave_intr_status *status)
205{
206	return 0;
207}
208
209static const struct sdw_slave_ops sdw_mockup_slave_ops = {
210	.read_prop = sdw_mockup_read_prop,
211	.interrupt_callback = sdw_mockup_interrupt_callback,
212	.update_status = sdw_mockup_update_status,
213	.bus_config = sdw_mockup_bus_config,
214};
215
216static int sdw_mockup_sdw_probe(struct sdw_slave *slave,
217				const struct sdw_device_id *id)
218{
219	struct device *dev = &slave->dev;
220	struct sdw_mockup_priv *sdw_mockup;
221	int ret;
222
223	sdw_mockup = devm_kzalloc(dev, sizeof(*sdw_mockup), GFP_KERNEL);
224	if (!sdw_mockup)
225		return -ENOMEM;
226
227	dev_set_drvdata(dev, sdw_mockup);
228	sdw_mockup->slave = slave;
229
230	slave->is_mockup_device = true;
231
232	ret =  devm_snd_soc_register_component(dev,
233					       &snd_soc_sdw_mockup_component,
234					       sdw_mockup_dai,
235					       ARRAY_SIZE(sdw_mockup_dai));
236
237	return ret;
238}
239
240static int sdw_mockup_sdw_remove(struct sdw_slave *slave)
241{
242	return 0;
243}
244
245/*
246 * Intel reserved parts ID with the following mapping expected:
247 * 0xAAAA: generic full-duplex codec
248 * 0xAA55: headset codec (mock-up of RT711/RT5682) - full-duplex
249 * 0x55AA: amplifier (mock-up of RT1308/Maxim 98373) - playback only with
250 * IV feedback
251 * 0x5555: mic codec (mock-up of RT715) - capture-only
252 */
253static const struct sdw_device_id sdw_mockup_id[] = {
254	SDW_SLAVE_ENTRY_EXT(0x0105, 0xAAAA, 0x0, 0, 0),
255	SDW_SLAVE_ENTRY_EXT(0x0105, 0xAA55, 0x0, 0, 0),
256	SDW_SLAVE_ENTRY_EXT(0x0105, 0x55AA, 0x0, 0, 0),
257	SDW_SLAVE_ENTRY_EXT(0x0105, 0x5555, 0x0, 0, 0),
258	{},
259};
260MODULE_DEVICE_TABLE(sdw, sdw_mockup_id);
261
262static struct sdw_driver sdw_mockup_sdw_driver = {
263	.driver = {
264		.name = "sdw-mockup",
265		.owner = THIS_MODULE,
266	},
267	.probe = sdw_mockup_sdw_probe,
268	.remove = sdw_mockup_sdw_remove,
269	.ops = &sdw_mockup_slave_ops,
270	.id_table = sdw_mockup_id,
271};
272module_sdw_driver(sdw_mockup_sdw_driver);
273
274MODULE_DESCRIPTION("ASoC SDW mockup codec driver");
275MODULE_AUTHOR("Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>");
276MODULE_LICENSE("GPL");
277