Linux Audio

Check our new training course

Loading...
v6.8
  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) 2018 Intel Corporation. All rights reserved.
  7//
  8// Author: Liam Girdwood <liam.r.girdwood@linux.intel.com>
  9//
 10
 11#include "ops.h"
 12#include "sof-priv.h"
 13#include "sof-audio.h"
 14
 15/*
 16 * Helper function to determine the target DSP state during
 17 * system suspend. This function only cares about the device
 18 * D-states. Platform-specific substates, if any, should be
 19 * handled by the platform-specific parts.
 20 */
 21static u32 snd_sof_dsp_power_target(struct snd_sof_dev *sdev)
 22{
 23	u32 target_dsp_state;
 24
 25	switch (sdev->system_suspend_target) {
 26	case SOF_SUSPEND_S5:
 27	case SOF_SUSPEND_S4:
 28		/* DSP should be in D3 if the system is suspending to S3+ */
 29	case SOF_SUSPEND_S3:
 30		/* DSP should be in D3 if the system is suspending to S3 */
 31		target_dsp_state = SOF_DSP_PM_D3;
 32		break;
 33	case SOF_SUSPEND_S0IX:
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 34		/*
 35		 * Currently, the only criterion for retaining the DSP in D0
 36		 * is that there are streams that ignored the suspend trigger.
 37		 * Additional criteria such Soundwire clock-stop mode and
 38		 * device suspend latency considerations will be added later.
 39		 */
 40		if (snd_sof_stream_suspend_ignored(sdev))
 41			target_dsp_state = SOF_DSP_PM_D0;
 42		else
 43			target_dsp_state = SOF_DSP_PM_D3;
 44		break;
 45	default:
 46		/* This case would be during runtime suspend */
 47		target_dsp_state = SOF_DSP_PM_D3;
 48		break;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 49	}
 50
 51	return target_dsp_state;
 
 52}
 53
 54#if IS_ENABLED(CONFIG_SND_SOC_SOF_DEBUG_ENABLE_DEBUGFS_CACHE)
 55static void sof_cache_debugfs(struct snd_sof_dev *sdev)
 56{
 57	struct snd_sof_dfsentry *dfse;
 58
 59	list_for_each_entry(dfse, &sdev->dfsentry_list, list) {
 60
 61		/* nothing to do if debugfs buffer is not IO mem */
 62		if (dfse->type == SOF_DFSENTRY_TYPE_BUF)
 63			continue;
 64
 65		/* cache memory that is only accessible in D0 */
 66		if (dfse->access_type == SOF_DEBUGFS_ACCESS_D0_ONLY)
 67			memcpy_fromio(dfse->cache_buf, dfse->io_mem,
 68				      dfse->size);
 69	}
 70}
 71#endif
 72
 73static int sof_resume(struct device *dev, bool runtime_resume)
 74{
 75	struct snd_sof_dev *sdev = dev_get_drvdata(dev);
 76	const struct sof_ipc_pm_ops *pm_ops = sof_ipc_get_ops(sdev, pm);
 77	const struct sof_ipc_tplg_ops *tplg_ops = sof_ipc_get_ops(sdev, tplg);
 78	u32 old_state = sdev->dsp_power_state.state;
 79	int ret;
 80
 81	/* do nothing if dsp resume callbacks are not set */
 82	if (!runtime_resume && !sof_ops(sdev)->resume)
 83		return 0;
 84
 85	if (runtime_resume && !sof_ops(sdev)->runtime_resume)
 86		return 0;
 87
 88	/* DSP was never successfully started, nothing to resume */
 89	if (sdev->first_boot)
 90		return 0;
 91
 92	/*
 93	 * if the runtime_resume flag is set, call the runtime_resume routine
 94	 * or else call the system resume routine
 95	 */
 96	if (runtime_resume)
 97		ret = snd_sof_dsp_runtime_resume(sdev);
 98	else
 99		ret = snd_sof_dsp_resume(sdev);
100	if (ret < 0) {
101		dev_err(sdev->dev,
102			"error: failed to power up DSP after resume\n");
103		return ret;
104	}
105
106	if (sdev->dspless_mode_selected) {
107		sof_set_fw_state(sdev, SOF_DSPLESS_MODE);
108		return 0;
109	}
110
111	/*
112	 * Nothing further to be done for platforms that support the low power
113	 * D0 substate. Resume trace and return when resuming from
114	 * low-power D0 substate
115	 */
116	if (!runtime_resume && sof_ops(sdev)->set_power_state &&
117	    old_state == SOF_DSP_PM_D0) {
118		ret = sof_fw_trace_resume(sdev);
119		if (ret < 0)
120			/* non fatal */
121			dev_warn(sdev->dev,
122				 "failed to enable trace after resume %d\n", ret);
123		return 0;
124	}
125
126	sof_set_fw_state(sdev, SOF_FW_BOOT_PREPARE);
127
128	/* load the firmware */
129	ret = snd_sof_load_firmware(sdev);
130	if (ret < 0) {
131		dev_err(sdev->dev,
132			"error: failed to load DSP firmware after resume %d\n",
133			ret);
134		sof_set_fw_state(sdev, SOF_FW_BOOT_FAILED);
135		return ret;
136	}
137
138	sof_set_fw_state(sdev, SOF_FW_BOOT_IN_PROGRESS);
139
140	/*
141	 * Boot the firmware. The FW boot status will be modified
142	 * in snd_sof_run_firmware() depending on the outcome.
143	 */
144	ret = snd_sof_run_firmware(sdev);
145	if (ret < 0) {
146		dev_err(sdev->dev,
147			"error: failed to boot DSP firmware after resume %d\n",
148			ret);
149		sof_set_fw_state(sdev, SOF_FW_BOOT_FAILED);
150		return ret;
151	}
152
153	/* resume DMA trace */
154	ret = sof_fw_trace_resume(sdev);
155	if (ret < 0) {
156		/* non fatal */
157		dev_warn(sdev->dev,
158			 "warning: failed to init trace after resume %d\n",
159			 ret);
160	}
161
162	/* restore pipelines */
163	if (tplg_ops && tplg_ops->set_up_all_pipelines) {
164		ret = tplg_ops->set_up_all_pipelines(sdev, false);
165		if (ret < 0) {
166			dev_err(sdev->dev, "Failed to restore pipeline after resume %d\n", ret);
167			goto setup_fail;
168		}
169	}
170
171	/* Notify clients not managed by pm framework about core resume */
172	sof_resume_clients(sdev);
173
174	/* notify DSP of system resume */
175	if (pm_ops && pm_ops->ctx_restore) {
176		ret = pm_ops->ctx_restore(sdev);
177		if (ret < 0)
178			dev_err(sdev->dev, "ctx_restore IPC error during resume: %d\n", ret);
179	}
180
181setup_fail:
182#if IS_ENABLED(CONFIG_SND_SOC_SOF_DEBUG_ENABLE_DEBUGFS_CACHE)
183	if (ret < 0) {
184		/*
185		 * Debugfs cannot be read in runtime suspend, so cache
186		 * the contents upon failure. This allows to capture
187		 * possible DSP coredump information.
188		 */
189		sof_cache_debugfs(sdev);
190	}
191#endif
192
193	return ret;
194}
195
196static int sof_suspend(struct device *dev, bool runtime_suspend)
197{
198	struct snd_sof_dev *sdev = dev_get_drvdata(dev);
199	const struct sof_ipc_pm_ops *pm_ops = sof_ipc_get_ops(sdev, pm);
200	const struct sof_ipc_tplg_ops *tplg_ops = sof_ipc_get_ops(sdev, tplg);
201	pm_message_t pm_state;
202	u32 target_state = snd_sof_dsp_power_target(sdev);
203	u32 old_state = sdev->dsp_power_state.state;
204	int ret;
205
206	/* do nothing if dsp suspend callback is not set */
207	if (!runtime_suspend && !sof_ops(sdev)->suspend)
208		return 0;
209
210	if (runtime_suspend && !sof_ops(sdev)->runtime_suspend)
211		return 0;
212
213	/* we need to tear down pipelines only if the DSP hardware is
214	 * active, which happens for PCI devices. if the device is
215	 * suspended, it is brought back to full power and then
216	 * suspended again
217	 */
218	if (tplg_ops && tplg_ops->tear_down_all_pipelines && (old_state == SOF_DSP_PM_D0))
219		tplg_ops->tear_down_all_pipelines(sdev, false);
220
221	if (sdev->fw_state != SOF_FW_BOOT_COMPLETE)
222		goto suspend;
223
224	/* prepare for streams to be resumed properly upon resume */
225	if (!runtime_suspend) {
226		ret = snd_sof_dsp_hw_params_upon_resume(sdev);
227		if (ret < 0) {
228			dev_err(sdev->dev,
229				"error: setting hw_params flag during suspend %d\n",
230				ret);
231			return ret;
232		}
233	}
234
235	pm_state.event = target_state;
236
237	/* suspend DMA trace */
238	sof_fw_trace_suspend(sdev, pm_state);
239
240	/* Notify clients not managed by pm framework about core suspend */
241	sof_suspend_clients(sdev, pm_state);
242
243	/* Skip to platform-specific suspend if DSP is entering D0 */
244	if (target_state == SOF_DSP_PM_D0)
245		goto suspend;
246
247#if IS_ENABLED(CONFIG_SND_SOC_SOF_DEBUG_ENABLE_DEBUGFS_CACHE)
248	/* cache debugfs contents during runtime suspend */
249	if (runtime_suspend)
250		sof_cache_debugfs(sdev);
251#endif
252	/* notify DSP of upcoming power down */
253	if (pm_ops && pm_ops->ctx_save) {
254		ret = pm_ops->ctx_save(sdev);
255		if (ret == -EBUSY || ret == -EAGAIN) {
256			/*
257			 * runtime PM has logic to handle -EBUSY/-EAGAIN so
258			 * pass these errors up
259			 */
260			dev_err(sdev->dev, "ctx_save IPC error during suspend: %d\n", ret);
261			return ret;
262		} else if (ret < 0) {
263			/* FW in unexpected state, continue to power down */
264			dev_warn(sdev->dev, "ctx_save IPC error: %d, proceeding with suspend\n",
265				 ret);
266		}
 
267	}
268
269suspend:
270
271	/* return if the DSP was not probed successfully */
272	if (sdev->fw_state == SOF_FW_BOOT_NOT_STARTED)
273		return 0;
274
275	/* platform-specific suspend */
276	if (runtime_suspend)
277		ret = snd_sof_dsp_runtime_suspend(sdev);
278	else
279		ret = snd_sof_dsp_suspend(sdev, target_state);
280	if (ret < 0)
281		dev_err(sdev->dev,
282			"error: failed to power down DSP during suspend %d\n",
283			ret);
284
285	/* Do not reset FW state if DSP is in D0 */
286	if (target_state == SOF_DSP_PM_D0)
287		return ret;
288
289	/* reset FW state */
290	sof_set_fw_state(sdev, SOF_FW_BOOT_NOT_STARTED);
291	sdev->enabled_cores_mask = 0;
292
293	return ret;
294}
295
296int snd_sof_dsp_power_down_notify(struct snd_sof_dev *sdev)
297{
298	const struct sof_ipc_pm_ops *pm_ops = sof_ipc_get_ops(sdev, pm);
299
300	/* Notify DSP of upcoming power down */
301	if (sof_ops(sdev)->remove && pm_ops && pm_ops->ctx_save)
302		return pm_ops->ctx_save(sdev);
303
304	return 0;
305}
306
307int snd_sof_runtime_suspend(struct device *dev)
308{
309	return sof_suspend(dev, true);
310}
311EXPORT_SYMBOL(snd_sof_runtime_suspend);
312
313int snd_sof_runtime_idle(struct device *dev)
314{
315	struct snd_sof_dev *sdev = dev_get_drvdata(dev);
316
317	return snd_sof_dsp_runtime_idle(sdev);
318}
319EXPORT_SYMBOL(snd_sof_runtime_idle);
320
321int snd_sof_runtime_resume(struct device *dev)
322{
323	return sof_resume(dev, true);
324}
325EXPORT_SYMBOL(snd_sof_runtime_resume);
326
327int snd_sof_resume(struct device *dev)
328{
329	return sof_resume(dev, false);
330}
331EXPORT_SYMBOL(snd_sof_resume);
332
333int snd_sof_suspend(struct device *dev)
334{
335	return sof_suspend(dev, false);
336}
337EXPORT_SYMBOL(snd_sof_suspend);
338
339int snd_sof_prepare(struct device *dev)
340{
341	struct snd_sof_dev *sdev = dev_get_drvdata(dev);
342	const struct sof_dev_desc *desc = sdev->pdata->desc;
343
344	/* will suspend to S3 by default */
345	sdev->system_suspend_target = SOF_SUSPEND_S3;
346
347	/*
348	 * if the firmware is crashed or boot failed then we try to aim for S3
349	 * to reboot the firmware
350	 */
351	if (sdev->fw_state == SOF_FW_CRASHED ||
352	    sdev->fw_state == SOF_FW_BOOT_FAILED)
353		return 0;
354
355	if (!desc->use_acpi_target_states)
356		return 0;
357
358#if defined(CONFIG_ACPI)
359	switch (acpi_target_system_state()) {
360	case ACPI_STATE_S0:
361		sdev->system_suspend_target = SOF_SUSPEND_S0IX;
362		break;
363	case ACPI_STATE_S1:
364	case ACPI_STATE_S2:
365	case ACPI_STATE_S3:
366		sdev->system_suspend_target = SOF_SUSPEND_S3;
367		break;
368	case ACPI_STATE_S4:
369		sdev->system_suspend_target = SOF_SUSPEND_S4;
370		break;
371	case ACPI_STATE_S5:
372		sdev->system_suspend_target = SOF_SUSPEND_S5;
373		break;
374	default:
375		break;
376	}
377#endif
378
379	return 0;
380}
381EXPORT_SYMBOL(snd_sof_prepare);
382
383void snd_sof_complete(struct device *dev)
384{
385	struct snd_sof_dev *sdev = dev_get_drvdata(dev);
386
387	sdev->system_suspend_target = SOF_SUSPEND_NONE;
388}
389EXPORT_SYMBOL(snd_sof_complete);
v5.4
  1// SPDX-License-Identifier: (GPL-2.0 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) 2018 Intel Corporation. All rights reserved.
  7//
  8// Author: Liam Girdwood <liam.r.girdwood@linux.intel.com>
  9//
 10
 11#include "ops.h"
 12#include "sof-priv.h"
 
 13
 14static int sof_restore_kcontrols(struct snd_sof_dev *sdev)
 15{
 16	struct snd_sof_control *scontrol;
 17	int ipc_cmd, ctrl_type;
 18	int ret = 0;
 19
 20	/* restore kcontrol values */
 21	list_for_each_entry(scontrol, &sdev->kcontrol_list, list) {
 22		/* reset readback offset for scontrol after resuming */
 23		scontrol->readback_offset = 0;
 24
 25		/* notify DSP of kcontrol values */
 26		switch (scontrol->cmd) {
 27		case SOF_CTRL_CMD_VOLUME:
 28		case SOF_CTRL_CMD_ENUM:
 29		case SOF_CTRL_CMD_SWITCH:
 30			ipc_cmd = SOF_IPC_COMP_SET_VALUE;
 31			ctrl_type = SOF_CTRL_TYPE_VALUE_CHAN_SET;
 32			ret = snd_sof_ipc_set_get_comp_data(sdev->ipc, scontrol,
 33							    ipc_cmd, ctrl_type,
 34							    scontrol->cmd,
 35							    true);
 36			break;
 37		case SOF_CTRL_CMD_BINARY:
 38			ipc_cmd = SOF_IPC_COMP_SET_DATA;
 39			ctrl_type = SOF_CTRL_TYPE_DATA_SET;
 40			ret = snd_sof_ipc_set_get_comp_data(sdev->ipc, scontrol,
 41							    ipc_cmd, ctrl_type,
 42							    scontrol->cmd,
 43							    true);
 44			break;
 45
 46		default:
 47			break;
 48		}
 49
 50		if (ret < 0) {
 51			dev_err(sdev->dev,
 52				"error: failed kcontrol value set for widget: %d\n",
 53				scontrol->comp_id);
 54
 55			return ret;
 56		}
 57	}
 58
 59	return 0;
 60}
 61
 62static int sof_restore_pipelines(struct snd_sof_dev *sdev)
 63{
 64	struct snd_sof_widget *swidget;
 65	struct snd_sof_route *sroute;
 66	struct sof_ipc_pipe_new *pipeline;
 67	struct snd_sof_dai *dai;
 68	struct sof_ipc_comp_dai *comp_dai;
 69	struct sof_ipc_cmd_hdr *hdr;
 70	int ret;
 71
 72	/* restore pipeline components */
 73	list_for_each_entry_reverse(swidget, &sdev->widget_list, list) {
 74		struct sof_ipc_comp_reply r;
 75
 76		/* skip if there is no private data */
 77		if (!swidget->private)
 78			continue;
 79
 80		switch (swidget->id) {
 81		case snd_soc_dapm_dai_in:
 82		case snd_soc_dapm_dai_out:
 83			dai = swidget->private;
 84			comp_dai = &dai->comp_dai;
 85			ret = sof_ipc_tx_message(sdev->ipc,
 86						 comp_dai->comp.hdr.cmd,
 87						 comp_dai, sizeof(*comp_dai),
 88						 &r, sizeof(r));
 89			break;
 90		case snd_soc_dapm_scheduler:
 91
 92			/*
 93			 * During suspend, all DSP cores are powered off.
 94			 * Therefore upon resume, create the pipeline comp
 95			 * and power up the core that the pipeline is
 96			 * scheduled on.
 97			 */
 98			pipeline = swidget->private;
 99			ret = sof_load_pipeline_ipc(sdev, pipeline, &r);
100			break;
101		default:
102			hdr = swidget->private;
103			ret = sof_ipc_tx_message(sdev->ipc, hdr->cmd,
104						 swidget->private, hdr->size,
105						 &r, sizeof(r));
106			break;
107		}
108		if (ret < 0) {
109			dev_err(sdev->dev,
110				"error: failed to load widget type %d with ID: %d\n",
111				swidget->widget->id, swidget->comp_id);
112
113			return ret;
114		}
115	}
116
117	/* restore pipeline connections */
118	list_for_each_entry_reverse(sroute, &sdev->route_list, list) {
119		struct sof_ipc_pipe_comp_connect *connect;
120		struct sof_ipc_reply reply;
121
122		/* skip if there's no private data */
123		if (!sroute->private)
124			continue;
125
126		connect = sroute->private;
127
128		/* send ipc */
129		ret = sof_ipc_tx_message(sdev->ipc,
130					 connect->hdr.cmd,
131					 connect, sizeof(*connect),
132					 &reply, sizeof(reply));
133		if (ret < 0) {
134			dev_err(sdev->dev,
135				"error: failed to load route sink %s control %s source %s\n",
136				sroute->route->sink,
137				sroute->route->control ? sroute->route->control
138					: "none",
139				sroute->route->source);
140
141			return ret;
142		}
143	}
144
145	/* restore dai links */
146	list_for_each_entry_reverse(dai, &sdev->dai_list, list) {
147		struct sof_ipc_reply reply;
148		struct sof_ipc_dai_config *config = dai->dai_config;
149
150		if (!config) {
151			dev_err(sdev->dev, "error: no config for DAI %s\n",
152				dai->name);
153			continue;
154		}
155
156		/*
157		 * The link DMA channel would be invalidated for running
158		 * streams but not for streams that were in the PAUSED
159		 * state during suspend. So invalidate it here before setting
160		 * the dai config in the DSP.
161		 */
162		if (config->type == SOF_DAI_INTEL_HDA)
163			config->hda.link_dma_ch = DMA_CHAN_INVALID;
164
165		ret = sof_ipc_tx_message(sdev->ipc,
166					 config->hdr.cmd, config,
167					 config->hdr.size,
168					 &reply, sizeof(reply));
169
170		if (ret < 0) {
171			dev_err(sdev->dev,
172				"error: failed to set dai config for %s\n",
173				dai->name);
174
175			return ret;
176		}
177	}
178
179	/* complete pipeline */
180	list_for_each_entry(swidget, &sdev->widget_list, list) {
181		switch (swidget->id) {
182		case snd_soc_dapm_scheduler:
183			swidget->complete =
184				snd_sof_complete_pipeline(sdev, swidget);
185			break;
186		default:
187			break;
188		}
189	}
190
191	/* restore pipeline kcontrols */
192	ret = sof_restore_kcontrols(sdev);
193	if (ret < 0)
194		dev_err(sdev->dev,
195			"error: restoring kcontrols after resume\n");
196
197	return ret;
198}
199
200static int sof_send_pm_ipc(struct snd_sof_dev *sdev, int cmd)
201{
202	struct sof_ipc_pm_ctx pm_ctx;
203	struct sof_ipc_reply reply;
204
205	memset(&pm_ctx, 0, sizeof(pm_ctx));
206
207	/* configure ctx save ipc message */
208	pm_ctx.hdr.size = sizeof(pm_ctx);
209	pm_ctx.hdr.cmd = SOF_IPC_GLB_PM_MSG | cmd;
210
211	/* send ctx save ipc to dsp */
212	return sof_ipc_tx_message(sdev->ipc, pm_ctx.hdr.cmd, &pm_ctx,
213				 sizeof(pm_ctx), &reply, sizeof(reply));
214}
215
216static int sof_set_hw_params_upon_resume(struct snd_sof_dev *sdev)
217{
218	struct snd_pcm_substream *substream;
219	struct snd_sof_pcm *spcm;
220	snd_pcm_state_t state;
221	int dir;
222
223	/*
224	 * SOF requires hw_params to be set-up internally upon resume.
225	 * So, set the flag to indicate this for those streams that
226	 * have been suspended.
227	 */
228	list_for_each_entry(spcm, &sdev->pcm_list, list) {
229		for (dir = 0; dir <= SNDRV_PCM_STREAM_CAPTURE; dir++) {
230			substream = spcm->stream[dir].substream;
231			if (!substream || !substream->runtime)
232				continue;
233
234			state = substream->runtime->status->state;
235			if (state == SNDRV_PCM_STATE_SUSPENDED)
236				spcm->prepared[dir] = false;
237		}
238	}
239
240	/* set internal flag for BE */
241	return snd_sof_dsp_hw_params_upon_resume(sdev);
242}
243
244#if IS_ENABLED(CONFIG_SND_SOC_SOF_DEBUG_ENABLE_DEBUGFS_CACHE)
245static void sof_cache_debugfs(struct snd_sof_dev *sdev)
246{
247	struct snd_sof_dfsentry *dfse;
248
249	list_for_each_entry(dfse, &sdev->dfsentry_list, list) {
250
251		/* nothing to do if debugfs buffer is not IO mem */
252		if (dfse->type == SOF_DFSENTRY_TYPE_BUF)
253			continue;
254
255		/* cache memory that is only accessible in D0 */
256		if (dfse->access_type == SOF_DEBUGFS_ACCESS_D0_ONLY)
257			memcpy_fromio(dfse->cache_buf, dfse->io_mem,
258				      dfse->size);
259	}
260}
261#endif
262
263static int sof_resume(struct device *dev, bool runtime_resume)
264{
265	struct snd_sof_dev *sdev = dev_get_drvdata(dev);
 
 
 
266	int ret;
267
268	/* do nothing if dsp resume callbacks are not set */
269	if (!sof_ops(sdev)->resume || !sof_ops(sdev)->runtime_resume)
 
 
 
 
 
 
 
270		return 0;
271
272	/*
273	 * if the runtime_resume flag is set, call the runtime_resume routine
274	 * or else call the system resume routine
275	 */
276	if (runtime_resume)
277		ret = snd_sof_dsp_runtime_resume(sdev);
278	else
279		ret = snd_sof_dsp_resume(sdev);
280	if (ret < 0) {
281		dev_err(sdev->dev,
282			"error: failed to power up DSP after resume\n");
283		return ret;
284	}
285
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
286	/* load the firmware */
287	ret = snd_sof_load_firmware(sdev);
288	if (ret < 0) {
289		dev_err(sdev->dev,
290			"error: failed to load DSP firmware after resume %d\n",
291			ret);
 
292		return ret;
293	}
294
295	/* boot the firmware */
 
 
 
 
 
296	ret = snd_sof_run_firmware(sdev);
297	if (ret < 0) {
298		dev_err(sdev->dev,
299			"error: failed to boot DSP firmware after resume %d\n",
300			ret);
 
301		return ret;
302	}
303
304	/* resume DMA trace, only need send ipc */
305	ret = snd_sof_init_trace_ipc(sdev);
306	if (ret < 0) {
307		/* non fatal */
308		dev_warn(sdev->dev,
309			 "warning: failed to init trace after resume %d\n",
310			 ret);
311	}
312
313	/* restore pipelines */
314	ret = sof_restore_pipelines(sdev);
315	if (ret < 0) {
316		dev_err(sdev->dev,
317			"error: failed to restore pipeline after resume %d\n",
318			ret);
319		return ret;
320	}
321
 
 
 
322	/* notify DSP of system resume */
323	ret = sof_send_pm_ipc(sdev, SOF_IPC_PM_CTX_RESTORE);
324	if (ret < 0)
325		dev_err(sdev->dev,
326			"error: ctx_restore ipc error during resume %d\n",
327			ret);
 
 
 
 
 
 
 
 
 
 
 
 
328
329	return ret;
330}
331
332static int sof_suspend(struct device *dev, bool runtime_suspend)
333{
334	struct snd_sof_dev *sdev = dev_get_drvdata(dev);
 
 
 
 
 
335	int ret;
336
337	/* do nothing if dsp suspend callback is not set */
338	if (!sof_ops(sdev)->suspend)
 
 
 
339		return 0;
340
341	/* release trace */
342	snd_sof_release_trace(sdev);
 
 
 
 
 
 
 
 
343
344	/* set restore_stream for all streams during system suspend */
345	if (!runtime_suspend) {
346		ret = sof_set_hw_params_upon_resume(sdev);
347		if (ret < 0) {
348			dev_err(sdev->dev,
349				"error: setting hw_params flag during suspend %d\n",
350				ret);
351			return ret;
352		}
353	}
354
 
 
 
 
 
 
 
 
 
 
 
 
355#if IS_ENABLED(CONFIG_SND_SOC_SOF_DEBUG_ENABLE_DEBUGFS_CACHE)
356	/* cache debugfs contents during runtime suspend */
357	if (runtime_suspend)
358		sof_cache_debugfs(sdev);
359#endif
360	/* notify DSP of upcoming power down */
361	ret = sof_send_pm_ipc(sdev, SOF_IPC_PM_CTX_SAVE);
362	if (ret == -EBUSY || ret == -EAGAIN) {
363		/*
364		 * runtime PM has logic to handle -EBUSY/-EAGAIN so
365		 * pass these errors up
366		 */
367		dev_err(sdev->dev,
368			"error: ctx_save ipc error during suspend %d\n",
369			ret);
370		return ret;
371	} else if (ret < 0) {
372		/* FW in unexpected state, continue to power down */
373		dev_warn(sdev->dev,
374			 "ctx_save ipc error %d, proceeding with suspend\n",
375			 ret);
376	}
377
378	/* power down all DSP cores */
 
 
 
 
 
 
379	if (runtime_suspend)
380		ret = snd_sof_dsp_runtime_suspend(sdev);
381	else
382		ret = snd_sof_dsp_suspend(sdev);
383	if (ret < 0)
384		dev_err(sdev->dev,
385			"error: failed to power down DSP during suspend %d\n",
386			ret);
387
 
 
 
 
 
 
 
 
388	return ret;
389}
390
 
 
 
 
 
 
 
 
 
 
 
391int snd_sof_runtime_suspend(struct device *dev)
392{
393	return sof_suspend(dev, true);
394}
395EXPORT_SYMBOL(snd_sof_runtime_suspend);
396
397int snd_sof_runtime_idle(struct device *dev)
398{
399	struct snd_sof_dev *sdev = dev_get_drvdata(dev);
400
401	return snd_sof_dsp_runtime_idle(sdev);
402}
403EXPORT_SYMBOL(snd_sof_runtime_idle);
404
405int snd_sof_runtime_resume(struct device *dev)
406{
407	return sof_resume(dev, true);
408}
409EXPORT_SYMBOL(snd_sof_runtime_resume);
410
411int snd_sof_resume(struct device *dev)
412{
413	return sof_resume(dev, false);
414}
415EXPORT_SYMBOL(snd_sof_resume);
416
417int snd_sof_suspend(struct device *dev)
418{
419	return sof_suspend(dev, false);
420}
421EXPORT_SYMBOL(snd_sof_suspend);