xref: /openbmc/linux/sound/soc/sof/ipc4-topology.c (revision 234489ac)
1 // SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause)
2 //
3 // This file is provided under a dual BSD/GPLv2 license.  When using or
4 // redistributing this file, you may do so under either license.
5 //
6 // Copyright(c) 2022 Intel Corporation. All rights reserved.
7 //
8 //
9 #include <linux/bitfield.h>
10 #include <uapi/sound/sof/tokens.h>
11 #include <sound/pcm_params.h>
12 #include <sound/sof/ext_manifest4.h>
13 #include <sound/intel-nhlt.h>
14 #include "sof-priv.h"
15 #include "sof-audio.h"
16 #include "ipc4-priv.h"
17 #include "ipc4-topology.h"
18 #include "ops.h"
19 
20 #define SOF_IPC4_GAIN_PARAM_ID  0
21 #define SOF_IPC4_TPLG_ABI_SIZE 6
22 #define SOF_IPC4_CHAIN_DMA_BUF_SIZE_MS 2
23 
24 static DEFINE_IDA(alh_group_ida);
25 static DEFINE_IDA(pipeline_ida);
26 
27 static const struct sof_topology_token ipc4_sched_tokens[] = {
28 	{SOF_TKN_SCHED_LP_MODE, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
29 		offsetof(struct sof_ipc4_pipeline, lp_mode)},
30 	{SOF_TKN_SCHED_USE_CHAIN_DMA, SND_SOC_TPLG_TUPLE_TYPE_BOOL, get_token_u16,
31 		offsetof(struct sof_ipc4_pipeline, use_chain_dma)},
32 	{SOF_TKN_SCHED_CORE, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
33 		offsetof(struct sof_ipc4_pipeline, core_id)},
34 };
35 
36 static const struct sof_topology_token pipeline_tokens[] = {
37 	{SOF_TKN_SCHED_DYNAMIC_PIPELINE, SND_SOC_TPLG_TUPLE_TYPE_BOOL, get_token_u16,
38 		offsetof(struct snd_sof_widget, dynamic_pipeline_widget)},
39 };
40 
41 static const struct sof_topology_token ipc4_comp_tokens[] = {
42 	{SOF_TKN_COMP_CPC, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
43 		offsetof(struct sof_ipc4_base_module_cfg, cpc)},
44 	{SOF_TKN_COMP_IS_PAGES, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
45 		offsetof(struct sof_ipc4_base_module_cfg, is_pages)},
46 };
47 
48 static const struct sof_topology_token ipc4_in_audio_format_tokens[] = {
49 	{SOF_TKN_CAVS_AUDIO_FORMAT_IN_RATE, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
50 		offsetof(struct sof_ipc4_pin_format, audio_fmt.sampling_frequency)},
51 	{SOF_TKN_CAVS_AUDIO_FORMAT_IN_BIT_DEPTH, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
52 		offsetof(struct sof_ipc4_pin_format, audio_fmt.bit_depth)},
53 	{SOF_TKN_CAVS_AUDIO_FORMAT_IN_CH_MAP, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
54 		offsetof(struct sof_ipc4_pin_format, audio_fmt.ch_map)},
55 	{SOF_TKN_CAVS_AUDIO_FORMAT_IN_CH_CFG, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
56 		offsetof(struct sof_ipc4_pin_format, audio_fmt.ch_cfg)},
57 	{SOF_TKN_CAVS_AUDIO_FORMAT_IN_INTERLEAVING_STYLE, SND_SOC_TPLG_TUPLE_TYPE_WORD,
58 		get_token_u32, offsetof(struct sof_ipc4_pin_format,
59 		audio_fmt.interleaving_style)},
60 	{SOF_TKN_CAVS_AUDIO_FORMAT_IN_FMT_CFG, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
61 		offsetof(struct sof_ipc4_pin_format, audio_fmt.fmt_cfg)},
62 	{SOF_TKN_CAVS_AUDIO_FORMAT_INPUT_PIN_INDEX, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
63 		offsetof(struct sof_ipc4_pin_format, pin_index)},
64 	{SOF_TKN_CAVS_AUDIO_FORMAT_IBS, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
65 		offsetof(struct sof_ipc4_pin_format, buffer_size)},
66 };
67 
68 static const struct sof_topology_token ipc4_out_audio_format_tokens[] = {
69 	{SOF_TKN_CAVS_AUDIO_FORMAT_OUT_RATE, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
70 		offsetof(struct sof_ipc4_pin_format, audio_fmt.sampling_frequency)},
71 	{SOF_TKN_CAVS_AUDIO_FORMAT_OUT_BIT_DEPTH, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
72 		offsetof(struct sof_ipc4_pin_format, audio_fmt.bit_depth)},
73 	{SOF_TKN_CAVS_AUDIO_FORMAT_OUT_CH_MAP, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
74 		offsetof(struct sof_ipc4_pin_format, audio_fmt.ch_map)},
75 	{SOF_TKN_CAVS_AUDIO_FORMAT_OUT_CH_CFG, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
76 		offsetof(struct sof_ipc4_pin_format, audio_fmt.ch_cfg)},
77 	{SOF_TKN_CAVS_AUDIO_FORMAT_OUT_INTERLEAVING_STYLE, SND_SOC_TPLG_TUPLE_TYPE_WORD,
78 		get_token_u32, offsetof(struct sof_ipc4_pin_format,
79 		audio_fmt.interleaving_style)},
80 	{SOF_TKN_CAVS_AUDIO_FORMAT_OUT_FMT_CFG, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
81 		offsetof(struct sof_ipc4_pin_format, audio_fmt.fmt_cfg)},
82 	{SOF_TKN_CAVS_AUDIO_FORMAT_OUTPUT_PIN_INDEX, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
83 		offsetof(struct sof_ipc4_pin_format, pin_index)},
84 	{SOF_TKN_CAVS_AUDIO_FORMAT_OBS, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
85 		offsetof(struct sof_ipc4_pin_format, buffer_size)},
86 };
87 
88 static const struct sof_topology_token ipc4_copier_deep_buffer_tokens[] = {
89 	{SOF_TKN_INTEL_COPIER_DEEP_BUFFER_DMA_MS, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32, 0},
90 };
91 
92 static const struct sof_topology_token ipc4_copier_tokens[] = {
93 	{SOF_TKN_INTEL_COPIER_NODE_TYPE, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32, 0},
94 };
95 
96 static const struct sof_topology_token ipc4_audio_fmt_num_tokens[] = {
97 	{SOF_TKN_COMP_NUM_INPUT_AUDIO_FORMATS, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
98 		offsetof(struct sof_ipc4_available_audio_format, num_input_formats)},
99 	{SOF_TKN_COMP_NUM_OUTPUT_AUDIO_FORMATS, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
100 		offsetof(struct sof_ipc4_available_audio_format, num_output_formats)},
101 };
102 
103 static const struct sof_topology_token dai_tokens[] = {
104 	{SOF_TKN_DAI_TYPE, SND_SOC_TPLG_TUPLE_TYPE_STRING, get_token_dai_type,
105 		offsetof(struct sof_ipc4_copier, dai_type)},
106 	{SOF_TKN_DAI_INDEX, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
107 		offsetof(struct sof_ipc4_copier, dai_index)},
108 };
109 
110 /* Component extended tokens */
111 static const struct sof_topology_token comp_ext_tokens[] = {
112 	{SOF_TKN_COMP_UUID, SND_SOC_TPLG_TUPLE_TYPE_UUID, get_token_uuid,
113 		offsetof(struct snd_sof_widget, uuid)},
114 	{SOF_TKN_COMP_CORE_ID, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
115 		offsetof(struct snd_sof_widget, core)},
116 };
117 
118 static const struct sof_topology_token gain_tokens[] = {
119 	{SOF_TKN_GAIN_RAMP_TYPE, SND_SOC_TPLG_TUPLE_TYPE_WORD,
120 		get_token_u32, offsetof(struct sof_ipc4_gain_data, curve_type)},
121 	{SOF_TKN_GAIN_RAMP_DURATION,
122 		SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
123 		offsetof(struct sof_ipc4_gain_data, curve_duration_l)},
124 	{SOF_TKN_GAIN_VAL, SND_SOC_TPLG_TUPLE_TYPE_WORD,
125 		get_token_u32, offsetof(struct sof_ipc4_gain_data, init_val)},
126 };
127 
128 /* SRC */
129 static const struct sof_topology_token src_tokens[] = {
130 	{SOF_TKN_SRC_RATE_OUT, SND_SOC_TPLG_TUPLE_TYPE_WORD, get_token_u32,
131 		offsetof(struct sof_ipc4_src, sink_rate)},
132 };
133 
134 static const struct sof_token_info ipc4_token_list[SOF_TOKEN_COUNT] = {
135 	[SOF_DAI_TOKENS] = {"DAI tokens", dai_tokens, ARRAY_SIZE(dai_tokens)},
136 	[SOF_PIPELINE_TOKENS] = {"Pipeline tokens", pipeline_tokens, ARRAY_SIZE(pipeline_tokens)},
137 	[SOF_SCHED_TOKENS] = {"Scheduler tokens", ipc4_sched_tokens,
138 		ARRAY_SIZE(ipc4_sched_tokens)},
139 	[SOF_COMP_EXT_TOKENS] = {"Comp extended tokens", comp_ext_tokens,
140 		ARRAY_SIZE(comp_ext_tokens)},
141 	[SOF_COMP_TOKENS] = {"IPC4 Component tokens",
142 		ipc4_comp_tokens, ARRAY_SIZE(ipc4_comp_tokens)},
143 	[SOF_IN_AUDIO_FORMAT_TOKENS] = {"IPC4 Input Audio format tokens",
144 		ipc4_in_audio_format_tokens, ARRAY_SIZE(ipc4_in_audio_format_tokens)},
145 	[SOF_OUT_AUDIO_FORMAT_TOKENS] = {"IPC4 Output Audio format tokens",
146 		ipc4_out_audio_format_tokens, ARRAY_SIZE(ipc4_out_audio_format_tokens)},
147 	[SOF_COPIER_DEEP_BUFFER_TOKENS] = {"IPC4 Copier deep buffer tokens",
148 		ipc4_copier_deep_buffer_tokens, ARRAY_SIZE(ipc4_copier_deep_buffer_tokens)},
149 	[SOF_COPIER_TOKENS] = {"IPC4 Copier tokens", ipc4_copier_tokens,
150 		ARRAY_SIZE(ipc4_copier_tokens)},
151 	[SOF_AUDIO_FMT_NUM_TOKENS] = {"IPC4 Audio format number tokens",
152 		ipc4_audio_fmt_num_tokens, ARRAY_SIZE(ipc4_audio_fmt_num_tokens)},
153 	[SOF_GAIN_TOKENS] = {"Gain tokens", gain_tokens, ARRAY_SIZE(gain_tokens)},
154 	[SOF_SRC_TOKENS] = {"SRC tokens", src_tokens, ARRAY_SIZE(src_tokens)},
155 };
156 
157 static void sof_ipc4_dbg_audio_format(struct device *dev, struct sof_ipc4_pin_format *pin_fmt,
158 				      int num_formats)
159 {
160 	int i;
161 
162 	for (i = 0; i < num_formats; i++) {
163 		struct sof_ipc4_audio_format *fmt = &pin_fmt[i].audio_fmt;
164 		dev_dbg(dev,
165 			"Pin index #%d: %uHz, %ubit (ch_map %#x ch_cfg %u interleaving_style %u fmt_cfg %#x) buffer size %d\n",
166 			pin_fmt[i].pin_index, fmt->sampling_frequency, fmt->bit_depth, fmt->ch_map,
167 			fmt->ch_cfg, fmt->interleaving_style, fmt->fmt_cfg,
168 			pin_fmt[i].buffer_size);
169 	}
170 }
171 
172 static const struct sof_ipc4_audio_format *
173 sof_ipc4_get_input_pin_audio_fmt(struct snd_sof_widget *swidget, int pin_index)
174 {
175 	struct sof_ipc4_base_module_cfg_ext *base_cfg_ext;
176 	struct sof_ipc4_process *process;
177 	int i;
178 
179 	if (swidget->id != snd_soc_dapm_effect) {
180 		struct sof_ipc4_base_module_cfg *base = swidget->private;
181 
182 		/* For non-process modules, base module config format is used for all input pins */
183 		return &base->audio_fmt;
184 	}
185 
186 	process = swidget->private;
187 	base_cfg_ext = process->base_config_ext;
188 
189 	/*
190 	 * If there are multiple input formats available for a pin, the first available format
191 	 * is chosen.
192 	 */
193 	for (i = 0; i < base_cfg_ext->num_input_pin_fmts; i++) {
194 		struct sof_ipc4_pin_format *pin_format = &base_cfg_ext->pin_formats[i];
195 
196 		if (pin_format->pin_index == pin_index)
197 			return &pin_format->audio_fmt;
198 	}
199 
200 	return NULL;
201 }
202 
203 /**
204  * sof_ipc4_get_audio_fmt - get available audio formats from swidget->tuples
205  * @scomp: pointer to pointer to SOC component
206  * @swidget: pointer to struct snd_sof_widget containing tuples
207  * @available_fmt: pointer to struct sof_ipc4_available_audio_format being filling in
208  * @module_base_cfg: Pointer to the base_config in the module init IPC payload
209  *
210  * Return: 0 if successful
211  */
212 static int sof_ipc4_get_audio_fmt(struct snd_soc_component *scomp,
213 				  struct snd_sof_widget *swidget,
214 				  struct sof_ipc4_available_audio_format *available_fmt,
215 				  struct sof_ipc4_base_module_cfg *module_base_cfg)
216 {
217 	struct sof_ipc4_pin_format *in_format = NULL;
218 	struct sof_ipc4_pin_format *out_format;
219 	int ret;
220 
221 	ret = sof_update_ipc_object(scomp, available_fmt,
222 				    SOF_AUDIO_FMT_NUM_TOKENS, swidget->tuples,
223 				    swidget->num_tuples, sizeof(available_fmt), 1);
224 	if (ret) {
225 		dev_err(scomp->dev, "Failed to parse audio format token count\n");
226 		return ret;
227 	}
228 
229 	if (!available_fmt->num_input_formats && !available_fmt->num_output_formats) {
230 		dev_err(scomp->dev, "No input/output pin formats set in topology\n");
231 		return -EINVAL;
232 	}
233 
234 	dev_dbg(scomp->dev,
235 		"Number of input audio formats: %d. Number of output audio formats: %d\n",
236 		available_fmt->num_input_formats, available_fmt->num_output_formats);
237 
238 	/* set cpc and is_pages in the module's base_config */
239 	ret = sof_update_ipc_object(scomp, module_base_cfg, SOF_COMP_TOKENS, swidget->tuples,
240 				    swidget->num_tuples, sizeof(*module_base_cfg), 1);
241 	if (ret) {
242 		dev_err(scomp->dev, "parse comp tokens for %s failed, error: %d\n",
243 			swidget->widget->name, ret);
244 		return ret;
245 	}
246 
247 	dev_dbg(scomp->dev, "widget %s cpc: %d is_pages: %d\n",
248 		swidget->widget->name, module_base_cfg->cpc, module_base_cfg->is_pages);
249 
250 	if (available_fmt->num_input_formats) {
251 		in_format = kcalloc(available_fmt->num_input_formats,
252 				    sizeof(*in_format), GFP_KERNEL);
253 		if (!in_format)
254 			return -ENOMEM;
255 		available_fmt->input_pin_fmts = in_format;
256 
257 		ret = sof_update_ipc_object(scomp, in_format,
258 					    SOF_IN_AUDIO_FORMAT_TOKENS, swidget->tuples,
259 					    swidget->num_tuples, sizeof(*in_format),
260 					    available_fmt->num_input_formats);
261 		if (ret) {
262 			dev_err(scomp->dev, "parse input audio fmt tokens failed %d\n", ret);
263 			goto err_in;
264 		}
265 
266 		dev_dbg(scomp->dev, "Input audio formats for %s\n", swidget->widget->name);
267 		sof_ipc4_dbg_audio_format(scomp->dev, in_format,
268 					  available_fmt->num_input_formats);
269 	}
270 
271 	if (available_fmt->num_output_formats) {
272 		out_format = kcalloc(available_fmt->num_output_formats, sizeof(*out_format),
273 				     GFP_KERNEL);
274 		if (!out_format) {
275 			ret = -ENOMEM;
276 			goto err_in;
277 		}
278 
279 		ret = sof_update_ipc_object(scomp, out_format,
280 					    SOF_OUT_AUDIO_FORMAT_TOKENS, swidget->tuples,
281 					    swidget->num_tuples, sizeof(*out_format),
282 					    available_fmt->num_output_formats);
283 		if (ret) {
284 			dev_err(scomp->dev, "parse output audio fmt tokens failed\n");
285 			goto err_out;
286 		}
287 
288 		available_fmt->output_pin_fmts = out_format;
289 		dev_dbg(scomp->dev, "Output audio formats for %s\n", swidget->widget->name);
290 		sof_ipc4_dbg_audio_format(scomp->dev, out_format,
291 					  available_fmt->num_output_formats);
292 	}
293 
294 	return 0;
295 
296 err_out:
297 	kfree(out_format);
298 err_in:
299 	kfree(in_format);
300 	available_fmt->input_pin_fmts = NULL;
301 	return ret;
302 }
303 
304 /* release the memory allocated in sof_ipc4_get_audio_fmt */
305 static void sof_ipc4_free_audio_fmt(struct sof_ipc4_available_audio_format *available_fmt)
306 
307 {
308 	kfree(available_fmt->output_pin_fmts);
309 	available_fmt->output_pin_fmts = NULL;
310 	kfree(available_fmt->input_pin_fmts);
311 	available_fmt->input_pin_fmts = NULL;
312 }
313 
314 static void sof_ipc4_widget_free_comp_pipeline(struct snd_sof_widget *swidget)
315 {
316 	kfree(swidget->private);
317 }
318 
319 static int sof_ipc4_widget_set_module_info(struct snd_sof_widget *swidget)
320 {
321 	struct snd_soc_component *scomp = swidget->scomp;
322 	struct snd_sof_dev *sdev = snd_soc_component_get_drvdata(scomp);
323 
324 	swidget->module_info = sof_ipc4_find_module_by_uuid(sdev, &swidget->uuid);
325 
326 	if (swidget->module_info)
327 		return 0;
328 
329 	dev_err(sdev->dev, "failed to find module info for widget %s with UUID %pUL\n",
330 		swidget->widget->name, &swidget->uuid);
331 	return -EINVAL;
332 }
333 
334 static int sof_ipc4_widget_setup_msg(struct snd_sof_widget *swidget, struct sof_ipc4_msg *msg)
335 {
336 	struct sof_ipc4_fw_module *fw_module;
337 	uint32_t type;
338 	int ret;
339 
340 	ret = sof_ipc4_widget_set_module_info(swidget);
341 	if (ret)
342 		return ret;
343 
344 	fw_module = swidget->module_info;
345 
346 	msg->primary = fw_module->man4_module_entry.id;
347 	msg->primary |= SOF_IPC4_MSG_TYPE_SET(SOF_IPC4_MOD_INIT_INSTANCE);
348 	msg->primary |= SOF_IPC4_MSG_DIR(SOF_IPC4_MSG_REQUEST);
349 	msg->primary |= SOF_IPC4_MSG_TARGET(SOF_IPC4_MODULE_MSG);
350 
351 	msg->extension = SOF_IPC4_MOD_EXT_CORE_ID(swidget->core);
352 
353 	type = (fw_module->man4_module_entry.type & SOF_IPC4_MODULE_DP) ? 1 : 0;
354 	msg->extension |= SOF_IPC4_MOD_EXT_DOMAIN(type);
355 
356 	return 0;
357 }
358 
359 static void sof_ipc4_widget_update_kcontrol_module_id(struct snd_sof_widget *swidget)
360 {
361 	struct snd_soc_component *scomp = swidget->scomp;
362 	struct snd_sof_dev *sdev = snd_soc_component_get_drvdata(scomp);
363 	struct sof_ipc4_fw_module *fw_module = swidget->module_info;
364 	struct snd_sof_control *scontrol;
365 
366 	/* update module ID for all kcontrols for this widget */
367 	list_for_each_entry(scontrol, &sdev->kcontrol_list, list) {
368 		if (scontrol->comp_id == swidget->comp_id) {
369 			struct sof_ipc4_control_data *cdata = scontrol->ipc_control_data;
370 			struct sof_ipc4_msg *msg = &cdata->msg;
371 
372 			msg->primary |= fw_module->man4_module_entry.id;
373 		}
374 	}
375 }
376 
377 static int sof_ipc4_widget_setup_pcm(struct snd_sof_widget *swidget)
378 {
379 	struct sof_ipc4_available_audio_format *available_fmt;
380 	struct snd_soc_component *scomp = swidget->scomp;
381 	struct sof_ipc4_copier *ipc4_copier;
382 	int node_type = 0;
383 	int ret;
384 
385 	ipc4_copier = kzalloc(sizeof(*ipc4_copier), GFP_KERNEL);
386 	if (!ipc4_copier)
387 		return -ENOMEM;
388 
389 	swidget->private = ipc4_copier;
390 	available_fmt = &ipc4_copier->available_fmt;
391 
392 	dev_dbg(scomp->dev, "Updating IPC structure for %s\n", swidget->widget->name);
393 
394 	ret = sof_ipc4_get_audio_fmt(scomp, swidget, available_fmt,
395 				     &ipc4_copier->data.base_config);
396 	if (ret)
397 		goto free_copier;
398 
399 	/*
400 	 * This callback is used by host copier and module-to-module copier,
401 	 * and only host copier needs to set gtw_cfg.
402 	 */
403 	if (!WIDGET_IS_AIF(swidget->id))
404 		goto skip_gtw_cfg;
405 
406 	ret = sof_update_ipc_object(scomp, &node_type,
407 				    SOF_COPIER_TOKENS, swidget->tuples,
408 				    swidget->num_tuples, sizeof(node_type), 1);
409 
410 	if (ret) {
411 		dev_err(scomp->dev, "parse host copier node type token failed %d\n",
412 			ret);
413 		goto free_available_fmt;
414 	}
415 	dev_dbg(scomp->dev, "host copier '%s' node_type %u\n", swidget->widget->name, node_type);
416 
417 skip_gtw_cfg:
418 	ipc4_copier->gtw_attr = kzalloc(sizeof(*ipc4_copier->gtw_attr), GFP_KERNEL);
419 	if (!ipc4_copier->gtw_attr) {
420 		ret = -ENOMEM;
421 		goto free_available_fmt;
422 	}
423 
424 	ipc4_copier->copier_config = (uint32_t *)ipc4_copier->gtw_attr;
425 	ipc4_copier->data.gtw_cfg.config_length =
426 		sizeof(struct sof_ipc4_gtw_attributes) >> 2;
427 
428 	switch (swidget->id) {
429 	case snd_soc_dapm_aif_in:
430 	case snd_soc_dapm_aif_out:
431 		ipc4_copier->data.gtw_cfg.node_id = SOF_IPC4_NODE_TYPE(node_type);
432 		break;
433 	case snd_soc_dapm_buffer:
434 		ipc4_copier->data.gtw_cfg.node_id = SOF_IPC4_INVALID_NODE_ID;
435 		ipc4_copier->ipc_config_size = 0;
436 		break;
437 	default:
438 		dev_err(scomp->dev, "invalid widget type %d\n", swidget->id);
439 		ret = -EINVAL;
440 		goto free_gtw_attr;
441 	}
442 
443 	/* set up module info and message header */
444 	ret = sof_ipc4_widget_setup_msg(swidget, &ipc4_copier->msg);
445 	if (ret)
446 		goto free_gtw_attr;
447 
448 	return 0;
449 
450 free_gtw_attr:
451 	kfree(ipc4_copier->gtw_attr);
452 free_available_fmt:
453 	sof_ipc4_free_audio_fmt(available_fmt);
454 free_copier:
455 	kfree(ipc4_copier);
456 	swidget->private = NULL;
457 	return ret;
458 }
459 
460 static void sof_ipc4_widget_free_comp_pcm(struct snd_sof_widget *swidget)
461 {
462 	struct sof_ipc4_copier *ipc4_copier = swidget->private;
463 	struct sof_ipc4_available_audio_format *available_fmt;
464 
465 	if (!ipc4_copier)
466 		return;
467 
468 	available_fmt = &ipc4_copier->available_fmt;
469 	kfree(available_fmt->output_pin_fmts);
470 	kfree(ipc4_copier->gtw_attr);
471 	kfree(ipc4_copier);
472 	swidget->private = NULL;
473 }
474 
475 static int sof_ipc4_widget_setup_comp_dai(struct snd_sof_widget *swidget)
476 {
477 	struct sof_ipc4_available_audio_format *available_fmt;
478 	struct snd_soc_component *scomp = swidget->scomp;
479 	struct snd_sof_dai *dai = swidget->private;
480 	struct sof_ipc4_copier *ipc4_copier;
481 	struct snd_sof_widget *pipe_widget;
482 	struct sof_ipc4_pipeline *pipeline;
483 	int node_type = 0;
484 	int ret;
485 
486 	ipc4_copier = kzalloc(sizeof(*ipc4_copier), GFP_KERNEL);
487 	if (!ipc4_copier)
488 		return -ENOMEM;
489 
490 	available_fmt = &ipc4_copier->available_fmt;
491 
492 	dev_dbg(scomp->dev, "Updating IPC structure for %s\n", swidget->widget->name);
493 
494 	ret = sof_ipc4_get_audio_fmt(scomp, swidget, available_fmt,
495 				     &ipc4_copier->data.base_config);
496 	if (ret)
497 		goto free_copier;
498 
499 	ret = sof_update_ipc_object(scomp, &node_type,
500 				    SOF_COPIER_TOKENS, swidget->tuples,
501 				    swidget->num_tuples, sizeof(node_type), 1);
502 	if (ret) {
503 		dev_err(scomp->dev, "parse dai node type failed %d\n", ret);
504 		goto free_available_fmt;
505 	}
506 
507 	ret = sof_update_ipc_object(scomp, ipc4_copier,
508 				    SOF_DAI_TOKENS, swidget->tuples,
509 				    swidget->num_tuples, sizeof(u32), 1);
510 	if (ret) {
511 		dev_err(scomp->dev, "parse dai copier node token failed %d\n", ret);
512 		goto free_available_fmt;
513 	}
514 
515 	dev_dbg(scomp->dev, "dai %s node_type %u dai_type %u dai_index %d\n", swidget->widget->name,
516 		node_type, ipc4_copier->dai_type, ipc4_copier->dai_index);
517 
518 	ipc4_copier->data.gtw_cfg.node_id = SOF_IPC4_NODE_TYPE(node_type);
519 
520 	pipe_widget = swidget->spipe->pipe_widget;
521 	pipeline = pipe_widget->private;
522 	if (pipeline->use_chain_dma && ipc4_copier->dai_type != SOF_DAI_INTEL_HDA) {
523 		dev_err(scomp->dev,
524 			"Bad DAI type '%d', Chained DMA is only supported by HDA DAIs (%d).\n",
525 			ipc4_copier->dai_type, SOF_DAI_INTEL_HDA);
526 		ret = -ENODEV;
527 		goto free_available_fmt;
528 	}
529 
530 	switch (ipc4_copier->dai_type) {
531 	case SOF_DAI_INTEL_ALH:
532 	{
533 		struct snd_sof_dev *sdev = snd_soc_component_get_drvdata(scomp);
534 		struct sof_ipc4_alh_configuration_blob *blob;
535 		struct snd_soc_dapm_path *p;
536 		struct snd_sof_widget *w;
537 		int src_num = 0;
538 
539 		snd_soc_dapm_widget_for_each_source_path(swidget->widget, p)
540 			src_num++;
541 
542 		if (swidget->id == snd_soc_dapm_dai_in && src_num == 0) {
543 			/*
544 			 * The blob will not be used if the ALH copier is playback direction
545 			 * and doesn't connect to any source.
546 			 * It is fine to call kfree(ipc4_copier->copier_config) since
547 			 * ipc4_copier->copier_config is null.
548 			 */
549 			ret = 0;
550 			break;
551 		}
552 
553 		blob = kzalloc(sizeof(*blob), GFP_KERNEL);
554 		if (!blob) {
555 			ret = -ENOMEM;
556 			goto free_available_fmt;
557 		}
558 
559 		list_for_each_entry(w, &sdev->widget_list, list) {
560 			if (w->widget->sname &&
561 			    strcmp(w->widget->sname, swidget->widget->sname))
562 				continue;
563 
564 			blob->alh_cfg.count++;
565 		}
566 
567 		ipc4_copier->copier_config = (uint32_t *)blob;
568 		ipc4_copier->data.gtw_cfg.config_length = sizeof(*blob) >> 2;
569 		break;
570 	}
571 	case SOF_DAI_INTEL_SSP:
572 		/* set SSP DAI index as the node_id */
573 		ipc4_copier->data.gtw_cfg.node_id |=
574 			SOF_IPC4_NODE_INDEX_INTEL_SSP(ipc4_copier->dai_index);
575 		break;
576 	case SOF_DAI_INTEL_DMIC:
577 		/* set DMIC DAI index as the node_id */
578 		ipc4_copier->data.gtw_cfg.node_id |=
579 			SOF_IPC4_NODE_INDEX_INTEL_DMIC(ipc4_copier->dai_index);
580 		break;
581 	default:
582 		ipc4_copier->gtw_attr = kzalloc(sizeof(*ipc4_copier->gtw_attr), GFP_KERNEL);
583 		if (!ipc4_copier->gtw_attr) {
584 			ret = -ENOMEM;
585 			goto free_available_fmt;
586 		}
587 
588 		ipc4_copier->copier_config = (uint32_t *)ipc4_copier->gtw_attr;
589 		ipc4_copier->data.gtw_cfg.config_length =
590 			sizeof(struct sof_ipc4_gtw_attributes) >> 2;
591 		break;
592 	}
593 
594 	dai->scomp = scomp;
595 	dai->private = ipc4_copier;
596 
597 	/* set up module info and message header */
598 	ret = sof_ipc4_widget_setup_msg(swidget, &ipc4_copier->msg);
599 	if (ret)
600 		goto free_copier_config;
601 
602 	return 0;
603 
604 free_copier_config:
605 	kfree(ipc4_copier->copier_config);
606 free_available_fmt:
607 	sof_ipc4_free_audio_fmt(available_fmt);
608 free_copier:
609 	kfree(ipc4_copier);
610 	dai->private = NULL;
611 	dai->scomp = NULL;
612 	return ret;
613 }
614 
615 static void sof_ipc4_widget_free_comp_dai(struct snd_sof_widget *swidget)
616 {
617 	struct sof_ipc4_available_audio_format *available_fmt;
618 	struct snd_sof_dai *dai = swidget->private;
619 	struct sof_ipc4_copier *ipc4_copier;
620 
621 	if (!dai)
622 		return;
623 
624 	if (!dai->private) {
625 		kfree(dai);
626 		swidget->private = NULL;
627 		return;
628 	}
629 
630 	ipc4_copier = dai->private;
631 	available_fmt = &ipc4_copier->available_fmt;
632 
633 	kfree(available_fmt->output_pin_fmts);
634 	if (ipc4_copier->dai_type != SOF_DAI_INTEL_SSP &&
635 	    ipc4_copier->dai_type != SOF_DAI_INTEL_DMIC)
636 		kfree(ipc4_copier->copier_config);
637 	kfree(dai->private);
638 	kfree(dai);
639 	swidget->private = NULL;
640 }
641 
642 static int sof_ipc4_widget_setup_comp_pipeline(struct snd_sof_widget *swidget)
643 {
644 	struct snd_soc_component *scomp = swidget->scomp;
645 	struct sof_ipc4_pipeline *pipeline;
646 	int ret;
647 
648 	pipeline = kzalloc(sizeof(*pipeline), GFP_KERNEL);
649 	if (!pipeline)
650 		return -ENOMEM;
651 
652 	ret = sof_update_ipc_object(scomp, pipeline, SOF_SCHED_TOKENS, swidget->tuples,
653 				    swidget->num_tuples, sizeof(*pipeline), 1);
654 	if (ret) {
655 		dev_err(scomp->dev, "parsing scheduler tokens failed\n");
656 		goto err;
657 	}
658 
659 	swidget->core = pipeline->core_id;
660 
661 	if (pipeline->use_chain_dma) {
662 		dev_dbg(scomp->dev, "Set up chain DMA for %s\n", swidget->widget->name);
663 		swidget->private = pipeline;
664 		return 0;
665 	}
666 
667 	/* parse one set of pipeline tokens */
668 	ret = sof_update_ipc_object(scomp, swidget, SOF_PIPELINE_TOKENS, swidget->tuples,
669 				    swidget->num_tuples, sizeof(*swidget), 1);
670 	if (ret) {
671 		dev_err(scomp->dev, "parsing pipeline tokens failed\n");
672 		goto err;
673 	}
674 
675 	/* TODO: Get priority from topology */
676 	pipeline->priority = 0;
677 
678 	dev_dbg(scomp->dev, "pipeline '%s': id %d, pri %d, core_id %u, lp mode %d\n",
679 		swidget->widget->name, swidget->pipeline_id,
680 		pipeline->priority, pipeline->core_id, pipeline->lp_mode);
681 
682 	swidget->private = pipeline;
683 
684 	pipeline->msg.primary = SOF_IPC4_GLB_PIPE_PRIORITY(pipeline->priority);
685 	pipeline->msg.primary |= SOF_IPC4_MSG_TYPE_SET(SOF_IPC4_GLB_CREATE_PIPELINE);
686 	pipeline->msg.primary |= SOF_IPC4_MSG_DIR(SOF_IPC4_MSG_REQUEST);
687 	pipeline->msg.primary |= SOF_IPC4_MSG_TARGET(SOF_IPC4_FW_GEN_MSG);
688 
689 	pipeline->msg.extension = pipeline->lp_mode;
690 	pipeline->msg.extension |= SOF_IPC4_GLB_PIPE_EXT_CORE_ID(pipeline->core_id);
691 	pipeline->state = SOF_IPC4_PIPE_UNINITIALIZED;
692 
693 	return 0;
694 err:
695 	kfree(pipeline);
696 	return ret;
697 }
698 
699 static int sof_ipc4_widget_setup_comp_pga(struct snd_sof_widget *swidget)
700 {
701 	struct snd_soc_component *scomp = swidget->scomp;
702 	struct sof_ipc4_gain *gain;
703 	int ret;
704 
705 	gain = kzalloc(sizeof(*gain), GFP_KERNEL);
706 	if (!gain)
707 		return -ENOMEM;
708 
709 	swidget->private = gain;
710 
711 	gain->data.channels = SOF_IPC4_GAIN_ALL_CHANNELS_MASK;
712 	gain->data.init_val = SOF_IPC4_VOL_ZERO_DB;
713 
714 	ret = sof_ipc4_get_audio_fmt(scomp, swidget, &gain->available_fmt, &gain->base_config);
715 	if (ret)
716 		goto err;
717 
718 	ret = sof_update_ipc_object(scomp, &gain->data, SOF_GAIN_TOKENS, swidget->tuples,
719 				    swidget->num_tuples, sizeof(gain->data), 1);
720 	if (ret) {
721 		dev_err(scomp->dev, "Parsing gain tokens failed\n");
722 		goto err;
723 	}
724 
725 	dev_dbg(scomp->dev,
726 		"pga widget %s: ramp type: %d, ramp duration %d, initial gain value: %#x, cpc %d\n",
727 		swidget->widget->name, gain->data.curve_type, gain->data.curve_duration_l,
728 		gain->data.init_val, gain->base_config.cpc);
729 
730 	ret = sof_ipc4_widget_setup_msg(swidget, &gain->msg);
731 	if (ret)
732 		goto err;
733 
734 	sof_ipc4_widget_update_kcontrol_module_id(swidget);
735 
736 	return 0;
737 err:
738 	sof_ipc4_free_audio_fmt(&gain->available_fmt);
739 	kfree(gain);
740 	swidget->private = NULL;
741 	return ret;
742 }
743 
744 static void sof_ipc4_widget_free_comp_pga(struct snd_sof_widget *swidget)
745 {
746 	struct sof_ipc4_gain *gain = swidget->private;
747 
748 	if (!gain)
749 		return;
750 
751 	sof_ipc4_free_audio_fmt(&gain->available_fmt);
752 	kfree(swidget->private);
753 	swidget->private = NULL;
754 }
755 
756 static int sof_ipc4_widget_setup_comp_mixer(struct snd_sof_widget *swidget)
757 {
758 	struct snd_soc_component *scomp = swidget->scomp;
759 	struct sof_ipc4_mixer *mixer;
760 	int ret;
761 
762 	dev_dbg(scomp->dev, "Updating IPC structure for %s\n", swidget->widget->name);
763 
764 	mixer = kzalloc(sizeof(*mixer), GFP_KERNEL);
765 	if (!mixer)
766 		return -ENOMEM;
767 
768 	swidget->private = mixer;
769 
770 	ret = sof_ipc4_get_audio_fmt(scomp, swidget, &mixer->available_fmt,
771 				     &mixer->base_config);
772 	if (ret)
773 		goto err;
774 
775 	ret = sof_ipc4_widget_setup_msg(swidget, &mixer->msg);
776 	if (ret)
777 		goto err;
778 
779 	return 0;
780 err:
781 	sof_ipc4_free_audio_fmt(&mixer->available_fmt);
782 	kfree(mixer);
783 	swidget->private = NULL;
784 	return ret;
785 }
786 
787 static int sof_ipc4_widget_setup_comp_src(struct snd_sof_widget *swidget)
788 {
789 	struct snd_soc_component *scomp = swidget->scomp;
790 	struct sof_ipc4_src *src;
791 	int ret;
792 
793 	dev_dbg(scomp->dev, "Updating IPC structure for %s\n", swidget->widget->name);
794 
795 	src = kzalloc(sizeof(*src), GFP_KERNEL);
796 	if (!src)
797 		return -ENOMEM;
798 
799 	swidget->private = src;
800 
801 	ret = sof_ipc4_get_audio_fmt(scomp, swidget, &src->available_fmt, &src->base_config);
802 	if (ret)
803 		goto err;
804 
805 	ret = sof_update_ipc_object(scomp, src, SOF_SRC_TOKENS, swidget->tuples,
806 				    swidget->num_tuples, sizeof(*src), 1);
807 	if (ret) {
808 		dev_err(scomp->dev, "Parsing SRC tokens failed\n");
809 		goto err;
810 	}
811 
812 	dev_dbg(scomp->dev, "SRC sink rate %d\n", src->sink_rate);
813 
814 	ret = sof_ipc4_widget_setup_msg(swidget, &src->msg);
815 	if (ret)
816 		goto err;
817 
818 	return 0;
819 err:
820 	sof_ipc4_free_audio_fmt(&src->available_fmt);
821 	kfree(src);
822 	swidget->private = NULL;
823 	return ret;
824 }
825 
826 static void sof_ipc4_widget_free_comp_src(struct snd_sof_widget *swidget)
827 {
828 	struct sof_ipc4_src *src = swidget->private;
829 
830 	if (!src)
831 		return;
832 
833 	sof_ipc4_free_audio_fmt(&src->available_fmt);
834 	kfree(swidget->private);
835 	swidget->private = NULL;
836 }
837 
838 static void sof_ipc4_widget_free_comp_mixer(struct snd_sof_widget *swidget)
839 {
840 	struct sof_ipc4_mixer *mixer = swidget->private;
841 
842 	if (!mixer)
843 		return;
844 
845 	sof_ipc4_free_audio_fmt(&mixer->available_fmt);
846 	kfree(swidget->private);
847 	swidget->private = NULL;
848 }
849 
850 /*
851  * Add the process modules support. The process modules are defined as snd_soc_dapm_effect modules.
852  */
853 static int sof_ipc4_widget_setup_comp_process(struct snd_sof_widget *swidget)
854 {
855 	struct snd_soc_component *scomp = swidget->scomp;
856 	struct sof_ipc4_fw_module *fw_module;
857 	struct sof_ipc4_process *process;
858 	void *cfg;
859 	int ret;
860 
861 	process = kzalloc(sizeof(*process), GFP_KERNEL);
862 	if (!process)
863 		return -ENOMEM;
864 
865 	swidget->private = process;
866 
867 	ret = sof_ipc4_get_audio_fmt(scomp, swidget, &process->available_fmt,
868 				     &process->base_config);
869 	if (ret)
870 		goto err;
871 
872 	ret = sof_ipc4_widget_setup_msg(swidget, &process->msg);
873 	if (ret)
874 		goto err;
875 
876 	/* parse process init module payload config type from module info */
877 	fw_module = swidget->module_info;
878 	process->init_config = FIELD_GET(SOF_IPC4_MODULE_INIT_CONFIG_MASK,
879 					 fw_module->man4_module_entry.type);
880 
881 	process->ipc_config_size = sizeof(struct sof_ipc4_base_module_cfg);
882 
883 	/* allocate memory for base config extension if needed */
884 	if (process->init_config == SOF_IPC4_MODULE_INIT_CONFIG_TYPE_BASE_CFG_WITH_EXT) {
885 		struct sof_ipc4_base_module_cfg_ext *base_cfg_ext;
886 		u32 ext_size = struct_size(base_cfg_ext, pin_formats,
887 						swidget->num_input_pins + swidget->num_output_pins);
888 
889 		base_cfg_ext = kzalloc(ext_size, GFP_KERNEL);
890 		if (!base_cfg_ext) {
891 			ret = -ENOMEM;
892 			goto free_available_fmt;
893 		}
894 
895 		base_cfg_ext->num_input_pin_fmts = swidget->num_input_pins;
896 		base_cfg_ext->num_output_pin_fmts = swidget->num_output_pins;
897 		process->base_config_ext = base_cfg_ext;
898 		process->base_config_ext_size = ext_size;
899 		process->ipc_config_size += ext_size;
900 	}
901 
902 	cfg = kzalloc(process->ipc_config_size, GFP_KERNEL);
903 	if (!cfg) {
904 		ret = -ENOMEM;
905 		goto free_base_cfg_ext;
906 	}
907 
908 	process->ipc_config_data = cfg;
909 
910 	sof_ipc4_widget_update_kcontrol_module_id(swidget);
911 
912 	return 0;
913 free_base_cfg_ext:
914 	kfree(process->base_config_ext);
915 	process->base_config_ext = NULL;
916 free_available_fmt:
917 	sof_ipc4_free_audio_fmt(&process->available_fmt);
918 err:
919 	kfree(process);
920 	swidget->private = NULL;
921 	return ret;
922 }
923 
924 static void sof_ipc4_widget_free_comp_process(struct snd_sof_widget *swidget)
925 {
926 	struct sof_ipc4_process *process = swidget->private;
927 
928 	if (!process)
929 		return;
930 
931 	kfree(process->ipc_config_data);
932 	kfree(process->base_config_ext);
933 	sof_ipc4_free_audio_fmt(&process->available_fmt);
934 	kfree(swidget->private);
935 	swidget->private = NULL;
936 }
937 
938 static void
939 sof_ipc4_update_pipeline_mem_usage(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget,
940 				   struct sof_ipc4_base_module_cfg *base_config)
941 {
942 	struct sof_ipc4_fw_module *fw_module = swidget->module_info;
943 	struct snd_sof_widget *pipe_widget;
944 	struct sof_ipc4_pipeline *pipeline;
945 	int task_mem, queue_mem;
946 	int ibs, bss, total;
947 
948 	ibs = base_config->ibs;
949 	bss = base_config->is_pages;
950 
951 	task_mem = SOF_IPC4_PIPELINE_OBJECT_SIZE;
952 	task_mem += SOF_IPC4_MODULE_INSTANCE_LIST_ITEM_SIZE + bss;
953 
954 	if (fw_module->man4_module_entry.type & SOF_IPC4_MODULE_LL) {
955 		task_mem += SOF_IPC4_FW_ROUNDUP(SOF_IPC4_LL_TASK_OBJECT_SIZE);
956 		task_mem += SOF_IPC4_FW_MAX_QUEUE_COUNT * SOF_IPC4_MODULE_INSTANCE_LIST_ITEM_SIZE;
957 		task_mem += SOF_IPC4_LL_TASK_LIST_ITEM_SIZE;
958 	} else {
959 		task_mem += SOF_IPC4_FW_ROUNDUP(SOF_IPC4_DP_TASK_OBJECT_SIZE);
960 		task_mem += SOF_IPC4_DP_TASK_LIST_SIZE;
961 	}
962 
963 	ibs = SOF_IPC4_FW_ROUNDUP(ibs);
964 	queue_mem = SOF_IPC4_FW_MAX_QUEUE_COUNT * (SOF_IPC4_DATA_QUEUE_OBJECT_SIZE +  ibs);
965 
966 	total = SOF_IPC4_FW_PAGE(task_mem + queue_mem);
967 
968 	pipe_widget = swidget->spipe->pipe_widget;
969 	pipeline = pipe_widget->private;
970 	pipeline->mem_usage += total;
971 }
972 
973 static int sof_ipc4_widget_assign_instance_id(struct snd_sof_dev *sdev,
974 					      struct snd_sof_widget *swidget)
975 {
976 	struct sof_ipc4_fw_module *fw_module = swidget->module_info;
977 	int max_instances = fw_module->man4_module_entry.instance_max_count;
978 
979 	swidget->instance_id = ida_alloc_max(&fw_module->m_ida, max_instances, GFP_KERNEL);
980 	if (swidget->instance_id < 0) {
981 		dev_err(sdev->dev, "failed to assign instance id for widget %s",
982 			swidget->widget->name);
983 		return swidget->instance_id;
984 	}
985 
986 	return 0;
987 }
988 
989 /* update hw_params based on the audio stream format */
990 static int sof_ipc4_update_hw_params(struct snd_sof_dev *sdev, struct snd_pcm_hw_params *params,
991 				     struct sof_ipc4_audio_format *fmt)
992 {
993 	snd_pcm_format_t snd_fmt;
994 	struct snd_interval *i;
995 	struct snd_mask *m;
996 	int valid_bits = SOF_IPC4_AUDIO_FORMAT_CFG_V_BIT_DEPTH(fmt->fmt_cfg);
997 	unsigned int channels, rate;
998 
999 	switch (valid_bits) {
1000 	case 16:
1001 		snd_fmt = SNDRV_PCM_FORMAT_S16_LE;
1002 		break;
1003 	case 24:
1004 		snd_fmt = SNDRV_PCM_FORMAT_S24_LE;
1005 		break;
1006 	case 32:
1007 		snd_fmt = SNDRV_PCM_FORMAT_S32_LE;
1008 		break;
1009 	default:
1010 		dev_err(sdev->dev, "invalid PCM valid_bits %d\n", valid_bits);
1011 		return -EINVAL;
1012 	}
1013 
1014 	m = hw_param_mask(params, SNDRV_PCM_HW_PARAM_FORMAT);
1015 	snd_mask_none(m);
1016 	snd_mask_set_format(m, snd_fmt);
1017 
1018 	rate = fmt->sampling_frequency;
1019 	i = hw_param_interval(params, SNDRV_PCM_HW_PARAM_RATE);
1020 	i->min = rate;
1021 	i->max = rate;
1022 
1023 	channels = SOF_IPC4_AUDIO_FORMAT_CFG_CHANNELS_COUNT(fmt->fmt_cfg);
1024 	i = hw_param_interval(params, SNDRV_PCM_HW_PARAM_CHANNELS);
1025 	i->min = channels;
1026 	i->max = channels;
1027 
1028 	return 0;
1029 }
1030 
1031 static int sof_ipc4_init_audio_fmt(struct snd_sof_dev *sdev,
1032 				   struct snd_sof_widget *swidget,
1033 				   struct sof_ipc4_base_module_cfg *base_config,
1034 				   struct snd_pcm_hw_params *params,
1035 				   struct sof_ipc4_available_audio_format *available_fmt,
1036 				   struct sof_ipc4_pin_format *pin_fmts, u32 pin_fmts_size)
1037 {
1038 	u32 valid_bits;
1039 	u32 channels;
1040 	u32 rate;
1041 	int sample_valid_bits;
1042 	int i;
1043 
1044 	if (!pin_fmts) {
1045 		dev_err(sdev->dev, "no reference formats for %s\n", swidget->widget->name);
1046 		return -EINVAL;
1047 	}
1048 
1049 	switch (params_format(params)) {
1050 	case SNDRV_PCM_FORMAT_S16_LE:
1051 		sample_valid_bits = 16;
1052 		break;
1053 	case SNDRV_PCM_FORMAT_S24_LE:
1054 		sample_valid_bits = 24;
1055 		break;
1056 	case SNDRV_PCM_FORMAT_S32_LE:
1057 		sample_valid_bits = 32;
1058 		break;
1059 	default:
1060 		dev_err(sdev->dev, "invalid pcm frame format %d\n", params_format(params));
1061 		return -EINVAL;
1062 	}
1063 
1064 	if (!pin_fmts_size) {
1065 		dev_err(sdev->dev, "no formats available for %s\n", swidget->widget->name);
1066 		return -EINVAL;
1067 	}
1068 
1069 	/*
1070 	 * Search supported audio formats with pin index 0 to match rate, channels ,and
1071 	 * sample_valid_bytes from runtime params
1072 	 */
1073 	for (i = 0; i < pin_fmts_size; i++) {
1074 		struct sof_ipc4_audio_format *fmt = &pin_fmts[i].audio_fmt;
1075 
1076 		if (pin_fmts[i].pin_index)
1077 			continue;
1078 
1079 		rate = fmt->sampling_frequency;
1080 		channels = SOF_IPC4_AUDIO_FORMAT_CFG_CHANNELS_COUNT(fmt->fmt_cfg);
1081 		valid_bits = SOF_IPC4_AUDIO_FORMAT_CFG_V_BIT_DEPTH(fmt->fmt_cfg);
1082 		if (params_rate(params) == rate && params_channels(params) == channels &&
1083 		    sample_valid_bits == valid_bits) {
1084 			dev_dbg(sdev->dev, "matched audio format index for %uHz, %ubit, %u channels: %d\n",
1085 				rate, valid_bits, channels, i);
1086 			break;
1087 		}
1088 	}
1089 
1090 	if (i == pin_fmts_size) {
1091 		dev_err(sdev->dev, "%s: Unsupported audio format: %uHz, %ubit, %u channels\n",
1092 			__func__, params_rate(params), sample_valid_bits, params_channels(params));
1093 		return -EINVAL;
1094 	}
1095 
1096 	/* copy input format */
1097 	if (available_fmt->num_input_formats && i < available_fmt->num_input_formats) {
1098 		memcpy(&base_config->audio_fmt, &available_fmt->input_pin_fmts[i].audio_fmt,
1099 		       sizeof(struct sof_ipc4_audio_format));
1100 
1101 		/* set base_cfg ibs/obs */
1102 		base_config->ibs = available_fmt->input_pin_fmts[i].buffer_size;
1103 
1104 		dev_dbg(sdev->dev, "Init input audio formats for %s\n", swidget->widget->name);
1105 		sof_ipc4_dbg_audio_format(sdev->dev, &available_fmt->input_pin_fmts[i], 1);
1106 	}
1107 
1108 	if (available_fmt->num_output_formats && i < available_fmt->num_output_formats)
1109 		base_config->obs = available_fmt->output_pin_fmts[i].buffer_size;
1110 
1111 	/* Return the index of the matched format */
1112 	return i;
1113 }
1114 
1115 static void sof_ipc4_unprepare_copier_module(struct snd_sof_widget *swidget)
1116 {
1117 	struct sof_ipc4_copier *ipc4_copier = NULL;
1118 	struct snd_sof_widget *pipe_widget;
1119 	struct sof_ipc4_pipeline *pipeline;
1120 
1121 	/* reset pipeline memory usage */
1122 	pipe_widget = swidget->spipe->pipe_widget;
1123 	pipeline = pipe_widget->private;
1124 	pipeline->mem_usage = 0;
1125 
1126 	if (WIDGET_IS_AIF(swidget->id) || swidget->id == snd_soc_dapm_buffer) {
1127 		if (pipeline->use_chain_dma) {
1128 			pipeline->msg.primary = 0;
1129 			pipeline->msg.extension = 0;
1130 		}
1131 		ipc4_copier = swidget->private;
1132 	} else if (WIDGET_IS_DAI(swidget->id)) {
1133 		struct snd_sof_dai *dai = swidget->private;
1134 
1135 		ipc4_copier = dai->private;
1136 
1137 		if (pipeline->use_chain_dma) {
1138 			pipeline->msg.primary = 0;
1139 			pipeline->msg.extension = 0;
1140 		}
1141 
1142 		if (ipc4_copier->dai_type == SOF_DAI_INTEL_ALH) {
1143 			struct sof_ipc4_copier_data *copier_data = &ipc4_copier->data;
1144 			struct sof_ipc4_alh_configuration_blob *blob;
1145 			unsigned int group_id;
1146 
1147 			blob = (struct sof_ipc4_alh_configuration_blob *)ipc4_copier->copier_config;
1148 			if (blob->alh_cfg.count > 1) {
1149 				group_id = SOF_IPC4_NODE_INDEX(ipc4_copier->data.gtw_cfg.node_id) -
1150 					   ALH_MULTI_GTW_BASE;
1151 				ida_free(&alh_group_ida, group_id);
1152 			}
1153 
1154 			/* clear the node ID */
1155 			copier_data->gtw_cfg.node_id &= ~SOF_IPC4_NODE_INDEX_MASK;
1156 		}
1157 	}
1158 
1159 	if (ipc4_copier) {
1160 		kfree(ipc4_copier->ipc_config_data);
1161 		ipc4_copier->ipc_config_data = NULL;
1162 		ipc4_copier->ipc_config_size = 0;
1163 	}
1164 }
1165 
1166 #if IS_ENABLED(CONFIG_ACPI) && IS_ENABLED(CONFIG_SND_INTEL_NHLT)
1167 static int snd_sof_get_hw_config_params(struct snd_sof_dev *sdev, struct snd_sof_dai *dai,
1168 					int *sample_rate, int *channel_count, int *bit_depth)
1169 {
1170 	struct snd_soc_tplg_hw_config *hw_config;
1171 	struct snd_sof_dai_link *slink;
1172 	bool dai_link_found = false;
1173 	bool hw_cfg_found = false;
1174 	int i;
1175 
1176 	/* get current hw_config from link */
1177 	list_for_each_entry(slink, &sdev->dai_link_list, list) {
1178 		if (!strcmp(slink->link->name, dai->name)) {
1179 			dai_link_found = true;
1180 			break;
1181 		}
1182 	}
1183 
1184 	if (!dai_link_found) {
1185 		dev_err(sdev->dev, "%s: no DAI link found for DAI %s\n", __func__, dai->name);
1186 		return -EINVAL;
1187 	}
1188 
1189 	for (i = 0; i < slink->num_hw_configs; i++) {
1190 		hw_config = &slink->hw_configs[i];
1191 		if (dai->current_config == le32_to_cpu(hw_config->id)) {
1192 			hw_cfg_found = true;
1193 			break;
1194 		}
1195 	}
1196 
1197 	if (!hw_cfg_found) {
1198 		dev_err(sdev->dev, "%s: no matching hw_config found for DAI %s\n", __func__,
1199 			dai->name);
1200 		return -EINVAL;
1201 	}
1202 
1203 	*bit_depth = le32_to_cpu(hw_config->tdm_slot_width);
1204 	*channel_count = le32_to_cpu(hw_config->tdm_slots);
1205 	*sample_rate = le32_to_cpu(hw_config->fsync_rate);
1206 
1207 	dev_dbg(sdev->dev, "sample rate: %d sample width: %d channels: %d\n",
1208 		*sample_rate, *bit_depth, *channel_count);
1209 
1210 	return 0;
1211 }
1212 
1213 static int snd_sof_get_nhlt_endpoint_data(struct snd_sof_dev *sdev, struct snd_sof_dai *dai,
1214 					  struct snd_pcm_hw_params *params, u32 dai_index,
1215 					  u32 linktype, u8 dir, u32 **dst, u32 *len)
1216 {
1217 	struct sof_ipc4_fw_data *ipc4_data = sdev->private;
1218 	struct nhlt_specific_cfg *cfg;
1219 	int sample_rate, channel_count;
1220 	int bit_depth, ret;
1221 	u32 nhlt_type;
1222 
1223 	/* convert to NHLT type */
1224 	switch (linktype) {
1225 	case SOF_DAI_INTEL_DMIC:
1226 		nhlt_type = NHLT_LINK_DMIC;
1227 		bit_depth = params_width(params);
1228 		channel_count = params_channels(params);
1229 		sample_rate = params_rate(params);
1230 		break;
1231 	case SOF_DAI_INTEL_SSP:
1232 		nhlt_type = NHLT_LINK_SSP;
1233 		ret = snd_sof_get_hw_config_params(sdev, dai, &sample_rate, &channel_count,
1234 						   &bit_depth);
1235 		if (ret < 0)
1236 			return ret;
1237 		break;
1238 	default:
1239 		return 0;
1240 	}
1241 
1242 	dev_dbg(sdev->dev, "dai index %d nhlt type %d direction %d\n",
1243 		dai_index, nhlt_type, dir);
1244 
1245 	/* find NHLT blob with matching params */
1246 	cfg = intel_nhlt_get_endpoint_blob(sdev->dev, ipc4_data->nhlt, dai_index, nhlt_type,
1247 					   bit_depth, bit_depth, channel_count, sample_rate,
1248 					   dir, 0);
1249 
1250 	if (!cfg) {
1251 		dev_err(sdev->dev,
1252 			"no matching blob for sample rate: %d sample width: %d channels: %d\n",
1253 			sample_rate, bit_depth, channel_count);
1254 		return -EINVAL;
1255 	}
1256 
1257 	/* config length should be in dwords */
1258 	*len = cfg->size >> 2;
1259 	*dst = (u32 *)cfg->caps;
1260 
1261 	return 0;
1262 }
1263 #else
1264 static int snd_sof_get_nhlt_endpoint_data(struct snd_sof_dev *sdev, struct snd_sof_dai *dai,
1265 					  struct snd_pcm_hw_params *params, u32 dai_index,
1266 					  u32 linktype, u8 dir, u32 **dst, u32 *len)
1267 {
1268 	return 0;
1269 }
1270 #endif
1271 
1272 static int ipc4_set_fmt_mask(struct snd_mask *fmt, unsigned int bit_depth)
1273 {
1274 	switch (bit_depth) {
1275 	case 16:
1276 		snd_mask_set_format(fmt, SNDRV_PCM_FORMAT_S16_LE);
1277 		break;
1278 	case 24:
1279 		snd_mask_set_format(fmt, SNDRV_PCM_FORMAT_S24_LE);
1280 		break;
1281 	case 32:
1282 		snd_mask_set_format(fmt, SNDRV_PCM_FORMAT_S32_LE);
1283 		break;
1284 	default:
1285 		return -EINVAL;
1286 	}
1287 
1288 	return 0;
1289 }
1290 
1291 static int ipc4_copier_set_capture_fmt(struct snd_sof_dev *sdev,
1292 				       struct snd_pcm_hw_params *pipeline_params,
1293 				       struct snd_pcm_hw_params *fe_params,
1294 				       struct sof_ipc4_available_audio_format *available_fmt)
1295 {
1296 	struct sof_ipc4_audio_format *audio_fmt;
1297 	unsigned int sample_valid_bits;
1298 	bool multiple_formats = false;
1299 	bool fe_format_match = false;
1300 	struct snd_mask *fmt;
1301 	int i;
1302 
1303 	for (i = 0; i < available_fmt->num_output_formats; i++) {
1304 		unsigned int val;
1305 
1306 		audio_fmt = &available_fmt->output_pin_fmts[i].audio_fmt;
1307 		val = SOF_IPC4_AUDIO_FORMAT_CFG_V_BIT_DEPTH(audio_fmt->fmt_cfg);
1308 
1309 		if (i == 0)
1310 			sample_valid_bits = val;
1311 		else if (sample_valid_bits != val)
1312 			multiple_formats = true;
1313 
1314 		if (snd_pcm_format_width(params_format(fe_params)) == val)
1315 			fe_format_match = true;
1316 	}
1317 
1318 	fmt = hw_param_mask(pipeline_params, SNDRV_PCM_HW_PARAM_FORMAT);
1319 	snd_mask_none(fmt);
1320 
1321 	if (multiple_formats) {
1322 		if (fe_format_match) {
1323 			/* multiple formats defined and one matches FE */
1324 			snd_mask_set_format(fmt, params_format(fe_params));
1325 			return 0;
1326 		}
1327 
1328 		dev_err(sdev->dev, "Multiple audio formats for single dai_out not supported\n");
1329 		return -EINVAL;
1330 	}
1331 
1332 	return ipc4_set_fmt_mask(fmt, sample_valid_bits);
1333 }
1334 
1335 static int
1336 sof_ipc4_prepare_copier_module(struct snd_sof_widget *swidget,
1337 			       struct snd_pcm_hw_params *fe_params,
1338 			       struct snd_sof_platform_stream_params *platform_params,
1339 			       struct snd_pcm_hw_params *pipeline_params, int dir)
1340 {
1341 	struct sof_ipc4_available_audio_format *available_fmt;
1342 	struct snd_soc_component *scomp = swidget->scomp;
1343 	struct snd_sof_dev *sdev = snd_soc_component_get_drvdata(scomp);
1344 	struct sof_ipc4_pin_format *format_list_to_search;
1345 	struct sof_ipc4_copier_data *copier_data;
1346 	struct snd_pcm_hw_params *ref_params;
1347 	struct sof_ipc4_copier *ipc4_copier;
1348 	struct snd_sof_dai *dai;
1349 	struct snd_mask *fmt;
1350 	int out_sample_valid_bits;
1351 	void **ipc_config_data;
1352 	int *ipc_config_size;
1353 	u32 **data;
1354 	int ipc_size, ret;
1355 	u32 deep_buffer_dma_ms = 0;
1356 	u32 format_list_count;
1357 
1358 	dev_dbg(sdev->dev, "copier %s, type %d", swidget->widget->name, swidget->id);
1359 
1360 	switch (swidget->id) {
1361 	case snd_soc_dapm_aif_in:
1362 	case snd_soc_dapm_aif_out:
1363 	{
1364 		struct sof_ipc4_gtw_attributes *gtw_attr;
1365 		struct snd_sof_widget *pipe_widget;
1366 		struct sof_ipc4_pipeline *pipeline;
1367 
1368 		/* parse the deep buffer dma size */
1369 		ret = sof_update_ipc_object(scomp, &deep_buffer_dma_ms,
1370 					    SOF_COPIER_DEEP_BUFFER_TOKENS, swidget->tuples,
1371 					    swidget->num_tuples, sizeof(u32), 1);
1372 		if (ret) {
1373 			dev_err(scomp->dev, "Failed to parse deep buffer dma size for %s\n",
1374 				swidget->widget->name);
1375 			return ret;
1376 		}
1377 
1378 		ipc4_copier = (struct sof_ipc4_copier *)swidget->private;
1379 		gtw_attr = ipc4_copier->gtw_attr;
1380 		copier_data = &ipc4_copier->data;
1381 		available_fmt = &ipc4_copier->available_fmt;
1382 
1383 		pipe_widget = swidget->spipe->pipe_widget;
1384 		pipeline = pipe_widget->private;
1385 
1386 		if (pipeline->use_chain_dma) {
1387 			u32 host_dma_id;
1388 			u32 fifo_size;
1389 
1390 			host_dma_id = platform_params->stream_tag - 1;
1391 			pipeline->msg.primary |= SOF_IPC4_GLB_CHAIN_DMA_HOST_ID(host_dma_id);
1392 
1393 			/* Set SCS bit for S16_LE format only */
1394 			if (params_format(fe_params) == SNDRV_PCM_FORMAT_S16_LE)
1395 				pipeline->msg.primary |= SOF_IPC4_GLB_CHAIN_DMA_SCS_MASK;
1396 
1397 			/*
1398 			 * Despite its name the bitfield 'fifo_size' is used to define DMA buffer
1399 			 * size. The expression calculates 2ms buffer size.
1400 			 */
1401 			fifo_size = DIV_ROUND_UP((SOF_IPC4_CHAIN_DMA_BUF_SIZE_MS *
1402 						  params_rate(fe_params) *
1403 						  params_channels(fe_params) *
1404 						  params_physical_width(fe_params)), 8000);
1405 			pipeline->msg.extension |= SOF_IPC4_GLB_EXT_CHAIN_DMA_FIFO_SIZE(fifo_size);
1406 
1407 			/*
1408 			 * Chain DMA does not support stream timestamping, set node_id to invalid
1409 			 * to skip the code in sof_ipc4_get_stream_start_offset().
1410 			 */
1411 			copier_data->gtw_cfg.node_id = SOF_IPC4_INVALID_NODE_ID;
1412 
1413 			return 0;
1414 		}
1415 
1416 		/*
1417 		 * Use the input_pin_fmts to match pcm params for playback and the output_pin_fmts
1418 		 * for capture.
1419 		 */
1420 		if (dir == SNDRV_PCM_STREAM_PLAYBACK) {
1421 			format_list_to_search = available_fmt->input_pin_fmts;
1422 			format_list_count = available_fmt->num_input_formats;
1423 		} else {
1424 			format_list_to_search = available_fmt->output_pin_fmts;
1425 			format_list_count = available_fmt->num_output_formats;
1426 		}
1427 
1428 		copier_data->gtw_cfg.node_id &= ~SOF_IPC4_NODE_INDEX_MASK;
1429 		copier_data->gtw_cfg.node_id |=
1430 			SOF_IPC4_NODE_INDEX(platform_params->stream_tag - 1);
1431 
1432 		/* set gateway attributes */
1433 		gtw_attr->lp_buffer_alloc = pipeline->lp_mode;
1434 		ref_params = fe_params;
1435 		break;
1436 	}
1437 	case snd_soc_dapm_dai_in:
1438 	case snd_soc_dapm_dai_out:
1439 	{
1440 		struct snd_sof_widget *pipe_widget = swidget->spipe->pipe_widget;
1441 		struct sof_ipc4_pipeline *pipeline = pipe_widget->private;
1442 
1443 		if (pipeline->use_chain_dma)
1444 			return 0;
1445 
1446 		dai = swidget->private;
1447 
1448 		ipc4_copier = (struct sof_ipc4_copier *)dai->private;
1449 		copier_data = &ipc4_copier->data;
1450 		available_fmt = &ipc4_copier->available_fmt;
1451 		if (dir == SNDRV_PCM_STREAM_CAPTURE) {
1452 			format_list_to_search = available_fmt->output_pin_fmts;
1453 			format_list_count = available_fmt->num_output_formats;
1454 
1455 			ret = ipc4_copier_set_capture_fmt(sdev, pipeline_params, fe_params,
1456 							  available_fmt);
1457 			if (ret < 0)
1458 				return ret;
1459 		} else {
1460 			format_list_to_search = available_fmt->input_pin_fmts;
1461 			format_list_count = available_fmt->num_input_formats;
1462 		}
1463 
1464 		ref_params = pipeline_params;
1465 
1466 		ret = snd_sof_get_nhlt_endpoint_data(sdev, dai, fe_params, ipc4_copier->dai_index,
1467 						     ipc4_copier->dai_type, dir,
1468 						     &ipc4_copier->copier_config,
1469 						     &copier_data->gtw_cfg.config_length);
1470 		if (ret < 0)
1471 			return ret;
1472 
1473 		break;
1474 	}
1475 	case snd_soc_dapm_buffer:
1476 	{
1477 		ipc4_copier = (struct sof_ipc4_copier *)swidget->private;
1478 		copier_data = &ipc4_copier->data;
1479 		available_fmt = &ipc4_copier->available_fmt;
1480 
1481 		/* Use the input formats to match pcm params */
1482 		format_list_to_search = available_fmt->input_pin_fmts;
1483 		format_list_count = available_fmt->num_input_formats;
1484 		ref_params = pipeline_params;
1485 
1486 		break;
1487 	}
1488 	default:
1489 		dev_err(sdev->dev, "unsupported type %d for copier %s",
1490 			swidget->id, swidget->widget->name);
1491 		return -EINVAL;
1492 	}
1493 
1494 	/* set input and output audio formats */
1495 	ret = sof_ipc4_init_audio_fmt(sdev, swidget, &copier_data->base_config, ref_params,
1496 				      available_fmt, format_list_to_search, format_list_count);
1497 	if (ret < 0)
1498 		return ret;
1499 
1500 	/*
1501 	 * Set the output format. Current topology defines pin 0 input and output formats in pairs.
1502 	 * This assumes that the pin 0 formats are defined before all other pins.
1503 	 * So pick the output audio format with the same index as the chosen
1504 	 * input format. This logic will need to be updated when the format definitions
1505 	 * in topology change.
1506 	 */
1507 	memcpy(&copier_data->out_format, &available_fmt->output_pin_fmts[ret].audio_fmt,
1508 	       sizeof(struct sof_ipc4_audio_format));
1509 	dev_dbg(sdev->dev, "Output audio format for %s\n", swidget->widget->name);
1510 	sof_ipc4_dbg_audio_format(sdev->dev, &available_fmt->output_pin_fmts[ret], 1);
1511 
1512 	switch (swidget->id) {
1513 	case snd_soc_dapm_dai_in:
1514 	case snd_soc_dapm_dai_out:
1515 	{
1516 		/*
1517 		 * Only SOF_DAI_INTEL_ALH needs copier_data to set blob.
1518 		 * That's why only ALH dai's blob is set after sof_ipc4_init_audio_fmt
1519 		 */
1520 		if (ipc4_copier->dai_type == SOF_DAI_INTEL_ALH) {
1521 			struct sof_ipc4_alh_configuration_blob *blob;
1522 			struct sof_ipc4_copier_data *alh_data;
1523 			struct sof_ipc4_copier *alh_copier;
1524 			struct snd_sof_widget *w;
1525 			u32 ch_count = 0;
1526 			u32 ch_mask = 0;
1527 			u32 ch_map;
1528 			u32 step;
1529 			u32 mask;
1530 			int i;
1531 
1532 			blob = (struct sof_ipc4_alh_configuration_blob *)ipc4_copier->copier_config;
1533 
1534 			blob->gw_attr.lp_buffer_alloc = 0;
1535 
1536 			/* Get channel_mask from ch_map */
1537 			ch_map = copier_data->base_config.audio_fmt.ch_map;
1538 			for (i = 0; ch_map; i++) {
1539 				if ((ch_map & 0xf) != 0xf) {
1540 					ch_mask |= BIT(i);
1541 					ch_count++;
1542 				}
1543 				ch_map >>= 4;
1544 			}
1545 
1546 			step = ch_count / blob->alh_cfg.count;
1547 			mask =  GENMASK(step - 1, 0);
1548 			/*
1549 			 * Set each gtw_cfg.node_id to blob->alh_cfg.mapping[]
1550 			 * for all widgets with the same stream name
1551 			 */
1552 			i = 0;
1553 			list_for_each_entry(w, &sdev->widget_list, list) {
1554 				if (w->widget->sname &&
1555 				    strcmp(w->widget->sname, swidget->widget->sname))
1556 					continue;
1557 
1558 				dai = w->private;
1559 				alh_copier = (struct sof_ipc4_copier *)dai->private;
1560 				alh_data = &alh_copier->data;
1561 				blob->alh_cfg.mapping[i].alh_id = alh_data->gtw_cfg.node_id;
1562 				/*
1563 				 * Set the same channel mask for playback as the audio data is
1564 				 * duplicated for all speakers. For capture, split the channels
1565 				 * among the aggregated DAIs. For example, with 4 channels on 2
1566 				 * aggregated DAIs, the channel_mask should be 0x3 and 0xc for the
1567 				 * two DAI's.
1568 				 * The channel masks used depend on the cpu_dais used in the
1569 				 * dailink at the machine driver level, which actually comes from
1570 				 * the tables in soc_acpi files depending on the _ADR and devID
1571 				 * registers for each codec.
1572 				 */
1573 				if (w->id == snd_soc_dapm_dai_in)
1574 					blob->alh_cfg.mapping[i].channel_mask = ch_mask;
1575 				else
1576 					blob->alh_cfg.mapping[i].channel_mask = mask << (step * i);
1577 
1578 				i++;
1579 			}
1580 			if (blob->alh_cfg.count > 1) {
1581 				int group_id;
1582 
1583 				group_id = ida_alloc_max(&alh_group_ida, ALH_MULTI_GTW_COUNT - 1,
1584 							 GFP_KERNEL);
1585 
1586 				if (group_id < 0)
1587 					return group_id;
1588 
1589 				/* add multi-gateway base */
1590 				group_id += ALH_MULTI_GTW_BASE;
1591 				copier_data->gtw_cfg.node_id &= ~SOF_IPC4_NODE_INDEX_MASK;
1592 				copier_data->gtw_cfg.node_id |= SOF_IPC4_NODE_INDEX(group_id);
1593 			}
1594 		}
1595 	}
1596 	}
1597 
1598 	/* modify the input params for the next widget */
1599 	fmt = hw_param_mask(pipeline_params, SNDRV_PCM_HW_PARAM_FORMAT);
1600 	out_sample_valid_bits =
1601 		SOF_IPC4_AUDIO_FORMAT_CFG_V_BIT_DEPTH(copier_data->out_format.fmt_cfg);
1602 	snd_mask_none(fmt);
1603 	ret = ipc4_set_fmt_mask(fmt, out_sample_valid_bits);
1604 	if (ret)
1605 		return ret;
1606 
1607 	/*
1608 	 * Set the gateway dma_buffer_size to 2ms buffer size to meet the FW expectation. In the
1609 	 * deep buffer case, set the dma_buffer_size depending on the deep_buffer_dma_ms set
1610 	 * in topology.
1611 	 */
1612 	switch (swidget->id) {
1613 	case snd_soc_dapm_dai_in:
1614 		copier_data->gtw_cfg.dma_buffer_size =
1615 			SOF_IPC4_MIN_DMA_BUFFER_SIZE * copier_data->base_config.ibs;
1616 		break;
1617 	case snd_soc_dapm_aif_in:
1618 			copier_data->gtw_cfg.dma_buffer_size =
1619 				max((u32)SOF_IPC4_MIN_DMA_BUFFER_SIZE, deep_buffer_dma_ms) *
1620 					copier_data->base_config.ibs;
1621 		break;
1622 	case snd_soc_dapm_dai_out:
1623 	case snd_soc_dapm_aif_out:
1624 		copier_data->gtw_cfg.dma_buffer_size =
1625 			SOF_IPC4_MIN_DMA_BUFFER_SIZE * copier_data->base_config.obs;
1626 		break;
1627 	default:
1628 		break;
1629 	}
1630 
1631 	data = &ipc4_copier->copier_config;
1632 	ipc_config_size = &ipc4_copier->ipc_config_size;
1633 	ipc_config_data = &ipc4_copier->ipc_config_data;
1634 
1635 	/* config_length is DWORD based */
1636 	ipc_size = sizeof(*copier_data) + copier_data->gtw_cfg.config_length * 4;
1637 
1638 	dev_dbg(sdev->dev, "copier %s, IPC size is %d", swidget->widget->name, ipc_size);
1639 
1640 	*ipc_config_data = kzalloc(ipc_size, GFP_KERNEL);
1641 	if (!*ipc_config_data)
1642 		return -ENOMEM;
1643 
1644 	*ipc_config_size = ipc_size;
1645 
1646 	/* copy IPC data */
1647 	memcpy(*ipc_config_data, (void *)copier_data, sizeof(*copier_data));
1648 	if (copier_data->gtw_cfg.config_length)
1649 		memcpy(*ipc_config_data + sizeof(*copier_data),
1650 		       *data, copier_data->gtw_cfg.config_length * 4);
1651 
1652 	/* update pipeline memory usage */
1653 	sof_ipc4_update_pipeline_mem_usage(sdev, swidget, &copier_data->base_config);
1654 
1655 	return 0;
1656 }
1657 
1658 static int sof_ipc4_prepare_gain_module(struct snd_sof_widget *swidget,
1659 					struct snd_pcm_hw_params *fe_params,
1660 					struct snd_sof_platform_stream_params *platform_params,
1661 					struct snd_pcm_hw_params *pipeline_params, int dir)
1662 {
1663 	struct snd_soc_component *scomp = swidget->scomp;
1664 	struct snd_sof_dev *sdev = snd_soc_component_get_drvdata(scomp);
1665 	struct sof_ipc4_gain *gain = swidget->private;
1666 	struct sof_ipc4_available_audio_format *available_fmt = &gain->available_fmt;
1667 	int ret;
1668 
1669 	ret = sof_ipc4_init_audio_fmt(sdev, swidget, &gain->base_config,
1670 				      pipeline_params, available_fmt,
1671 				      available_fmt->input_pin_fmts,
1672 				      available_fmt->num_input_formats);
1673 	if (ret < 0)
1674 		return ret;
1675 
1676 	/* update pipeline memory usage */
1677 	sof_ipc4_update_pipeline_mem_usage(sdev, swidget, &gain->base_config);
1678 
1679 	return 0;
1680 }
1681 
1682 static int sof_ipc4_prepare_mixer_module(struct snd_sof_widget *swidget,
1683 					 struct snd_pcm_hw_params *fe_params,
1684 					 struct snd_sof_platform_stream_params *platform_params,
1685 					 struct snd_pcm_hw_params *pipeline_params, int dir)
1686 {
1687 	struct snd_soc_component *scomp = swidget->scomp;
1688 	struct snd_sof_dev *sdev = snd_soc_component_get_drvdata(scomp);
1689 	struct sof_ipc4_mixer *mixer = swidget->private;
1690 	struct sof_ipc4_available_audio_format *available_fmt = &mixer->available_fmt;
1691 	int ret;
1692 
1693 	ret = sof_ipc4_init_audio_fmt(sdev, swidget, &mixer->base_config,
1694 				      pipeline_params, available_fmt,
1695 				      available_fmt->input_pin_fmts,
1696 				      available_fmt->num_input_formats);
1697 	if (ret < 0)
1698 		return ret;
1699 
1700 	/* update pipeline memory usage */
1701 	sof_ipc4_update_pipeline_mem_usage(sdev, swidget, &mixer->base_config);
1702 
1703 	return 0;
1704 }
1705 
1706 static int sof_ipc4_prepare_src_module(struct snd_sof_widget *swidget,
1707 				       struct snd_pcm_hw_params *fe_params,
1708 				       struct snd_sof_platform_stream_params *platform_params,
1709 				       struct snd_pcm_hw_params *pipeline_params, int dir)
1710 {
1711 	struct snd_soc_component *scomp = swidget->scomp;
1712 	struct snd_sof_dev *sdev = snd_soc_component_get_drvdata(scomp);
1713 	struct sof_ipc4_src *src = swidget->private;
1714 	struct sof_ipc4_available_audio_format *available_fmt = &src->available_fmt;
1715 	struct snd_interval *rate;
1716 	int ret;
1717 
1718 	ret = sof_ipc4_init_audio_fmt(sdev, swidget, &src->base_config,
1719 				      pipeline_params, available_fmt,
1720 				      available_fmt->input_pin_fmts,
1721 				      available_fmt->num_input_formats);
1722 	if (ret < 0)
1723 		return ret;
1724 
1725 	/* update pipeline memory usage */
1726 	sof_ipc4_update_pipeline_mem_usage(sdev, swidget, &src->base_config);
1727 
1728 	/* update pipeline_params for sink widgets */
1729 	rate = hw_param_interval(pipeline_params, SNDRV_PCM_HW_PARAM_RATE);
1730 	rate->min = src->sink_rate;
1731 	rate->max = rate->min;
1732 
1733 	return 0;
1734 }
1735 
1736 static int
1737 sof_ipc4_process_set_pin_formats(struct snd_sof_widget *swidget, int pin_type)
1738 {
1739 	struct sof_ipc4_process *process = swidget->private;
1740 	struct sof_ipc4_base_module_cfg_ext *base_cfg_ext = process->base_config_ext;
1741 	struct sof_ipc4_available_audio_format *available_fmt = &process->available_fmt;
1742 	struct sof_ipc4_pin_format *pin_format, *format_list_to_search;
1743 	struct snd_soc_component *scomp = swidget->scomp;
1744 	int num_pins, format_list_count;
1745 	int pin_format_offset = 0;
1746 	int i, j;
1747 
1748 	/* set number of pins, offset of pin format and format list to search based on pin type */
1749 	if (pin_type == SOF_PIN_TYPE_INPUT) {
1750 		num_pins = swidget->num_input_pins;
1751 		format_list_to_search = available_fmt->input_pin_fmts;
1752 		format_list_count = available_fmt->num_input_formats;
1753 	} else {
1754 		num_pins = swidget->num_output_pins;
1755 		pin_format_offset = swidget->num_input_pins;
1756 		format_list_to_search = available_fmt->output_pin_fmts;
1757 		format_list_count = available_fmt->num_output_formats;
1758 	}
1759 
1760 	for (i = pin_format_offset; i < num_pins + pin_format_offset; i++) {
1761 		pin_format = &base_cfg_ext->pin_formats[i];
1762 
1763 		/* Pin 0 audio formats are derived from the base config input/output format */
1764 		if (i == pin_format_offset) {
1765 			if (pin_type == SOF_PIN_TYPE_INPUT) {
1766 				pin_format->buffer_size = process->base_config.ibs;
1767 				pin_format->audio_fmt = process->base_config.audio_fmt;
1768 			} else {
1769 				pin_format->buffer_size = process->base_config.obs;
1770 				pin_format->audio_fmt = process->output_format;
1771 			}
1772 			continue;
1773 		}
1774 
1775 		/*
1776 		 * For all other pins, find the pin formats from those set in topology. If there
1777 		 * is more than one format specified for a pin, this will pick the first available
1778 		 * one.
1779 		 */
1780 		for (j = 0; j < format_list_count; j++) {
1781 			struct sof_ipc4_pin_format *pin_format_item = &format_list_to_search[j];
1782 
1783 			if (pin_format_item->pin_index == i - pin_format_offset) {
1784 				*pin_format = *pin_format_item;
1785 				break;
1786 			}
1787 		}
1788 
1789 		if (j == format_list_count) {
1790 			dev_err(scomp->dev, "%s pin %d format not found for %s\n",
1791 				(pin_type == SOF_PIN_TYPE_INPUT) ? "input" : "output",
1792 				i - pin_format_offset, swidget->widget->name);
1793 			return -EINVAL;
1794 		}
1795 	}
1796 
1797 	return 0;
1798 }
1799 
1800 static int sof_ipc4_process_add_base_cfg_extn(struct snd_sof_widget *swidget)
1801 {
1802 	int ret, i;
1803 
1804 	/* copy input and output pin formats */
1805 	for (i = 0; i <= SOF_PIN_TYPE_OUTPUT; i++) {
1806 		ret = sof_ipc4_process_set_pin_formats(swidget, i);
1807 		if (ret < 0)
1808 			return ret;
1809 	}
1810 
1811 	return 0;
1812 }
1813 
1814 static int sof_ipc4_prepare_process_module(struct snd_sof_widget *swidget,
1815 					   struct snd_pcm_hw_params *fe_params,
1816 					   struct snd_sof_platform_stream_params *platform_params,
1817 					   struct snd_pcm_hw_params *pipeline_params, int dir)
1818 {
1819 	struct snd_soc_component *scomp = swidget->scomp;
1820 	struct snd_sof_dev *sdev = snd_soc_component_get_drvdata(scomp);
1821 	struct sof_ipc4_process *process = swidget->private;
1822 	struct sof_ipc4_available_audio_format *available_fmt = &process->available_fmt;
1823 	void *cfg = process->ipc_config_data;
1824 	int ret;
1825 
1826 	ret = sof_ipc4_init_audio_fmt(sdev, swidget, &process->base_config,
1827 				      pipeline_params, available_fmt,
1828 				      available_fmt->input_pin_fmts,
1829 				      available_fmt->num_input_formats);
1830 	if (ret < 0)
1831 		return ret;
1832 
1833 	/* copy Pin 0 output format */
1834 	if (available_fmt->num_output_formats && ret < available_fmt->num_output_formats &&
1835 	    !available_fmt->output_pin_fmts[ret].pin_index) {
1836 		memcpy(&process->output_format, &available_fmt->output_pin_fmts[ret].audio_fmt,
1837 		       sizeof(struct sof_ipc4_audio_format));
1838 
1839 		/* modify the pipeline params with the pin 0 output format */
1840 		ret = sof_ipc4_update_hw_params(sdev, pipeline_params, &process->output_format);
1841 		if (ret)
1842 			return ret;
1843 	}
1844 
1845 	/* update pipeline memory usage */
1846 	sof_ipc4_update_pipeline_mem_usage(sdev, swidget, &process->base_config);
1847 
1848 	/* ipc_config_data is composed of the base_config followed by an optional extension */
1849 	memcpy(cfg, &process->base_config, sizeof(struct sof_ipc4_base_module_cfg));
1850 	cfg += sizeof(struct sof_ipc4_base_module_cfg);
1851 
1852 	if (process->init_config == SOF_IPC4_MODULE_INIT_CONFIG_TYPE_BASE_CFG_WITH_EXT) {
1853 		struct sof_ipc4_base_module_cfg_ext *base_cfg_ext = process->base_config_ext;
1854 
1855 		ret = sof_ipc4_process_add_base_cfg_extn(swidget);
1856 		if (ret < 0)
1857 			return ret;
1858 
1859 		memcpy(cfg, base_cfg_ext, process->base_config_ext_size);
1860 	}
1861 
1862 	return 0;
1863 }
1864 
1865 static int sof_ipc4_control_load_volume(struct snd_sof_dev *sdev, struct snd_sof_control *scontrol)
1866 {
1867 	struct sof_ipc4_control_data *control_data;
1868 	struct sof_ipc4_msg *msg;
1869 	int i;
1870 
1871 	scontrol->size = struct_size(control_data, chanv, scontrol->num_channels);
1872 
1873 	/* scontrol->ipc_control_data will be freed in sof_control_unload */
1874 	scontrol->ipc_control_data = kzalloc(scontrol->size, GFP_KERNEL);
1875 	if (!scontrol->ipc_control_data)
1876 		return -ENOMEM;
1877 
1878 	control_data = scontrol->ipc_control_data;
1879 	control_data->index = scontrol->index;
1880 
1881 	msg = &control_data->msg;
1882 	msg->primary = SOF_IPC4_MSG_TYPE_SET(SOF_IPC4_MOD_LARGE_CONFIG_SET);
1883 	msg->primary |= SOF_IPC4_MSG_DIR(SOF_IPC4_MSG_REQUEST);
1884 	msg->primary |= SOF_IPC4_MSG_TARGET(SOF_IPC4_MODULE_MSG);
1885 
1886 	msg->extension = SOF_IPC4_MOD_EXT_MSG_PARAM_ID(SOF_IPC4_GAIN_PARAM_ID);
1887 
1888 	/* set default volume values to 0dB in control */
1889 	for (i = 0; i < scontrol->num_channels; i++) {
1890 		control_data->chanv[i].channel = i;
1891 		control_data->chanv[i].value = SOF_IPC4_VOL_ZERO_DB;
1892 	}
1893 
1894 	return 0;
1895 }
1896 
1897 static int sof_ipc4_control_load_bytes(struct snd_sof_dev *sdev, struct snd_sof_control *scontrol)
1898 {
1899 	struct sof_ipc4_control_data *control_data;
1900 	struct sof_ipc4_msg *msg;
1901 	int ret;
1902 
1903 	if (scontrol->max_size < (sizeof(*control_data) + sizeof(struct sof_abi_hdr))) {
1904 		dev_err(sdev->dev, "insufficient size for a bytes control %s: %zu.\n",
1905 			scontrol->name, scontrol->max_size);
1906 		return -EINVAL;
1907 	}
1908 
1909 	if (scontrol->priv_size > scontrol->max_size - sizeof(*control_data)) {
1910 		dev_err(sdev->dev, "scontrol %s bytes data size %zu exceeds max %zu.\n",
1911 			scontrol->name, scontrol->priv_size,
1912 			scontrol->max_size - sizeof(*control_data));
1913 		return -EINVAL;
1914 	}
1915 
1916 	scontrol->size = sizeof(struct sof_ipc4_control_data) + scontrol->priv_size;
1917 
1918 	scontrol->ipc_control_data = kzalloc(scontrol->max_size, GFP_KERNEL);
1919 	if (!scontrol->ipc_control_data)
1920 		return -ENOMEM;
1921 
1922 	control_data = scontrol->ipc_control_data;
1923 	control_data->index = scontrol->index;
1924 	if (scontrol->priv_size > 0) {
1925 		memcpy(control_data->data, scontrol->priv, scontrol->priv_size);
1926 		kfree(scontrol->priv);
1927 		scontrol->priv = NULL;
1928 
1929 		if (control_data->data->magic != SOF_IPC4_ABI_MAGIC) {
1930 			dev_err(sdev->dev, "Wrong ABI magic (%#x) for control: %s\n",
1931 				control_data->data->magic, scontrol->name);
1932 			ret = -EINVAL;
1933 			goto err;
1934 		}
1935 
1936 		/* TODO: check the ABI version */
1937 
1938 		if (control_data->data->size + sizeof(struct sof_abi_hdr) !=
1939 		    scontrol->priv_size) {
1940 			dev_err(sdev->dev, "Control %s conflict in bytes %zu vs. priv size %zu.\n",
1941 				scontrol->name,
1942 				control_data->data->size + sizeof(struct sof_abi_hdr),
1943 				scontrol->priv_size);
1944 			ret = -EINVAL;
1945 			goto err;
1946 		}
1947 	}
1948 
1949 	msg = &control_data->msg;
1950 	msg->primary = SOF_IPC4_MSG_TYPE_SET(SOF_IPC4_MOD_LARGE_CONFIG_SET);
1951 	msg->primary |= SOF_IPC4_MSG_DIR(SOF_IPC4_MSG_REQUEST);
1952 	msg->primary |= SOF_IPC4_MSG_TARGET(SOF_IPC4_MODULE_MSG);
1953 
1954 	return 0;
1955 
1956 err:
1957 	kfree(scontrol->ipc_control_data);
1958 	scontrol->ipc_control_data = NULL;
1959 	return ret;
1960 }
1961 
1962 static int sof_ipc4_control_setup(struct snd_sof_dev *sdev, struct snd_sof_control *scontrol)
1963 {
1964 	switch (scontrol->info_type) {
1965 	case SND_SOC_TPLG_CTL_VOLSW:
1966 	case SND_SOC_TPLG_CTL_VOLSW_SX:
1967 	case SND_SOC_TPLG_CTL_VOLSW_XR_SX:
1968 		return sof_ipc4_control_load_volume(sdev, scontrol);
1969 	case SND_SOC_TPLG_CTL_BYTES:
1970 		return sof_ipc4_control_load_bytes(sdev, scontrol);
1971 	default:
1972 		break;
1973 	}
1974 
1975 	return 0;
1976 }
1977 
1978 static int sof_ipc4_widget_setup(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget)
1979 {
1980 	struct snd_sof_widget *pipe_widget = swidget->spipe->pipe_widget;
1981 	struct sof_ipc4_fw_data *ipc4_data = sdev->private;
1982 	struct sof_ipc4_pipeline *pipeline;
1983 	struct sof_ipc4_msg *msg;
1984 	void *ipc_data = NULL;
1985 	u32 ipc_size = 0;
1986 	int ret;
1987 
1988 	switch (swidget->id) {
1989 	case snd_soc_dapm_scheduler:
1990 		pipeline = swidget->private;
1991 
1992 		if (pipeline->use_chain_dma) {
1993 			dev_warn(sdev->dev, "use_chain_dma set for scheduler %s",
1994 				 swidget->widget->name);
1995 			return 0;
1996 		}
1997 
1998 		dev_dbg(sdev->dev, "pipeline: %d memory pages: %d\n", swidget->pipeline_id,
1999 			pipeline->mem_usage);
2000 
2001 		msg = &pipeline->msg;
2002 		msg->primary |= pipeline->mem_usage;
2003 
2004 		swidget->instance_id = ida_alloc_max(&pipeline_ida, ipc4_data->max_num_pipelines,
2005 						     GFP_KERNEL);
2006 		if (swidget->instance_id < 0) {
2007 			dev_err(sdev->dev, "failed to assign pipeline id for %s: %d\n",
2008 				swidget->widget->name, swidget->instance_id);
2009 			return swidget->instance_id;
2010 		}
2011 		msg->primary &= ~SOF_IPC4_GLB_PIPE_INSTANCE_MASK;
2012 		msg->primary |= SOF_IPC4_GLB_PIPE_INSTANCE_ID(swidget->instance_id);
2013 		break;
2014 	case snd_soc_dapm_aif_in:
2015 	case snd_soc_dapm_aif_out:
2016 	case snd_soc_dapm_buffer:
2017 	{
2018 		struct sof_ipc4_copier *ipc4_copier = swidget->private;
2019 
2020 		pipeline = pipe_widget->private;
2021 		if (pipeline->use_chain_dma)
2022 			return 0;
2023 
2024 		ipc_size = ipc4_copier->ipc_config_size;
2025 		ipc_data = ipc4_copier->ipc_config_data;
2026 
2027 		msg = &ipc4_copier->msg;
2028 		break;
2029 	}
2030 	case snd_soc_dapm_dai_in:
2031 	case snd_soc_dapm_dai_out:
2032 	{
2033 		struct snd_sof_dai *dai = swidget->private;
2034 		struct sof_ipc4_copier *ipc4_copier = dai->private;
2035 
2036 		pipeline = pipe_widget->private;
2037 		if (pipeline->use_chain_dma)
2038 			return 0;
2039 
2040 		ipc_size = ipc4_copier->ipc_config_size;
2041 		ipc_data = ipc4_copier->ipc_config_data;
2042 
2043 		msg = &ipc4_copier->msg;
2044 		break;
2045 	}
2046 	case snd_soc_dapm_pga:
2047 	{
2048 		struct sof_ipc4_gain *gain = swidget->private;
2049 
2050 		ipc_size = sizeof(struct sof_ipc4_base_module_cfg) +
2051 			   sizeof(struct sof_ipc4_gain_data);
2052 		ipc_data = gain;
2053 
2054 		msg = &gain->msg;
2055 		break;
2056 	}
2057 	case snd_soc_dapm_mixer:
2058 	{
2059 		struct sof_ipc4_mixer *mixer = swidget->private;
2060 
2061 		ipc_size = sizeof(mixer->base_config);
2062 		ipc_data = &mixer->base_config;
2063 
2064 		msg = &mixer->msg;
2065 		break;
2066 	}
2067 	case snd_soc_dapm_src:
2068 	{
2069 		struct sof_ipc4_src *src = swidget->private;
2070 
2071 		ipc_size = sizeof(struct sof_ipc4_base_module_cfg) + sizeof(src->sink_rate);
2072 		ipc_data = src;
2073 
2074 		msg = &src->msg;
2075 		break;
2076 	}
2077 	case snd_soc_dapm_effect:
2078 	{
2079 		struct sof_ipc4_process *process = swidget->private;
2080 
2081 		if (!process->ipc_config_size) {
2082 			dev_err(sdev->dev, "module %s has no config data!\n",
2083 				swidget->widget->name);
2084 			return -EINVAL;
2085 		}
2086 
2087 		ipc_size = process->ipc_config_size;
2088 		ipc_data = process->ipc_config_data;
2089 
2090 		msg = &process->msg;
2091 		break;
2092 	}
2093 	default:
2094 		dev_err(sdev->dev, "widget type %d not supported", swidget->id);
2095 		return -EINVAL;
2096 	}
2097 
2098 	if (swidget->id != snd_soc_dapm_scheduler) {
2099 		ret = sof_ipc4_widget_assign_instance_id(sdev, swidget);
2100 		if (ret < 0) {
2101 			dev_err(sdev->dev, "failed to assign instance id for %s\n",
2102 				swidget->widget->name);
2103 			return ret;
2104 		}
2105 
2106 		msg->primary &= ~SOF_IPC4_MOD_INSTANCE_MASK;
2107 		msg->primary |= SOF_IPC4_MOD_INSTANCE(swidget->instance_id);
2108 
2109 		msg->extension &= ~SOF_IPC4_MOD_EXT_PARAM_SIZE_MASK;
2110 		msg->extension |= ipc_size >> 2;
2111 
2112 		msg->extension &= ~SOF_IPC4_MOD_EXT_PPL_ID_MASK;
2113 		msg->extension |= SOF_IPC4_MOD_EXT_PPL_ID(pipe_widget->instance_id);
2114 	}
2115 	dev_dbg(sdev->dev, "Create widget %s instance %d - pipe %d - core %d\n",
2116 		swidget->widget->name, swidget->instance_id, swidget->pipeline_id, swidget->core);
2117 
2118 	msg->data_size = ipc_size;
2119 	msg->data_ptr = ipc_data;
2120 
2121 	ret = sof_ipc_tx_message_no_reply(sdev->ipc, msg, ipc_size);
2122 	if (ret < 0) {
2123 		dev_err(sdev->dev, "failed to create module %s\n", swidget->widget->name);
2124 
2125 		if (swidget->id != snd_soc_dapm_scheduler) {
2126 			struct sof_ipc4_fw_module *fw_module = swidget->module_info;
2127 
2128 			ida_free(&fw_module->m_ida, swidget->instance_id);
2129 		} else {
2130 			ida_free(&pipeline_ida, swidget->instance_id);
2131 		}
2132 	}
2133 
2134 	return ret;
2135 }
2136 
2137 static int sof_ipc4_widget_free(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget)
2138 {
2139 	struct sof_ipc4_fw_module *fw_module = swidget->module_info;
2140 	struct sof_ipc4_fw_data *ipc4_data = sdev->private;
2141 	int ret = 0;
2142 
2143 	mutex_lock(&ipc4_data->pipeline_state_mutex);
2144 
2145 	/* freeing a pipeline frees all the widgets associated with it */
2146 	if (swidget->id == snd_soc_dapm_scheduler) {
2147 		struct sof_ipc4_pipeline *pipeline = swidget->private;
2148 		struct sof_ipc4_msg msg = {{ 0 }};
2149 		u32 header;
2150 
2151 		if (pipeline->use_chain_dma) {
2152 			dev_warn(sdev->dev, "use_chain_dma set for scheduler %s",
2153 				 swidget->widget->name);
2154 			mutex_unlock(&ipc4_data->pipeline_state_mutex);
2155 			return 0;
2156 		}
2157 
2158 		header = SOF_IPC4_GLB_PIPE_INSTANCE_ID(swidget->instance_id);
2159 		header |= SOF_IPC4_MSG_TYPE_SET(SOF_IPC4_GLB_DELETE_PIPELINE);
2160 		header |= SOF_IPC4_MSG_DIR(SOF_IPC4_MSG_REQUEST);
2161 		header |= SOF_IPC4_MSG_TARGET(SOF_IPC4_FW_GEN_MSG);
2162 
2163 		msg.primary = header;
2164 
2165 		ret = sof_ipc_tx_message_no_reply(sdev->ipc, &msg, 0);
2166 		if (ret < 0)
2167 			dev_err(sdev->dev, "failed to free pipeline widget %s\n",
2168 				swidget->widget->name);
2169 
2170 		pipeline->mem_usage = 0;
2171 		pipeline->state = SOF_IPC4_PIPE_UNINITIALIZED;
2172 		ida_free(&pipeline_ida, swidget->instance_id);
2173 	} else {
2174 		struct snd_sof_widget *pipe_widget = swidget->spipe->pipe_widget;
2175 		struct sof_ipc4_pipeline *pipeline = pipe_widget->private;
2176 
2177 		if (!pipeline->use_chain_dma)
2178 			ida_free(&fw_module->m_ida, swidget->instance_id);
2179 	}
2180 
2181 	mutex_unlock(&ipc4_data->pipeline_state_mutex);
2182 
2183 	return ret;
2184 }
2185 
2186 static int sof_ipc4_get_queue_id(struct snd_sof_widget *src_widget,
2187 				 struct snd_sof_widget *sink_widget, bool pin_type)
2188 {
2189 	struct snd_sof_widget *current_swidget;
2190 	struct snd_soc_component *scomp;
2191 	struct ida *queue_ida;
2192 	const char *buddy_name;
2193 	char **pin_binding;
2194 	u32 num_pins;
2195 	int i;
2196 
2197 	if (pin_type == SOF_PIN_TYPE_OUTPUT) {
2198 		current_swidget = src_widget;
2199 		pin_binding = src_widget->output_pin_binding;
2200 		queue_ida = &src_widget->output_queue_ida;
2201 		num_pins = src_widget->num_output_pins;
2202 		buddy_name = sink_widget->widget->name;
2203 	} else {
2204 		current_swidget = sink_widget;
2205 		pin_binding = sink_widget->input_pin_binding;
2206 		queue_ida = &sink_widget->input_queue_ida;
2207 		num_pins = sink_widget->num_input_pins;
2208 		buddy_name = src_widget->widget->name;
2209 	}
2210 
2211 	scomp = current_swidget->scomp;
2212 
2213 	if (num_pins < 1) {
2214 		dev_err(scomp->dev, "invalid %s num_pins: %d for queue allocation for %s\n",
2215 			(pin_type == SOF_PIN_TYPE_OUTPUT ? "output" : "input"),
2216 			num_pins, current_swidget->widget->name);
2217 		return -EINVAL;
2218 	}
2219 
2220 	/* If there is only one input/output pin, queue id must be 0 */
2221 	if (num_pins == 1)
2222 		return 0;
2223 
2224 	/* Allocate queue ID from pin binding array if it is defined in topology. */
2225 	if (pin_binding) {
2226 		for (i = 0; i < num_pins; i++) {
2227 			if (!strcmp(pin_binding[i], buddy_name))
2228 				return i;
2229 		}
2230 		/*
2231 		 * Fail if no queue ID found from pin binding array, so that we don't
2232 		 * mixed use pin binding array and ida for queue ID allocation.
2233 		 */
2234 		dev_err(scomp->dev, "no %s queue id found from pin binding array for %s\n",
2235 			(pin_type == SOF_PIN_TYPE_OUTPUT ? "output" : "input"),
2236 			current_swidget->widget->name);
2237 		return -EINVAL;
2238 	}
2239 
2240 	/* If no pin binding array specified in topology, use ida to allocate one */
2241 	return ida_alloc_max(queue_ida, num_pins, GFP_KERNEL);
2242 }
2243 
2244 static void sof_ipc4_put_queue_id(struct snd_sof_widget *swidget, int queue_id,
2245 				  bool pin_type)
2246 {
2247 	struct ida *queue_ida;
2248 	char **pin_binding;
2249 	int num_pins;
2250 
2251 	if (pin_type == SOF_PIN_TYPE_OUTPUT) {
2252 		pin_binding = swidget->output_pin_binding;
2253 		queue_ida = &swidget->output_queue_ida;
2254 		num_pins = swidget->num_output_pins;
2255 	} else {
2256 		pin_binding = swidget->input_pin_binding;
2257 		queue_ida = &swidget->input_queue_ida;
2258 		num_pins = swidget->num_input_pins;
2259 	}
2260 
2261 	/* Nothing to free if queue ID is not allocated with ida. */
2262 	if (num_pins == 1 || pin_binding)
2263 		return;
2264 
2265 	ida_free(queue_ida, queue_id);
2266 }
2267 
2268 static int sof_ipc4_set_copier_sink_format(struct snd_sof_dev *sdev,
2269 					   struct snd_sof_widget *src_widget,
2270 					   struct snd_sof_widget *sink_widget,
2271 					   int sink_id)
2272 {
2273 	struct sof_ipc4_copier_config_set_sink_format format;
2274 	struct sof_ipc4_base_module_cfg *src_config;
2275 	const struct sof_ipc4_audio_format *pin_fmt;
2276 	struct sof_ipc4_fw_module *fw_module;
2277 	struct sof_ipc4_msg msg = {{ 0 }};
2278 	u32 header, extension;
2279 
2280 	dev_dbg(sdev->dev, "%s set copier sink %d format\n",
2281 		src_widget->widget->name, sink_id);
2282 
2283 	if (WIDGET_IS_DAI(src_widget->id)) {
2284 		struct snd_sof_dai *dai = src_widget->private;
2285 
2286 		src_config = dai->private;
2287 	} else {
2288 		src_config = src_widget->private;
2289 	}
2290 
2291 	fw_module = src_widget->module_info;
2292 
2293 	format.sink_id = sink_id;
2294 	memcpy(&format.source_fmt, &src_config->audio_fmt, sizeof(format.source_fmt));
2295 
2296 	pin_fmt = sof_ipc4_get_input_pin_audio_fmt(sink_widget, sink_id);
2297 	if (!pin_fmt) {
2298 		dev_err(sdev->dev, "Unable to get pin %d format for %s",
2299 			sink_id, sink_widget->widget->name);
2300 		return -EINVAL;
2301 	}
2302 
2303 	memcpy(&format.sink_fmt, pin_fmt, sizeof(format.sink_fmt));
2304 
2305 	msg.data_size = sizeof(format);
2306 	msg.data_ptr = &format;
2307 
2308 	header = fw_module->man4_module_entry.id;
2309 	header |= SOF_IPC4_MOD_INSTANCE(src_widget->instance_id);
2310 	header |= SOF_IPC4_MSG_TYPE_SET(SOF_IPC4_MOD_LARGE_CONFIG_SET);
2311 	header |= SOF_IPC4_MSG_DIR(SOF_IPC4_MSG_REQUEST);
2312 	header |= SOF_IPC4_MSG_TARGET(SOF_IPC4_MODULE_MSG);
2313 
2314 	extension = SOF_IPC4_MOD_EXT_MSG_SIZE(msg.data_size);
2315 	extension |=
2316 		SOF_IPC4_MOD_EXT_MSG_PARAM_ID(SOF_IPC4_COPIER_MODULE_CFG_PARAM_SET_SINK_FORMAT);
2317 	extension |= SOF_IPC4_MOD_EXT_MSG_LAST_BLOCK(1);
2318 	extension |= SOF_IPC4_MOD_EXT_MSG_FIRST_BLOCK(1);
2319 
2320 	msg.primary = header;
2321 	msg.extension = extension;
2322 
2323 	return sof_ipc_tx_message_no_reply(sdev->ipc, &msg, msg.data_size);
2324 }
2325 
2326 static int sof_ipc4_route_setup(struct snd_sof_dev *sdev, struct snd_sof_route *sroute)
2327 {
2328 	struct snd_sof_widget *src_widget = sroute->src_widget;
2329 	struct snd_sof_widget *sink_widget = sroute->sink_widget;
2330 	struct snd_sof_widget *src_pipe_widget = src_widget->spipe->pipe_widget;
2331 	struct snd_sof_widget *sink_pipe_widget = sink_widget->spipe->pipe_widget;
2332 	struct sof_ipc4_fw_module *src_fw_module = src_widget->module_info;
2333 	struct sof_ipc4_fw_module *sink_fw_module = sink_widget->module_info;
2334 	struct sof_ipc4_pipeline *src_pipeline = src_pipe_widget->private;
2335 	struct sof_ipc4_pipeline *sink_pipeline = sink_pipe_widget->private;
2336 	struct sof_ipc4_msg msg = {{ 0 }};
2337 	u32 header, extension;
2338 	int ret;
2339 
2340 	/* no route set up if chain DMA is used */
2341 	if (src_pipeline->use_chain_dma || sink_pipeline->use_chain_dma) {
2342 		if (!src_pipeline->use_chain_dma || !sink_pipeline->use_chain_dma) {
2343 			dev_err(sdev->dev,
2344 				"use_chain_dma must be set for both src %s and sink %s pipelines\n",
2345 				src_widget->widget->name, sink_widget->widget->name);
2346 			return -EINVAL;
2347 		}
2348 		return 0;
2349 	}
2350 
2351 	if (!src_fw_module || !sink_fw_module) {
2352 		dev_err(sdev->dev,
2353 			"cannot bind %s -> %s, no firmware module for: %s%s\n",
2354 			src_widget->widget->name, sink_widget->widget->name,
2355 			src_fw_module ? "" : " source",
2356 			sink_fw_module ? "" : " sink");
2357 
2358 		return -ENODEV;
2359 	}
2360 
2361 	sroute->src_queue_id = sof_ipc4_get_queue_id(src_widget, sink_widget,
2362 						     SOF_PIN_TYPE_OUTPUT);
2363 	if (sroute->src_queue_id < 0) {
2364 		dev_err(sdev->dev, "failed to get queue ID for source widget: %s\n",
2365 			src_widget->widget->name);
2366 		return sroute->src_queue_id;
2367 	}
2368 
2369 	sroute->dst_queue_id = sof_ipc4_get_queue_id(src_widget, sink_widget,
2370 						     SOF_PIN_TYPE_INPUT);
2371 	if (sroute->dst_queue_id < 0) {
2372 		dev_err(sdev->dev, "failed to get queue ID for sink widget: %s\n",
2373 			sink_widget->widget->name);
2374 		sof_ipc4_put_queue_id(src_widget, sroute->src_queue_id,
2375 				      SOF_PIN_TYPE_OUTPUT);
2376 		return sroute->dst_queue_id;
2377 	}
2378 
2379 	/* Pin 0 format is already set during copier module init */
2380 	if (sroute->src_queue_id > 0 && WIDGET_IS_COPIER(src_widget->id)) {
2381 		ret = sof_ipc4_set_copier_sink_format(sdev, src_widget, sink_widget,
2382 						      sroute->src_queue_id);
2383 		if (ret < 0) {
2384 			dev_err(sdev->dev, "failed to set sink format for %s source queue ID %d\n",
2385 				src_widget->widget->name, sroute->src_queue_id);
2386 			goto out;
2387 		}
2388 	}
2389 
2390 	dev_dbg(sdev->dev, "bind %s:%d -> %s:%d\n",
2391 		src_widget->widget->name, sroute->src_queue_id,
2392 		sink_widget->widget->name, sroute->dst_queue_id);
2393 
2394 	header = src_fw_module->man4_module_entry.id;
2395 	header |= SOF_IPC4_MOD_INSTANCE(src_widget->instance_id);
2396 	header |= SOF_IPC4_MSG_TYPE_SET(SOF_IPC4_MOD_BIND);
2397 	header |= SOF_IPC4_MSG_DIR(SOF_IPC4_MSG_REQUEST);
2398 	header |= SOF_IPC4_MSG_TARGET(SOF_IPC4_MODULE_MSG);
2399 
2400 	extension = sink_fw_module->man4_module_entry.id;
2401 	extension |= SOF_IPC4_MOD_EXT_DST_MOD_INSTANCE(sink_widget->instance_id);
2402 	extension |= SOF_IPC4_MOD_EXT_DST_MOD_QUEUE_ID(sroute->dst_queue_id);
2403 	extension |= SOF_IPC4_MOD_EXT_SRC_MOD_QUEUE_ID(sroute->src_queue_id);
2404 
2405 	msg.primary = header;
2406 	msg.extension = extension;
2407 
2408 	ret = sof_ipc_tx_message_no_reply(sdev->ipc, &msg, 0);
2409 	if (ret < 0) {
2410 		dev_err(sdev->dev, "failed to bind modules %s:%d -> %s:%d\n",
2411 			src_widget->widget->name, sroute->src_queue_id,
2412 			sink_widget->widget->name, sroute->dst_queue_id);
2413 		goto out;
2414 	}
2415 
2416 	return ret;
2417 
2418 out:
2419 	sof_ipc4_put_queue_id(src_widget, sroute->src_queue_id, SOF_PIN_TYPE_OUTPUT);
2420 	sof_ipc4_put_queue_id(sink_widget, sroute->dst_queue_id, SOF_PIN_TYPE_INPUT);
2421 	return ret;
2422 }
2423 
2424 static int sof_ipc4_route_free(struct snd_sof_dev *sdev, struct snd_sof_route *sroute)
2425 {
2426 	struct snd_sof_widget *src_widget = sroute->src_widget;
2427 	struct snd_sof_widget *sink_widget = sroute->sink_widget;
2428 	struct sof_ipc4_fw_module *src_fw_module = src_widget->module_info;
2429 	struct sof_ipc4_fw_module *sink_fw_module = sink_widget->module_info;
2430 	struct sof_ipc4_msg msg = {{ 0 }};
2431 	struct snd_sof_widget *src_pipe_widget = src_widget->spipe->pipe_widget;
2432 	struct snd_sof_widget *sink_pipe_widget = sink_widget->spipe->pipe_widget;
2433 	struct sof_ipc4_pipeline *src_pipeline = src_pipe_widget->private;
2434 	struct sof_ipc4_pipeline *sink_pipeline = sink_pipe_widget->private;
2435 	u32 header, extension;
2436 	int ret = 0;
2437 
2438 	/* no route is set up if chain DMA is used */
2439 	if (src_pipeline->use_chain_dma || sink_pipeline->use_chain_dma)
2440 		return 0;
2441 
2442 	dev_dbg(sdev->dev, "unbind modules %s:%d -> %s:%d\n",
2443 		src_widget->widget->name, sroute->src_queue_id,
2444 		sink_widget->widget->name, sroute->dst_queue_id);
2445 
2446 	/*
2447 	 * routes belonging to the same pipeline will be disconnected by the FW when the pipeline
2448 	 * is freed. So avoid sending this IPC which will be ignored by the FW anyway.
2449 	 */
2450 	if (src_widget->spipe->pipe_widget == sink_widget->spipe->pipe_widget)
2451 		goto out;
2452 
2453 	header = src_fw_module->man4_module_entry.id;
2454 	header |= SOF_IPC4_MOD_INSTANCE(src_widget->instance_id);
2455 	header |= SOF_IPC4_MSG_TYPE_SET(SOF_IPC4_MOD_UNBIND);
2456 	header |= SOF_IPC4_MSG_DIR(SOF_IPC4_MSG_REQUEST);
2457 	header |= SOF_IPC4_MSG_TARGET(SOF_IPC4_MODULE_MSG);
2458 
2459 	extension = sink_fw_module->man4_module_entry.id;
2460 	extension |= SOF_IPC4_MOD_EXT_DST_MOD_INSTANCE(sink_widget->instance_id);
2461 	extension |= SOF_IPC4_MOD_EXT_DST_MOD_QUEUE_ID(sroute->dst_queue_id);
2462 	extension |= SOF_IPC4_MOD_EXT_SRC_MOD_QUEUE_ID(sroute->src_queue_id);
2463 
2464 	msg.primary = header;
2465 	msg.extension = extension;
2466 
2467 	ret = sof_ipc_tx_message_no_reply(sdev->ipc, &msg, 0);
2468 	if (ret < 0)
2469 		dev_err(sdev->dev, "failed to unbind modules %s:%d -> %s:%d\n",
2470 			src_widget->widget->name, sroute->src_queue_id,
2471 			sink_widget->widget->name, sroute->dst_queue_id);
2472 out:
2473 	sof_ipc4_put_queue_id(sink_widget, sroute->dst_queue_id, SOF_PIN_TYPE_INPUT);
2474 	sof_ipc4_put_queue_id(src_widget, sroute->src_queue_id, SOF_PIN_TYPE_OUTPUT);
2475 
2476 	return ret;
2477 }
2478 
2479 static int sof_ipc4_dai_config(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget,
2480 			       unsigned int flags, struct snd_sof_dai_config_data *data)
2481 {
2482 	struct snd_sof_widget *pipe_widget = swidget->spipe->pipe_widget;
2483 	struct sof_ipc4_pipeline *pipeline = pipe_widget->private;
2484 	struct snd_sof_dai *dai = swidget->private;
2485 	struct sof_ipc4_gtw_attributes *gtw_attr;
2486 	struct sof_ipc4_copier_data *copier_data;
2487 	struct sof_ipc4_copier *ipc4_copier;
2488 
2489 	if (!dai || !dai->private) {
2490 		dev_err(sdev->dev, "Invalid DAI or DAI private data for %s\n",
2491 			swidget->widget->name);
2492 		return -EINVAL;
2493 	}
2494 
2495 	ipc4_copier = (struct sof_ipc4_copier *)dai->private;
2496 	copier_data = &ipc4_copier->data;
2497 
2498 	if (!data)
2499 		return 0;
2500 
2501 	switch (ipc4_copier->dai_type) {
2502 	case SOF_DAI_INTEL_HDA:
2503 		if (pipeline->use_chain_dma) {
2504 			pipeline->msg.primary &= ~SOF_IPC4_GLB_CHAIN_DMA_LINK_ID_MASK;
2505 			pipeline->msg.primary |= SOF_IPC4_GLB_CHAIN_DMA_LINK_ID(data->dai_data);
2506 			break;
2507 		}
2508 		gtw_attr = ipc4_copier->gtw_attr;
2509 		gtw_attr->lp_buffer_alloc = pipeline->lp_mode;
2510 		pipeline->skip_during_fe_trigger = true;
2511 		fallthrough;
2512 	case SOF_DAI_INTEL_ALH:
2513 		/*
2514 		 * Do not clear the node ID when this op is invoked with
2515 		 * SOF_DAI_CONFIG_FLAGS_HW_FREE. It is needed to free the group_ida during
2516 		 * unprepare.
2517 		 */
2518 		if (flags & SOF_DAI_CONFIG_FLAGS_HW_PARAMS) {
2519 			copier_data->gtw_cfg.node_id &= ~SOF_IPC4_NODE_INDEX_MASK;
2520 			copier_data->gtw_cfg.node_id |= SOF_IPC4_NODE_INDEX(data->dai_data);
2521 		}
2522 		break;
2523 	case SOF_DAI_INTEL_DMIC:
2524 	case SOF_DAI_INTEL_SSP:
2525 		/* nothing to do for SSP/DMIC */
2526 		break;
2527 	default:
2528 		dev_err(sdev->dev, "%s: unsupported dai type %d\n", __func__,
2529 			ipc4_copier->dai_type);
2530 		return -EINVAL;
2531 	}
2532 
2533 	return 0;
2534 }
2535 
2536 static int sof_ipc4_parse_manifest(struct snd_soc_component *scomp, int index,
2537 				   struct snd_soc_tplg_manifest *man)
2538 {
2539 	struct snd_sof_dev *sdev = snd_soc_component_get_drvdata(scomp);
2540 	struct sof_ipc4_fw_data *ipc4_data = sdev->private;
2541 	struct sof_manifest_tlv *manifest_tlv;
2542 	struct sof_manifest *manifest;
2543 	u32 size = le32_to_cpu(man->priv.size);
2544 	u8 *man_ptr = man->priv.data;
2545 	u32 len_check;
2546 	int i;
2547 
2548 	if (!size || size < SOF_IPC4_TPLG_ABI_SIZE) {
2549 		dev_err(scomp->dev, "%s: Invalid topology ABI size: %u\n",
2550 			__func__, size);
2551 		return -EINVAL;
2552 	}
2553 
2554 	manifest = (struct sof_manifest *)man_ptr;
2555 
2556 	dev_info(scomp->dev,
2557 		 "Topology: ABI %d:%d:%d Kernel ABI %u:%u:%u\n",
2558 		  le16_to_cpu(manifest->abi_major), le16_to_cpu(manifest->abi_minor),
2559 		  le16_to_cpu(manifest->abi_patch),
2560 		  SOF_ABI_MAJOR, SOF_ABI_MINOR, SOF_ABI_PATCH);
2561 
2562 	/* TODO: Add ABI compatibility check */
2563 
2564 	/* no more data after the ABI version */
2565 	if (size <= SOF_IPC4_TPLG_ABI_SIZE)
2566 		return 0;
2567 
2568 	manifest_tlv = manifest->items;
2569 	len_check = sizeof(struct sof_manifest);
2570 	for (i = 0; i < le16_to_cpu(manifest->count); i++) {
2571 		len_check += sizeof(struct sof_manifest_tlv) + le32_to_cpu(manifest_tlv->size);
2572 		if (len_check > size)
2573 			return -EINVAL;
2574 
2575 		switch (le32_to_cpu(manifest_tlv->type)) {
2576 		case SOF_MANIFEST_DATA_TYPE_NHLT:
2577 			/* no NHLT in BIOS, so use the one from topology manifest */
2578 			if (ipc4_data->nhlt)
2579 				break;
2580 			ipc4_data->nhlt = devm_kmemdup(sdev->dev, manifest_tlv->data,
2581 						       le32_to_cpu(manifest_tlv->size), GFP_KERNEL);
2582 			if (!ipc4_data->nhlt)
2583 				return -ENOMEM;
2584 			break;
2585 		default:
2586 			dev_warn(scomp->dev, "Skipping unknown manifest data type %d\n",
2587 				 manifest_tlv->type);
2588 			break;
2589 		}
2590 		man_ptr += sizeof(struct sof_manifest_tlv) + le32_to_cpu(manifest_tlv->size);
2591 		manifest_tlv = (struct sof_manifest_tlv *)man_ptr;
2592 	}
2593 
2594 	return 0;
2595 }
2596 
2597 static int sof_ipc4_dai_get_clk(struct snd_sof_dev *sdev, struct snd_sof_dai *dai, int clk_type)
2598 {
2599 	struct sof_ipc4_copier *ipc4_copier = dai->private;
2600 	struct snd_soc_tplg_hw_config *hw_config;
2601 	struct snd_sof_dai_link *slink;
2602 	bool dai_link_found = false;
2603 	bool hw_cfg_found = false;
2604 	int i;
2605 
2606 	if (!ipc4_copier)
2607 		return 0;
2608 
2609 	list_for_each_entry(slink, &sdev->dai_link_list, list) {
2610 		if (!strcmp(slink->link->name, dai->name)) {
2611 			dai_link_found = true;
2612 			break;
2613 		}
2614 	}
2615 
2616 	if (!dai_link_found) {
2617 		dev_err(sdev->dev, "no DAI link found for DAI %s\n", dai->name);
2618 		return -EINVAL;
2619 	}
2620 
2621 	for (i = 0; i < slink->num_hw_configs; i++) {
2622 		hw_config = &slink->hw_configs[i];
2623 		if (dai->current_config == le32_to_cpu(hw_config->id)) {
2624 			hw_cfg_found = true;
2625 			break;
2626 		}
2627 	}
2628 
2629 	if (!hw_cfg_found) {
2630 		dev_err(sdev->dev, "no matching hw_config found for DAI %s\n", dai->name);
2631 		return -EINVAL;
2632 	}
2633 
2634 	switch (ipc4_copier->dai_type) {
2635 	case SOF_DAI_INTEL_SSP:
2636 		switch (clk_type) {
2637 		case SOF_DAI_CLK_INTEL_SSP_MCLK:
2638 			return le32_to_cpu(hw_config->mclk_rate);
2639 		case SOF_DAI_CLK_INTEL_SSP_BCLK:
2640 			return le32_to_cpu(hw_config->bclk_rate);
2641 		default:
2642 			dev_err(sdev->dev, "Invalid clk type for SSP %d\n", clk_type);
2643 			break;
2644 		}
2645 		break;
2646 	default:
2647 		dev_err(sdev->dev, "DAI type %d not supported yet!\n", ipc4_copier->dai_type);
2648 		break;
2649 	}
2650 
2651 	return -EINVAL;
2652 }
2653 
2654 static int sof_ipc4_tear_down_all_pipelines(struct snd_sof_dev *sdev, bool verify)
2655 {
2656 	struct snd_sof_pcm *spcm;
2657 	int dir, ret;
2658 
2659 	/*
2660 	 * This function is called during system suspend, we need to make sure
2661 	 * that all streams have been freed up.
2662 	 * Freeing might have been skipped when xrun happened just at the start
2663 	 * of the suspend and it sent a SNDRV_PCM_TRIGGER_STOP to the active
2664 	 * stream. This will call sof_pcm_stream_free() with
2665 	 * free_widget_list = false which will leave the kernel and firmware out
2666 	 * of sync during suspend/resume.
2667 	 *
2668 	 * This will also make sure that paused streams handled correctly.
2669 	 */
2670 	list_for_each_entry(spcm, &sdev->pcm_list, list) {
2671 		for_each_pcm_streams(dir) {
2672 			struct snd_pcm_substream *substream = spcm->stream[dir].substream;
2673 
2674 			if (!substream || !substream->runtime || spcm->stream[dir].suspend_ignored)
2675 				continue;
2676 
2677 			if (spcm->stream[dir].list) {
2678 				ret = sof_pcm_stream_free(sdev, substream, spcm, dir, true);
2679 				if (ret < 0)
2680 					return ret;
2681 			}
2682 		}
2683 	}
2684 	return 0;
2685 }
2686 
2687 static int sof_ipc4_link_setup(struct snd_sof_dev *sdev, struct snd_soc_dai_link *link)
2688 {
2689 	if (link->no_pcm)
2690 		return 0;
2691 
2692 	/*
2693 	 * set default trigger order for all links. Exceptions to
2694 	 * the rule will be handled in sof_pcm_dai_link_fixup()
2695 	 * For playback, the sequence is the following: start BE,
2696 	 * start FE, stop FE, stop BE; for Capture the sequence is
2697 	 * inverted start FE, start BE, stop BE, stop FE
2698 	 */
2699 	link->trigger[SNDRV_PCM_STREAM_PLAYBACK] = SND_SOC_DPCM_TRIGGER_POST;
2700 	link->trigger[SNDRV_PCM_STREAM_CAPTURE] = SND_SOC_DPCM_TRIGGER_PRE;
2701 
2702 	return 0;
2703 }
2704 
2705 static enum sof_tokens common_copier_token_list[] = {
2706 	SOF_COMP_TOKENS,
2707 	SOF_AUDIO_FMT_NUM_TOKENS,
2708 	SOF_IN_AUDIO_FORMAT_TOKENS,
2709 	SOF_OUT_AUDIO_FORMAT_TOKENS,
2710 	SOF_COPIER_DEEP_BUFFER_TOKENS,
2711 	SOF_COPIER_TOKENS,
2712 	SOF_COMP_EXT_TOKENS,
2713 };
2714 
2715 static enum sof_tokens pipeline_token_list[] = {
2716 	SOF_SCHED_TOKENS,
2717 	SOF_PIPELINE_TOKENS,
2718 };
2719 
2720 static enum sof_tokens dai_token_list[] = {
2721 	SOF_COMP_TOKENS,
2722 	SOF_AUDIO_FMT_NUM_TOKENS,
2723 	SOF_IN_AUDIO_FORMAT_TOKENS,
2724 	SOF_OUT_AUDIO_FORMAT_TOKENS,
2725 	SOF_COPIER_TOKENS,
2726 	SOF_DAI_TOKENS,
2727 	SOF_COMP_EXT_TOKENS,
2728 };
2729 
2730 static enum sof_tokens pga_token_list[] = {
2731 	SOF_COMP_TOKENS,
2732 	SOF_GAIN_TOKENS,
2733 	SOF_AUDIO_FMT_NUM_TOKENS,
2734 	SOF_IN_AUDIO_FORMAT_TOKENS,
2735 	SOF_OUT_AUDIO_FORMAT_TOKENS,
2736 	SOF_COMP_EXT_TOKENS,
2737 };
2738 
2739 static enum sof_tokens mixer_token_list[] = {
2740 	SOF_COMP_TOKENS,
2741 	SOF_AUDIO_FMT_NUM_TOKENS,
2742 	SOF_IN_AUDIO_FORMAT_TOKENS,
2743 	SOF_OUT_AUDIO_FORMAT_TOKENS,
2744 	SOF_COMP_EXT_TOKENS,
2745 };
2746 
2747 static enum sof_tokens src_token_list[] = {
2748 	SOF_COMP_TOKENS,
2749 	SOF_SRC_TOKENS,
2750 	SOF_AUDIO_FMT_NUM_TOKENS,
2751 	SOF_IN_AUDIO_FORMAT_TOKENS,
2752 	SOF_OUT_AUDIO_FORMAT_TOKENS,
2753 	SOF_COMP_EXT_TOKENS,
2754 };
2755 
2756 static enum sof_tokens process_token_list[] = {
2757 	SOF_COMP_TOKENS,
2758 	SOF_AUDIO_FMT_NUM_TOKENS,
2759 	SOF_IN_AUDIO_FORMAT_TOKENS,
2760 	SOF_OUT_AUDIO_FORMAT_TOKENS,
2761 	SOF_COMP_EXT_TOKENS,
2762 };
2763 
2764 static const struct sof_ipc_tplg_widget_ops tplg_ipc4_widget_ops[SND_SOC_DAPM_TYPE_COUNT] = {
2765 	[snd_soc_dapm_aif_in] =  {sof_ipc4_widget_setup_pcm, sof_ipc4_widget_free_comp_pcm,
2766 				  common_copier_token_list, ARRAY_SIZE(common_copier_token_list),
2767 				  NULL, sof_ipc4_prepare_copier_module,
2768 				  sof_ipc4_unprepare_copier_module},
2769 	[snd_soc_dapm_aif_out] = {sof_ipc4_widget_setup_pcm, sof_ipc4_widget_free_comp_pcm,
2770 				  common_copier_token_list, ARRAY_SIZE(common_copier_token_list),
2771 				  NULL, sof_ipc4_prepare_copier_module,
2772 				  sof_ipc4_unprepare_copier_module},
2773 	[snd_soc_dapm_dai_in] = {sof_ipc4_widget_setup_comp_dai, sof_ipc4_widget_free_comp_dai,
2774 				 dai_token_list, ARRAY_SIZE(dai_token_list), NULL,
2775 				 sof_ipc4_prepare_copier_module,
2776 				 sof_ipc4_unprepare_copier_module},
2777 	[snd_soc_dapm_dai_out] = {sof_ipc4_widget_setup_comp_dai, sof_ipc4_widget_free_comp_dai,
2778 				  dai_token_list, ARRAY_SIZE(dai_token_list), NULL,
2779 				  sof_ipc4_prepare_copier_module,
2780 				  sof_ipc4_unprepare_copier_module},
2781 	[snd_soc_dapm_buffer] = {sof_ipc4_widget_setup_pcm, sof_ipc4_widget_free_comp_pcm,
2782 				 common_copier_token_list, ARRAY_SIZE(common_copier_token_list),
2783 				 NULL, sof_ipc4_prepare_copier_module,
2784 				 sof_ipc4_unprepare_copier_module},
2785 	[snd_soc_dapm_scheduler] = {sof_ipc4_widget_setup_comp_pipeline,
2786 				    sof_ipc4_widget_free_comp_pipeline,
2787 				    pipeline_token_list, ARRAY_SIZE(pipeline_token_list), NULL,
2788 				    NULL, NULL},
2789 	[snd_soc_dapm_pga] = {sof_ipc4_widget_setup_comp_pga, sof_ipc4_widget_free_comp_pga,
2790 			      pga_token_list, ARRAY_SIZE(pga_token_list), NULL,
2791 			      sof_ipc4_prepare_gain_module,
2792 			      NULL},
2793 	[snd_soc_dapm_mixer] = {sof_ipc4_widget_setup_comp_mixer, sof_ipc4_widget_free_comp_mixer,
2794 				mixer_token_list, ARRAY_SIZE(mixer_token_list),
2795 				NULL, sof_ipc4_prepare_mixer_module,
2796 				NULL},
2797 	[snd_soc_dapm_src] = {sof_ipc4_widget_setup_comp_src, sof_ipc4_widget_free_comp_src,
2798 				src_token_list, ARRAY_SIZE(src_token_list),
2799 				NULL, sof_ipc4_prepare_src_module,
2800 				NULL},
2801 	[snd_soc_dapm_effect] = {sof_ipc4_widget_setup_comp_process,
2802 				sof_ipc4_widget_free_comp_process,
2803 				process_token_list, ARRAY_SIZE(process_token_list),
2804 				NULL, sof_ipc4_prepare_process_module,
2805 				NULL},
2806 };
2807 
2808 const struct sof_ipc_tplg_ops ipc4_tplg_ops = {
2809 	.widget = tplg_ipc4_widget_ops,
2810 	.token_list = ipc4_token_list,
2811 	.control_setup = sof_ipc4_control_setup,
2812 	.control = &tplg_ipc4_control_ops,
2813 	.widget_setup = sof_ipc4_widget_setup,
2814 	.widget_free = sof_ipc4_widget_free,
2815 	.route_setup = sof_ipc4_route_setup,
2816 	.route_free = sof_ipc4_route_free,
2817 	.dai_config = sof_ipc4_dai_config,
2818 	.parse_manifest = sof_ipc4_parse_manifest,
2819 	.dai_get_clk = sof_ipc4_dai_get_clk,
2820 	.tear_down_all_pipelines = sof_ipc4_tear_down_all_pipelines,
2821 	.link_setup = sof_ipc4_link_setup,
2822 };
2823