ASoC: SOF: Intel: hda-ipc: Do not process IPC reply before firmware boot
authorPeter Ujfalusi <peter.ujfalusi@linux.intel.com>
Tue, 12 Jul 2022 12:23:56 +0000 (15:23 +0300)
committerMark Brown <broonie@kernel.org>
Tue, 12 Jul 2022 12:45:02 +0000 (13:45 +0100)
It is not yet clear, but it is possible to create a firmware so broken
that it will send a reply message before a FW_READY message (it is not
yet clear if FW_READY will arrive later).
Since the reply_data is allocated only after the FW_READY message, this
will lead to a NULL pointer dereference if not filtered out.

The issue was reported with IPC4 firmware but the same condition is present
for IPC3.

Reported-by: Kai Vehmanen <kai.vehmanen@linux.intel.com>
Signed-off-by: Peter Ujfalusi <peter.ujfalusi@linux.intel.com>
Reviewed-by: Ranjani Sridharan <ranjani.sridharan@linux.intel.com>
Reviewed-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20220712122357.31282-3-peter.ujfalusi@linux.intel.com
Signed-off-by: Mark Brown <broonie@kernel.org>
sound/soc/sof/intel/hda-ipc.c

index f08011249955279325bd36cd5ec193e94f850764..65e688f749eaf16955e29d2f7bd52273de024d99 100644 (file)
@@ -148,17 +148,23 @@ irqreturn_t hda_dsp_ipc4_irq_thread(int irq, void *context)
 
                if (primary & SOF_IPC4_MSG_DIR_MASK) {
                        /* Reply received */
-                       struct sof_ipc4_msg *data = sdev->ipc->msg.reply_data;
+                       if (likely(sdev->fw_state == SOF_FW_BOOT_COMPLETE)) {
+                               struct sof_ipc4_msg *data = sdev->ipc->msg.reply_data;
 
-                       data->primary = primary;
-                       data->extension = extension;
+                               data->primary = primary;
+                               data->extension = extension;
 
-                       spin_lock_irq(&sdev->ipc_lock);
+                               spin_lock_irq(&sdev->ipc_lock);
 
-                       snd_sof_ipc_get_reply(sdev);
-                       snd_sof_ipc_reply(sdev, data->primary);
+                               snd_sof_ipc_get_reply(sdev);
+                               snd_sof_ipc_reply(sdev, data->primary);
 
-                       spin_unlock_irq(&sdev->ipc_lock);
+                               spin_unlock_irq(&sdev->ipc_lock);
+                       } else {
+                               dev_dbg_ratelimited(sdev->dev,
+                                                   "IPC reply before FW_READY: %#x|%#x\n",
+                                                   primary, extension);
+                       }
                } else {
                        /* Notification received */
 
@@ -225,16 +231,21 @@ irqreturn_t hda_dsp_ipc_irq_thread(int irq, void *context)
                 * place, the message might not yet be marked as expecting a
                 * reply.
                 */
-               spin_lock_irq(&sdev->ipc_lock);
+               if (likely(sdev->fw_state == SOF_FW_BOOT_COMPLETE)) {
+                       spin_lock_irq(&sdev->ipc_lock);
 
-               /* handle immediate reply from DSP core */
-               hda_dsp_ipc_get_reply(sdev);
-               snd_sof_ipc_reply(sdev, msg);
+                       /* handle immediate reply from DSP core */
+                       hda_dsp_ipc_get_reply(sdev);
+                       snd_sof_ipc_reply(sdev, msg);
 
-               /* set the done bit */
-               hda_dsp_ipc_dsp_done(sdev);
+                       /* set the done bit */
+                       hda_dsp_ipc_dsp_done(sdev);
 
-               spin_unlock_irq(&sdev->ipc_lock);
+                       spin_unlock_irq(&sdev->ipc_lock);
+               } else {
+                       dev_dbg_ratelimited(sdev->dev, "IPC reply before FW_READY: %#x\n",
+                                           msg);
+               }
 
                ipc_irq = true;
        }