ASoC: SOF: Introduce struct snd_sof_pipeline
authorRanjani Sridharan <ranjani.sridharan@linux.intel.com>
Fri, 27 Jan 2023 12:00:25 +0000 (14:00 +0200)
committerMark Brown <broonie@kernel.org>
Fri, 27 Jan 2023 12:14:08 +0000 (12:14 +0000)
Introduce struct snd_sof_pipeline to save the information about
pipelines including the pipeline widget, their status wrt how many PCM's
are using them and whether they are complete or not.

In struct snd_sof_widget, replace pipe_widget with spipe and remove
complete. In struct snd_sof_pcm_stream_pipeline_list, replace
pipe_widgets with pipelines.

Update all users accordingly.

Signed-off-by: Ranjani Sridharan <ranjani.sridharan@linux.intel.com>
Reviewed-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Reviewed-by: Rander Wang <rander.wang@intel.com>
Reviewed-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Reviewed-by: Péter Ujfalusi <peter.ujfalusi@linux.intel.com>
Signed-off-by: Peter Ujfalusi <peter.ujfalusi@linux.intel.com>
Link: https://lore.kernel.org/r/20230127120031.10709-13-peter.ujfalusi@linux.intel.com
Signed-off-by: Mark Brown <broonie@kernel.org>
sound/soc/sof/core.c
sound/soc/sof/intel/hda-dai.c
sound/soc/sof/ipc3-topology.c
sound/soc/sof/ipc4-pcm.c
sound/soc/sof/ipc4-topology.c
sound/soc/sof/sof-audio.c
sound/soc/sof/sof-audio.h
sound/soc/sof/sof-priv.h
sound/soc/sof/topology.c

index 26a3f7c8c914a34dda1f060a00ee7ea0220218c4..7de8673a01ce10ea5aa1bf93ed4b28573f0685c5 100644 (file)
@@ -390,6 +390,7 @@ int snd_sof_device_probe(struct device *dev, struct snd_sof_pdata *plat_data)
        INIT_LIST_HEAD(&sdev->pcm_list);
        INIT_LIST_HEAD(&sdev->kcontrol_list);
        INIT_LIST_HEAD(&sdev->widget_list);
+       INIT_LIST_HEAD(&sdev->pipeline_list);
        INIT_LIST_HEAD(&sdev->dai_list);
        INIT_LIST_HEAD(&sdev->dai_link_list);
        INIT_LIST_HEAD(&sdev->route_list);
index 98eebb4b07e6d3e14b2b493eac0a067f61822c35..193b3e74820af3157d1d52bef32706ae777f3586 100644 (file)
@@ -468,7 +468,7 @@ static int ipc4_hda_dai_trigger(struct snd_pcm_substream *substream,
 
        w = snd_soc_dai_get_widget(dai, substream->stream);
        swidget = w->dobj.private;
-       pipe_widget = swidget->pipe_widget;
+       pipe_widget = swidget->spipe->pipe_widget;
        pipeline = pipe_widget->private;
 
        switch (cmd) {
index 3f52dfb19e01d884a819f1f8b1142db09586dfa9..88b9b9507231b3dfaa2de494f3751955410b3704 100644 (file)
@@ -2233,9 +2233,9 @@ static int sof_ipc3_set_up_all_pipelines(struct snd_sof_dev *sdev, bool verify)
                                        return ret;
                        }
 
-                       swidget->complete = sof_ipc3_complete_pipeline(sdev, swidget);
-                       if (swidget->complete < 0)
-                               return swidget->complete;
+                       swidget->spipe->complete = sof_ipc3_complete_pipeline(sdev, swidget);
+                       if (swidget->spipe->complete < 0)
+                               return swidget->spipe->complete;
                        break;
                default:
                        break;
@@ -2317,7 +2317,8 @@ static int sof_ipc3_tear_down_all_pipelines(struct snd_sof_dev *sdev, bool verif
                if (!verify && !swidget->dynamic_pipeline_widget &&
                    SOF_FW_VER(v->major, v->minor, v->micro) < SOF_FW_VER(2, 2, 0)) {
                        swidget->use_count = 0;
-                       swidget->complete = 0;
+                       if (swidget->spipe)
+                               swidget->spipe->complete = 0;
                        continue;
                }
 
index a5482185cd6cdb003239e696d0b2670201b4992e..17a116e8c47c7d66a1759a0b281e2e942f9059b2 100644 (file)
@@ -68,6 +68,7 @@ static int sof_ipc4_trigger_pipelines(struct snd_soc_component *component,
        struct ipc4_pipeline_set_state_data *data;
        struct snd_sof_widget *pipe_widget;
        struct sof_ipc4_pipeline *pipeline;
+       struct snd_sof_pipeline *spipe;
        struct snd_sof_pcm *spcm;
        int ret;
        int i, j;
@@ -79,7 +80,7 @@ static int sof_ipc4_trigger_pipelines(struct snd_soc_component *component,
        pipeline_list = &spcm->stream[substream->stream].pipeline_list;
 
        /* nothing to trigger if the list is empty */
-       if (!pipeline_list->pipe_widgets)
+       if (!pipeline_list->pipelines)
                return 0;
 
        /* allocate memory for the pipeline data */
@@ -96,7 +97,8 @@ static int sof_ipc4_trigger_pipelines(struct snd_soc_component *component,
         * sink->source would still be guaranteed for each fork independently.
         */
        for (i = pipeline_list->count - 1; i >= 0; i--) {
-               pipe_widget = pipeline_list->pipe_widgets[i];
+               spipe = pipeline_list->pipelines[i];
+               pipe_widget = spipe->pipe_widget;
                pipeline = pipe_widget->private;
                if (pipeline->state != state && !pipeline->skip_during_fe_trigger)
                        data->pipeline_ids[data->count++] = pipe_widget->instance_id;
@@ -122,7 +124,8 @@ static int sof_ipc4_trigger_pipelines(struct snd_soc_component *component,
        /* update PAUSED state for all pipelines that were just triggered */
        for (i = 0; i < data->count; i++) {
                for (j = 0; j < pipeline_list->count; j++) {
-                       pipe_widget = pipeline_list->pipe_widgets[j];
+                       spipe = pipeline_list->pipelines[j];
+                       pipe_widget = spipe->pipe_widget;
                        pipeline = pipe_widget->private;
 
                        if (data->pipeline_ids[i] == pipe_widget->instance_id) {
@@ -146,7 +149,8 @@ static int sof_ipc4_trigger_pipelines(struct snd_soc_component *component,
        /* update final state for all pipelines that were just triggered */
        for (i = 0; i < data->count; i++) {
                for (j = 0; j < pipeline_list->count; j++) {
-                       pipe_widget = pipeline_list->pipe_widgets[j];
+                       spipe = pipeline_list->pipelines[j];
+                       pipe_widget = spipe->pipe_widget;
                        pipeline = pipe_widget->private;
 
                        if (data->pipeline_ids[i] == pipe_widget->instance_id) {
@@ -274,8 +278,8 @@ static void sof_ipc4_pcm_free(struct snd_sof_dev *sdev, struct snd_sof_pcm *spcm
 
        for_each_pcm_streams(stream) {
                pipeline_list = &spcm->stream[stream].pipeline_list;
-               kfree(pipeline_list->pipe_widgets);
-               pipeline_list->pipe_widgets = NULL;
+               kfree(pipeline_list->pipelines);
+               pipeline_list->pipelines = NULL;
        }
 }
 
@@ -289,10 +293,9 @@ static int sof_ipc4_pcm_setup(struct snd_sof_dev *sdev, struct snd_sof_pcm *spcm
                pipeline_list = &spcm->stream[stream].pipeline_list;
 
                /* allocate memory for max number of pipeline IDs */
-               pipeline_list->pipe_widgets = kcalloc(ipc4_data->max_num_pipelines,
-                                                     sizeof(struct snd_sof_widget *),
-                                                     GFP_KERNEL);
-               if (!pipeline_list->pipe_widgets) {
+               pipeline_list->pipelines = kcalloc(ipc4_data->max_num_pipelines,
+                                                  sizeof(struct snd_sof_widget *), GFP_KERNEL);
+               if (!pipeline_list->pipelines) {
                        sof_ipc4_pcm_free(sdev, spcm);
                        return -ENOMEM;
                }
index f3b1a7f81216e95849df444e6dfc4d61c4f9a1ae..2f82b5e02a2789dd707533c11615b520c2bb9a49 100644 (file)
@@ -855,7 +855,7 @@ sof_ipc4_update_pipeline_mem_usage(struct snd_sof_dev *sdev, struct snd_sof_widg
 
        total = SOF_IPC4_FW_PAGE(task_mem + queue_mem);
 
-       pipe_widget = swidget->pipe_widget;
+       pipe_widget = swidget->spipe->pipe_widget;
        pipeline = pipe_widget->private;
        pipeline->mem_usage += total;
 }
@@ -969,7 +969,7 @@ static void sof_ipc4_unprepare_copier_module(struct snd_sof_widget *swidget)
        struct sof_ipc4_pipeline *pipeline;
 
        /* reset pipeline memory usage */
-       pipe_widget = swidget->pipe_widget;
+       pipe_widget = swidget->spipe->pipe_widget;
        pipeline = pipe_widget->private;
        pipeline->mem_usage = 0;
 
@@ -1136,7 +1136,7 @@ sof_ipc4_prepare_copier_module(struct snd_sof_widget *swidget,
                struct snd_sof_widget *pipe_widget;
                struct sof_ipc4_pipeline *pipeline;
 
-               pipe_widget = swidget->pipe_widget;
+               pipe_widget = swidget->spipe->pipe_widget;
                pipeline = pipe_widget->private;
                ipc4_copier = (struct sof_ipc4_copier *)swidget->private;
                gtw_attr = ipc4_copier->gtw_attr;
@@ -1495,7 +1495,7 @@ static int sof_ipc4_control_setup(struct snd_sof_dev *sdev, struct snd_sof_contr
 
 static int sof_ipc4_widget_setup(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget)
 {
-       struct snd_sof_widget *pipe_widget = swidget->pipe_widget;
+       struct snd_sof_widget *pipe_widget = swidget->spipe->pipe_widget;
        struct sof_ipc4_fw_data *ipc4_data = sdev->private;
        struct sof_ipc4_pipeline *pipeline;
        struct sof_ipc4_msg *msg;
@@ -1815,7 +1815,7 @@ static int sof_ipc4_route_free(struct snd_sof_dev *sdev, struct snd_sof_route *s
         * routes belonging to the same pipeline will be disconnected by the FW when the pipeline
         * is freed. So avoid sending this IPC which will be ignored by the FW anyway.
         */
-       if (src_widget->pipe_widget == sink_widget->pipe_widget)
+       if (src_widget->spipe->pipe_widget == sink_widget->spipe->pipe_widget)
                goto out;
 
        header = src_fw_module->man4_module_entry.id;
@@ -1846,7 +1846,7 @@ out:
 static int sof_ipc4_dai_config(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget,
                               unsigned int flags, struct snd_sof_dai_config_data *data)
 {
-       struct snd_sof_widget *pipe_widget = swidget->pipe_widget;
+       struct snd_sof_widget *pipe_widget = swidget->spipe->pipe_widget;
        struct sof_ipc4_pipeline *pipeline = pipe_widget->private;
        struct snd_sof_dai *dai = swidget->private;
        struct sof_ipc4_gtw_attributes *gtw_attr;
index 3d8b3ef54889405729a8be09d0c050e1deb8ce30..e626a1e0483ee2b39dd9e7143b9fe2fbd97c57ba 100644 (file)
@@ -31,6 +31,7 @@ static void sof_reset_route_setup_status(struct snd_sof_dev *sdev, struct snd_so
 int sof_widget_free(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget)
 {
        const struct sof_ipc_tplg_ops *tplg_ops = sof_ipc_get_ops(sdev, tplg);
+       struct snd_sof_widget *pipe_widget;
        int err = 0;
        int ret;
 
@@ -43,6 +44,8 @@ int sof_widget_free(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget)
        if (--swidget->use_count)
                return 0;
 
+       pipe_widget = swidget->spipe->pipe_widget;
+
        /* reset route setup status for all routes that contain this widget */
        sof_reset_route_setup_status(sdev, swidget);
 
@@ -67,12 +70,15 @@ int sof_widget_free(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget)
         * skip for static pipelines
         */
        if (swidget->dynamic_pipeline_widget && swidget->id != snd_soc_dapm_scheduler) {
-               ret = sof_widget_free(sdev, swidget->pipe_widget);
+               ret = sof_widget_free(sdev, pipe_widget);
                if (ret < 0 && !err)
                        err = ret;
-               swidget->pipe_widget->complete = 0;
        }
 
+       /* clear pipeline complete */
+       if (swidget->id == snd_soc_dapm_scheduler)
+               swidget->spipe->complete = 0;
+
        if (!err)
                dev_dbg(sdev->dev, "widget %s freed\n", swidget->widget->name);
 
@@ -103,14 +109,13 @@ int sof_widget_setup(struct snd_sof_dev *sdev, struct snd_sof_widget *swidget)
         * widget in the pipeline is freed. Skip setting up scheduler widget for static pipelines.
         */
        if (swidget->dynamic_pipeline_widget && swidget->id != snd_soc_dapm_scheduler) {
-               if (!swidget->pipe_widget) {
-                       dev_err(sdev->dev, "No scheduler widget set for %s\n",
-                               swidget->widget->name);
+               if (!swidget->spipe || !swidget->spipe->pipe_widget) {
+                       dev_err(sdev->dev, "No pipeline set for %s\n", swidget->widget->name);
                        ret = -EINVAL;
                        goto use_count_dec;
                }
 
-               ret = sof_widget_setup(sdev, swidget->pipe_widget);
+               ret = sof_widget_setup(sdev, swidget->spipe->pipe_widget);
                if (ret < 0)
                        goto use_count_dec;
        }
@@ -159,7 +164,7 @@ core_put:
        snd_sof_dsp_core_put(sdev, swidget->core);
 pipe_widget_free:
        if (swidget->id != snd_soc_dapm_scheduler)
-               sof_widget_free(sdev, swidget->pipe_widget);
+               sof_widget_free(sdev, swidget->spipe->pipe_widget);
 use_count_dec:
        swidget->use_count--;
        return ret;
@@ -408,7 +413,7 @@ static int sof_set_up_widgets_in_path(struct snd_sof_dev *sdev, struct snd_soc_d
        struct snd_sof_pcm_stream_pipeline_list *pipeline_list = &spcm->stream[dir].pipeline_list;
        struct snd_soc_dapm_widget_list *list = spcm->stream[dir].list;
        struct snd_sof_widget *swidget = widget->dobj.private;
-       struct snd_sof_widget *pipe_widget;
+       struct snd_sof_pipeline *spipe;
        struct snd_soc_dapm_path *p;
        int ret;
 
@@ -420,7 +425,7 @@ static int sof_set_up_widgets_in_path(struct snd_sof_dev *sdev, struct snd_soc_d
                        return ret;
 
                /* skip populating the pipe_widgets array if it is NULL */
-               if (!pipeline_list->pipe_widgets)
+               if (!pipeline_list->pipelines)
                        goto sink_setup;
 
                /*
@@ -429,14 +434,14 @@ static int sof_set_up_widgets_in_path(struct snd_sof_dev *sdev, struct snd_soc_d
                 * order source to sink.
                 */
                for (i = 0; i < pipeline_list->count; i++) {
-                       pipe_widget = pipeline_list->pipe_widgets[i];
-                       if (pipe_widget == swidget->pipe_widget)
+                       spipe = pipeline_list->pipelines[i];
+                       if (spipe == swidget->spipe)
                                break;
                }
 
                if (i == pipeline_list->count) {
                        pipeline_list->count++;
-                       pipeline_list->pipe_widgets[i] = swidget->pipe_widget;
+                       pipeline_list->pipelines[i] = swidget->spipe;
                }
        }
 
@@ -571,11 +576,20 @@ int sof_widget_list_setup(struct snd_sof_dev *sdev, struct snd_sof_pcm *spcm,
        for_each_dapm_widgets(list, i, widget) {
                struct snd_sof_widget *swidget = widget->dobj.private;
                struct snd_sof_widget *pipe_widget;
+               struct snd_sof_pipeline *spipe;
 
                if (!swidget)
                        continue;
 
-               pipe_widget = swidget->pipe_widget;
+               spipe = swidget->spipe;
+               if (!spipe) {
+                       dev_err(sdev->dev, "no pipeline found for %s\n",
+                               swidget->widget->name);
+                       ret = -EINVAL;
+                       goto widget_free;
+               }
+
+               pipe_widget = spipe->pipe_widget;
                if (!pipe_widget) {
                        dev_err(sdev->dev, "error: no pipeline widget found for %s\n",
                                swidget->widget->name);
@@ -583,13 +597,13 @@ int sof_widget_list_setup(struct snd_sof_dev *sdev, struct snd_sof_pcm *spcm,
                        goto widget_free;
                }
 
-               if (pipe_widget->complete)
+               if (spipe->complete)
                        continue;
 
                if (tplg_ops && tplg_ops->pipeline_complete) {
-                       pipe_widget->complete = tplg_ops->pipeline_complete(sdev, pipe_widget);
-                       if (pipe_widget->complete < 0) {
-                               ret = pipe_widget->complete;
+                       spipe->complete = tplg_ops->pipeline_complete(sdev, pipe_widget);
+                       if (spipe->complete < 0) {
+                               ret = spipe->complete;
                                goto widget_free;
                        }
                }
index bb5c61dd9b1e056d09ce778193d70f083ef5eea6..f1bbd1adc8b63432d7b45fd1a04b6690e70cae04 100644 (file)
@@ -288,11 +288,11 @@ struct sof_token_info {
 /**
  * struct snd_sof_pcm_stream_pipeline_list - List of pipelines associated with a PCM stream
  * @count: number of pipeline widgets in the @pipe_widgets array
- * @pipe_widgets: array of pipeline widgets
+ * @pipelines: array of pipelines
  */
 struct snd_sof_pcm_stream_pipeline_list {
        u32 count;
-       struct snd_sof_widget **pipe_widgets;
+       struct snd_sof_pipeline **pipelines;
 };
 
 /* PCM stream, mapped to FW component  */
@@ -382,11 +382,6 @@ struct snd_sof_widget {
        struct snd_soc_component *scomp;
        int comp_id;
        int pipeline_id;
-       /*
-        * complete flag is used to indicate that pipeline set up is complete for scheduler type
-        * widgets. It is unused for all other widget types.
-        */
-       int complete;
        /*
         * the prepared flag is used to indicate that a widget has been prepared for getting set
         * up in the DSP.
@@ -413,7 +408,7 @@ struct snd_sof_widget {
 
        struct snd_soc_dapm_widget *widget;
        struct list_head list;  /* list in sdev widget list */
-       struct snd_sof_widget *pipe_widget;
+       struct snd_sof_pipeline *spipe;
        void *module_info;
 
        const guid_t uuid;
@@ -451,6 +446,22 @@ struct snd_sof_widget {
        void *private;          /* core does not touch this */
 };
 
+/** struct snd_sof_pipeline - ASoC SOF pipeline
+ * @pipe_widget: Pointer to the pipeline widget
+ * @started_count: Count of number of PCM's that have started this pipeline
+ * @paused_count: Count of number of PCM's that have started and have currently paused this
+                 pipeline
+ * @complete: flag used to indicate that pipeline set up is complete.
+ * @list: List item in sdev pipeline_list
+ */
+struct snd_sof_pipeline {
+       struct snd_sof_widget *pipe_widget;
+       int started_count;
+       int paused_count;
+       int complete;
+       struct list_head list;
+};
+
 /* ASoC SOF DAPM route */
 struct snd_sof_route {
        struct snd_soc_component *scomp;
index 86fc5c6a9c39db464bf25743c47eb083a258b1e3..208a30ff3db9bb34056c717d9c16af915873e1ac 100644 (file)
@@ -578,6 +578,7 @@ struct snd_sof_dev {
        struct list_head pcm_list;
        struct list_head kcontrol_list;
        struct list_head widget_list;
+       struct list_head pipeline_list;
        struct list_head dai_list;
        struct list_head dai_link_list;
        struct list_head route_list;
index 51f6fed45ae73ab9b6411646c94e1c2bfcb255ea..33ca3067262b8747ddc8216c9ec951300db500c2 100644 (file)
@@ -1402,7 +1402,6 @@ static int sof_widget_ready(struct snd_soc_component *scomp, int index,
        swidget->scomp = scomp;
        swidget->widget = w;
        swidget->comp_id = sdev->next_comp_id++;
-       swidget->complete = 0;
        swidget->id = w->id;
        swidget->pipeline_id = index;
        swidget->private = NULL;
@@ -1553,6 +1552,23 @@ static int sof_widget_ready(struct snd_soc_component *scomp, int index,
                }
        }
 
+       /* create and add pipeline for scheduler type widgets */
+       if (w->id == snd_soc_dapm_scheduler) {
+               struct snd_sof_pipeline *spipe;
+
+               spipe = kzalloc(sizeof(*spipe), GFP_KERNEL);
+               if (!spipe) {
+                       kfree(swidget->private);
+                       kfree(swidget->tuples);
+                       kfree(swidget);
+                       return -ENOMEM;
+               }
+
+               spipe->pipe_widget = swidget;
+               swidget->spipe = spipe;
+               list_add(&spipe->list, &sdev->pipeline_list);
+       }
+
        w->dobj.private = swidget;
        list_add(&swidget->list, &sdev->widget_list);
        return ret;
@@ -1608,6 +1624,15 @@ static int sof_widget_unload(struct snd_soc_component *scomp,
                sof_disconnect_dai_widget(scomp, widget);
 
                break;
+       case snd_soc_dapm_scheduler:
+       {
+               struct snd_sof_pipeline *spipe = swidget->spipe;
+
+               list_del(&spipe->list);
+               kfree(spipe);
+               swidget->spipe = NULL;
+               break;
+       }
        default:
                break;
        }
@@ -2082,18 +2107,19 @@ err:
 }
 
 /**
- * sof_set_pipe_widget - Set pipe_widget for a component
+ * sof_set_widget_pipeline - Set pipeline for a component
  * @sdev: pointer to struct snd_sof_dev
- * @pipe_widget: pointer to struct snd_sof_widget of type snd_soc_dapm_scheduler
+ * @spipe: pointer to struct snd_sof_pipeline
  * @swidget: pointer to struct snd_sof_widget that has the same pipeline ID as @pipe_widget
  *
  * Return: 0 if successful, -EINVAL on error.
  * The function checks if @swidget is associated with any volatile controls. If so, setting
  * the dynamic_pipeline_widget is disallowed.
  */
-static int sof_set_pipe_widget(struct snd_sof_dev *sdev, struct snd_sof_widget *pipe_widget,
-                              struct snd_sof_widget *swidget)
+static int sof_set_widget_pipeline(struct snd_sof_dev *sdev, struct snd_sof_pipeline *spipe,
+                                  struct snd_sof_widget *swidget)
 {
+       struct snd_sof_widget *pipe_widget = spipe->pipe_widget;
        struct snd_sof_control *scontrol;
 
        if (pipe_widget->dynamic_pipeline_widget) {
@@ -2108,8 +2134,8 @@ static int sof_set_pipe_widget(struct snd_sof_dev *sdev, struct snd_sof_widget *
                        }
        }
 
-       /* set the pipe_widget and apply the dynamic_pipeline_widget_flag */
-       swidget->pipe_widget = pipe_widget;
+       /* set the pipeline and apply the dynamic_pipeline_widget_flag */
+       swidget->spipe = spipe;
        swidget->dynamic_pipeline_widget = pipe_widget->dynamic_pipeline_widget;
 
        return 0;
@@ -2123,6 +2149,7 @@ static int sof_complete(struct snd_soc_component *scomp)
        struct snd_sof_widget *swidget, *comp_swidget;
        const struct sof_ipc_tplg_widget_ops *widget_ops;
        struct snd_sof_control *scontrol;
+       struct snd_sof_pipeline *spipe;
        int ret;
 
        widget_ops = tplg_ops ? tplg_ops->widget : NULL;
@@ -2155,23 +2182,21 @@ static int sof_complete(struct snd_soc_component *scomp)
        }
 
        /* set the pipe_widget and apply the dynamic_pipeline_widget_flag */
-       list_for_each_entry(swidget, &sdev->widget_list, list) {
-               switch (swidget->id) {
-               case snd_soc_dapm_scheduler:
-                       /*
-                        * Apply the dynamic_pipeline_widget flag and set the pipe_widget field
-                        * for all widgets that have the same pipeline ID as the scheduler widget
-                        */
-                       list_for_each_entry(comp_swidget, &sdev->widget_list, list)
-                               if (comp_swidget->pipeline_id == swidget->pipeline_id) {
-                                       ret = sof_set_pipe_widget(sdev, swidget, comp_swidget);
-                                       if (ret < 0)
-                                               return ret;
-                               }
-                       break;
-               default:
-                       break;
-               }
+       list_for_each_entry(spipe, &sdev->pipeline_list, list) {
+               struct snd_sof_widget *pipe_widget = spipe->pipe_widget;
+
+               /*
+                * Apply the dynamic_pipeline_widget flag and set the pipe_widget field
+                * for all widgets that have the same pipeline ID as the scheduler widget.
+                * Skip the scheduler widgets as they have their pipeline set during widget_ready
+                */
+               list_for_each_entry(comp_swidget, &sdev->widget_list, list)
+                       if (comp_swidget->widget->id != snd_soc_dapm_scheduler &&
+                           comp_swidget->pipeline_id == pipe_widget->pipeline_id) {
+                               ret = sof_set_widget_pipeline(sdev, spipe, comp_swidget);
+                               if (ret < 0)
+                                       return ret;
+                       }
        }
 
        /* verify topology components loading including dynamic pipelines */