ASoC: Intel: avs: Disconnect substream if suspend or resume fails
authorCezary Rojewski <cezary.rojewski@intel.com>
Wed, 16 Nov 2022 11:55:50 +0000 (12:55 +0100)
committerMark Brown <broonie@kernel.org>
Tue, 22 Nov 2022 19:32:27 +0000 (19:32 +0000)
To improve performance and overall system stability, suspend/resume
operations for ASoC cards always return success status and defer the
actual work.

Because of that, if a substream fails to resume, userspace may still
attempt to invoke commands on it as from their perspective the operation
completed successfully. Set substream's state to DISCONNECTED to ensure
no further commands are attempted.

Signed-off-by: Cezary Rojewski <cezary.rojewski@intel.com>
Link: https://lore.kernel.org/r/20221116115550.1100398-3-cezary.rojewski@intel.com
Signed-off-by: Mark Brown <broonie@kernel.org>
sound/soc/intel/avs/pcm.c

index b4e7514f878906fd695c21580da26aeddf299004..70d687fa9923e46fe04b05bb2bc75eef6bbbbf2e 100644 (file)
@@ -934,8 +934,11 @@ static int avs_component_pm_op(struct snd_soc_component *component, bool be,
                        rtd = snd_pcm_substream_chip(data->substream);
                        if (rtd->dai_link->no_pcm == be && !rtd->dai_link->ignore_suspend) {
                                ret = op(dai, data);
-                               if (ret < 0)
+                               if (ret < 0) {
+                                       __snd_pcm_set_state(data->substream->runtime,
+                                                           SNDRV_PCM_STATE_DISCONNECTED);
                                        return ret;
+                               }
                        }
                }
 
@@ -944,8 +947,11 @@ static int avs_component_pm_op(struct snd_soc_component *component, bool be,
                        rtd = snd_pcm_substream_chip(data->substream);
                        if (rtd->dai_link->no_pcm == be && !rtd->dai_link->ignore_suspend) {
                                ret = op(dai, data);
-                               if (ret < 0)
+                               if (ret < 0) {
+                                       __snd_pcm_set_state(data->substream->runtime,
+                                                           SNDRV_PCM_STATE_DISCONNECTED);
                                        return ret;
+                               }
                        }
                }
        }