ASoC: SOF: ipc4-topology: set copier sink format
authorBard Liao <yung-chuan.liao@linux.intel.com>
Thu, 9 Feb 2023 14:21:23 +0000 (16:21 +0200)
committerMark Brown <broonie@kernel.org>
Thu, 9 Feb 2023 14:31:17 +0000 (14:31 +0000)
MOD_INIT_INSTANCE IPC for a copier only contains the sink format for
output pin 0. Any additional output pins that are used need to have their
sink format set using the LARGE_CONFIG_SET IPC message.
Otherwise, firmware will report error or crash due to NULL format is used.

Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Reviewed-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>
Signed-off-by: Peter Ujfalusi <peter.ujfalusi@linux.intel.com>
Link: https://lore.kernel.org/r/20230209142123.17193-3-peter.ujfalusi@linux.intel.com
Signed-off-by: Mark Brown <broonie@kernel.org>
sound/soc/sof/ipc4-topology.c
sound/soc/sof/ipc4-topology.h
sound/soc/sof/sof-audio.h

index 02ddc48a1107960bf6ef8dc084a96d4f28305030..3e27c7a48ebd39a3136063ee9d4daccf07047cb7 100644 (file)
@@ -1742,6 +1742,55 @@ static void sof_ipc4_put_queue_id(struct snd_sof_widget *swidget, int queue_id,
        ida_free(queue_ida, queue_id);
 }
 
+static int sof_ipc4_set_copier_sink_format(struct snd_sof_dev *sdev,
+                                          struct snd_sof_widget *src_widget,
+                                          struct snd_sof_widget *sink_widget,
+                                          int sink_id)
+{
+       struct sof_ipc4_base_module_cfg *sink_config = sink_widget->private;
+       struct sof_ipc4_base_module_cfg *src_config;
+       struct sof_ipc4_copier_config_set_sink_format format;
+       struct sof_ipc4_fw_module *fw_module;
+       struct sof_ipc4_msg msg = {{ 0 }};
+       u32 header, extension;
+
+       dev_dbg(sdev->dev, "%s set copier sink %d format\n",
+               src_widget->widget->name, sink_id);
+
+       if (WIDGET_IS_DAI(src_widget->id)) {
+               struct snd_sof_dai *dai = src_widget->private;
+
+               src_config = dai->private;
+       } else {
+               src_config = src_widget->private;
+       }
+
+       fw_module = src_widget->module_info;
+
+       format.sink_id = sink_id;
+       memcpy(&format.source_fmt, &src_config->audio_fmt, sizeof(format.source_fmt));
+       memcpy(&format.sink_fmt, &sink_config->audio_fmt, sizeof(format.sink_fmt));
+       msg.data_size = sizeof(format);
+       msg.data_ptr = &format;
+
+       header = fw_module->man4_module_entry.id;
+       header |= SOF_IPC4_MOD_INSTANCE(src_widget->instance_id);
+       header |= SOF_IPC4_MSG_TYPE_SET(SOF_IPC4_MOD_LARGE_CONFIG_SET);
+       header |= SOF_IPC4_MSG_DIR(SOF_IPC4_MSG_REQUEST);
+       header |= SOF_IPC4_MSG_TARGET(SOF_IPC4_MODULE_MSG);
+
+       extension = SOF_IPC4_MOD_EXT_MSG_SIZE(msg.data_size);
+       extension |=
+               SOF_IPC4_MOD_EXT_MSG_PARAM_ID(SOF_IPC4_COPIER_MODULE_CFG_PARAM_SET_SINK_FORMAT);
+       extension |= SOF_IPC4_MOD_EXT_MSG_LAST_BLOCK(1);
+       extension |= SOF_IPC4_MOD_EXT_MSG_FIRST_BLOCK(1);
+
+       msg.primary = header;
+       msg.extension = extension;
+
+       return sof_ipc_tx_message(sdev->ipc, &msg, msg.data_size, NULL, 0);
+}
+
 static int sof_ipc4_route_setup(struct snd_sof_dev *sdev, struct snd_sof_route *sroute)
 {
        struct snd_sof_widget *src_widget = sroute->src_widget;
@@ -1770,6 +1819,17 @@ static int sof_ipc4_route_setup(struct snd_sof_dev *sdev, struct snd_sof_route *
                return sroute->dst_queue_id;
        }
 
+       /* Pin 0 format is already set during copier module init */
+       if (sroute->src_queue_id > 0 && WIDGET_IS_COPIER(src_widget->id)) {
+               ret = sof_ipc4_set_copier_sink_format(sdev, src_widget, sink_widget,
+                                                     sroute->src_queue_id);
+               if (ret < 0) {
+                       dev_err(sdev->dev, "failed to set sink format for %s source queue ID %d\n",
+                               src_widget->widget->name, sroute->src_queue_id);
+                       goto out;
+               }
+       }
+
        dev_dbg(sdev->dev, "bind %s:%d -> %s:%d\n",
                src_widget->widget->name, sroute->src_queue_id,
                sink_widget->widget->name, sroute->dst_queue_id);
@@ -1793,14 +1853,15 @@ static int sof_ipc4_route_setup(struct snd_sof_dev *sdev, struct snd_sof_route *
                dev_err(sdev->dev, "failed to bind modules %s:%d -> %s:%d\n",
                        src_widget->widget->name, sroute->src_queue_id,
                        sink_widget->widget->name, sroute->dst_queue_id);
-
-               sof_ipc4_put_queue_id(src_widget, sroute->src_queue_id,
-                                     SOF_PIN_TYPE_SOURCE);
-               sof_ipc4_put_queue_id(sink_widget, sroute->dst_queue_id,
-                                     SOF_PIN_TYPE_SINK);
+               goto out;
        }
 
        return ret;
+
+out:
+       sof_ipc4_put_queue_id(src_widget, sroute->src_queue_id, SOF_PIN_TYPE_SOURCE);
+       sof_ipc4_put_queue_id(sink_widget, sroute->dst_queue_id, SOF_PIN_TYPE_SINK);
+       return ret;
 }
 
 static int sof_ipc4_route_free(struct snd_sof_dev *sdev, struct snd_sof_route *sroute)
index ee5d31e68a774880133d3906c67221a3ef4f3733..72529179ac22343063ca3ca1ce9873cd36ca75b3 100644 (file)
 /* A magic number from FW */
 #define ALH_MULTI_GTW_COUNT    8
 
+enum sof_ipc4_copier_module_config_params {
+/*
+ * Use LARGE_CONFIG_SET to initialize timestamp event. Ipc mailbox must
+ * contain properly built CopierConfigTimestampInitData struct.
+ */
+       SOF_IPC4_COPIER_MODULE_CFG_PARAM_TIMESTAMP_INIT = 1,
+/*
+ * Use LARGE_CONFIG_SET to initialize copier sink. Ipc mailbox must contain
+ * properly built CopierConfigSetSinkFormat struct.
+ */
+       SOF_IPC4_COPIER_MODULE_CFG_PARAM_SET_SINK_FORMAT,
+/*
+ * Use LARGE_CONFIG_SET to initialize and enable on Copier data segment
+ * event. Ipc mailbox must contain properly built DataSegmentEnabled struct.
+ */
+       SOF_IPC4_COPIER_MODULE_CFG_PARAM_DATA_SEGMENT_ENABLED,
+/*
+ * Use LARGE_CONFIG_GET to retrieve Linear Link Position (LLP) value for non
+ * HD-A gateways.
+ */
+       SOF_IPC4_COPIER_MODULE_CFG_PARAM_LLP_READING,
+/*
+ * Use LARGE_CONFIG_GET to retrieve Linear Link Position (LLP) value for non
+ * HD-A gateways and corresponding total processed data
+ */
+       SOF_IPC4_COPIER_MODULE_CFG_PARAM_LLP_READING_EXTENDED,
+/*
+ * Use LARGE_CONFIG_SET to setup attenuation on output pins. Data is just uint32_t.
+ * note Config is only allowed when output pin is set up for 32bit and source
+ * is connected to Gateway
+ */
+       SOF_IPC4_COPIER_MODULE_CFG_ATTENUATION,
+};
+
+struct sof_ipc4_copier_config_set_sink_format {
+/* Id of sink */
+       u32 sink_id;
+/*
+ * Input format used by the source
+ * attention must be the same as present if already initialized.
+ */
+       struct sof_ipc4_audio_format source_fmt;
+/* Output format used by the sink */
+       struct sof_ipc4_audio_format sink_fmt;
+} __packed __aligned(4);
+
 /**
  * struct sof_ipc4_pipeline - pipeline config data
  * @priority: Priority of this pipeline
index d3104941e83ec8d4e45a7bbfe1f10e68b1cf140b..e0579af9d281543c66465c8319364c39976c8826 100644 (file)
@@ -42,6 +42,7 @@
 #define WIDGET_IS_DAI(id) ((id) == snd_soc_dapm_dai_in || (id) == snd_soc_dapm_dai_out)
 #define WIDGET_IS_AIF(id) ((id) == snd_soc_dapm_aif_in || (id) == snd_soc_dapm_aif_out)
 #define WIDGET_IS_AIF_OR_DAI(id) (WIDGET_IS_DAI(id) || WIDGET_IS_AIF(id))
+#define WIDGET_IS_COPIER(id) (WIDGET_IS_AIF_OR_DAI(id) || (id) == snd_soc_dapm_buffer)
 
 #define SOF_DAI_CLK_INTEL_SSP_MCLK     0
 #define SOF_DAI_CLK_INTEL_SSP_BCLK     1