]> www.infradead.org Git - users/griffoul/linux.git/commitdiff
ASoC: soc-link: add mark for snd_soc_link_startup/shutdown()
authorKuninori Morimoto <kuninori.morimoto.gx@renesas.com>
Mon, 28 Sep 2020 00:00:57 +0000 (09:00 +0900)
committerMark Brown <broonie@kernel.org>
Mon, 28 Sep 2020 16:01:43 +0000 (17:01 +0100)
soc_pcm_open() does rollback when failed (A),
but, it is almost same as soc_pcm_close().

static int soc_pcm_open(xxx)
{
...
if (ret < 0)
goto xxx_err;
...
return 0;

 ^ config_err:
 | ...
 | rtd_startup_err:
(A) ...
 | component_err:
 | ...
 v return ret;
}

The difference is
soc_pcm_close() is for all dai/component/substream,
rollback        is for succeeded part only.

This kind of duplicated code can be a hotbed of bugs,
thus, we want to share soc_pcm_close() and rollback.

Now, soc_pcm_open/close() are handling
1) snd_soc_dai_startup/shutdown()
=> 2) snd_soc_link_startup/shutdown()
3) snd_soc_component_module_get/put()
4) snd_soc_component_open/close()
5) pm_runtime_put/get()

This patch is for 2) snd_soc_link_startup/shutdown().

The idea of having bit-flag or counter is not enough for this purpose.
For example if one DAI is used for 2xPlaybacks for some reasons,
and if 1st Playback was succeeded but 2nd Playback was failed,
2nd Playback rollback doesn't need to call shutdown.
But it has succeeded bit-flag or counter via 1st Playback,
thus, 2nd Playback rollback will call unneeded shutdown.
And 1st Playback's necessary shutdown will not be called,
because bit-flag or counter was cleared by wrong 2nd Playback rollback.

To avoid such case, this patch marks substream pointer when startup() was
succeeded. If rollback needed, it will check rollback flag and marked
substream pointer.

One note here is that it cares *current* startup() only now.
but we might want to check *whole* marked substream in the future.
This patch is using macro named "push/pop", so that it can be easily
update.

Signed-off-by: Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
Link: https://lore.kernel.org/r/87k0webwnv.wl-kuninori.morimoto.gx@renesas.com
Signed-off-by: Mark Brown <broonie@kernel.org>
include/sound/soc-link.h
include/sound/soc.h
sound/soc/soc-link.c
sound/soc/soc-pcm.c

index 337ac566675789c07fb941a9cdc158060e6f6791..dac6c0ce6ede0e9ab825690b5d08d530323575bf 100644 (file)
@@ -14,7 +14,8 @@ int snd_soc_link_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
                                    struct snd_pcm_hw_params *params);
 
 int snd_soc_link_startup(struct snd_pcm_substream *substream);
-void snd_soc_link_shutdown(struct snd_pcm_substream *substream);
+void snd_soc_link_shutdown(struct snd_pcm_substream *substream,
+                          int rollback);
 int snd_soc_link_prepare(struct snd_pcm_substream *substream);
 int snd_soc_link_hw_params(struct snd_pcm_substream *substream,
                           struct snd_pcm_hw_params *params);
index 09a7d8409ade29bfee4fb93f511604648889d4a4..3b038c563ae1470488fbcb68bc1caeb0f98681c8 100644 (file)
@@ -1159,6 +1159,9 @@ struct snd_soc_pcm_runtime {
        unsigned int num; /* 0-based and monotonic increasing */
        struct list_head list; /* rtd list of the soc card */
 
+       /* function mark */
+       struct snd_pcm_substream *mark_startup;
+
        /* bit field */
        unsigned int pop_wait:1;
        unsigned int fe_compr:1; /* for Dynamic PCM */
index cec70b19863e21bd73f0455ba21de5a64d635e5d..2a8881978930d975b5be6e90aaad35dca839b7e5 100644 (file)
@@ -30,6 +30,14 @@ static inline int _soc_link_ret(struct snd_soc_pcm_runtime *rtd,
        return ret;
 }
 
+/*
+ * We might want to check substream by using list.
+ * In such case, we can update these macros.
+ */
+#define soc_link_mark_push(rtd, substream, tgt)                ((rtd)->mark_##tgt = substream)
+#define soc_link_mark_pop(rtd, substream, tgt)         ((rtd)->mark_##tgt = NULL)
+#define soc_link_mark_match(rtd, substream, tgt)       ((rtd)->mark_##tgt == substream)
+
 int snd_soc_link_init(struct snd_soc_pcm_runtime *rtd)
 {
        int ret = 0;
@@ -66,16 +74,27 @@ int snd_soc_link_startup(struct snd_pcm_substream *substream)
            rtd->dai_link->ops->startup)
                ret = rtd->dai_link->ops->startup(substream);
 
+       /* mark substream if succeeded */
+       if (ret == 0)
+               soc_link_mark_push(rtd, substream, startup);
+
        return soc_link_ret(rtd, ret);
 }
 
-void snd_soc_link_shutdown(struct snd_pcm_substream *substream)
+void snd_soc_link_shutdown(struct snd_pcm_substream *substream,
+                          int rollback)
 {
        struct snd_soc_pcm_runtime *rtd = asoc_substream_to_rtd(substream);
 
+       if (rollback && !soc_link_mark_match(rtd, substream, startup))
+               return;
+
        if (rtd->dai_link->ops &&
            rtd->dai_link->ops->shutdown)
                rtd->dai_link->ops->shutdown(substream);
+
+       /* remove marked substream */
+       soc_link_mark_pop(rtd, substream, startup);
 }
 
 int snd_soc_link_prepare(struct snd_pcm_substream *substream)
index 00ed55e40819a9e0993ad0909d236faf75ba2b03..45c19643d8e3a6074b174331d252db28e3a86310 100644 (file)
@@ -684,7 +684,7 @@ static int soc_pcm_close(struct snd_pcm_substream *substream)
        for_each_rtd_dais(rtd, i, dai)
                snd_soc_dai_shutdown(dai, substream, 0);
 
-       snd_soc_link_shutdown(substream);
+       snd_soc_link_shutdown(substream, 0);
 
        soc_pcm_components_close(substream);
 
@@ -815,7 +815,7 @@ config_err:
        for_each_rtd_dais_rollback(rtd, i, dai)
                snd_soc_dai_shutdown(dai, substream, 1);
 
-       snd_soc_link_shutdown(substream);
+       snd_soc_link_shutdown(substream, 1);
 rtd_startup_err:
        soc_pcm_components_close(substream);
 component_err: