]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
media: synopsys: Add support for multiple streams
authorGuoniu Zhou <guoniu.zhou@oss.nxp.com>
Tue, 19 May 2026 02:07:40 +0000 (10:07 +0800)
committerSakari Ailus <sakari.ailus@linux.intel.com>
Wed, 20 May 2026 11:28:37 +0000 (14:28 +0300)
The current driver only supports single stream operation. Add support
for multiple concurrent streams by tracking enabled streams with a
bitmask and only initializing the hardware once for the first stream.

This enables use cases such as surround view systems where multiple
camera streams need to be processed simultaneously through the same
CSI-2 receiver interface.

Reviewed-by: Frank Li <Frank.Li@nxp.com>
Signed-off-by: Guoniu Zhou <guoniu.zhou@oss.nxp.com>
Signed-off-by: Sakari Ailus <sakari.ailus@linux.intel.com>
drivers/media/platform/synopsys/dw-mipi-csi2rx.c

index f45466ede2bb74a3f46135ca85453a2c6951b163..92178a3dec5d5e6aceb8e5cbb0714cc696f2981d 100644 (file)
@@ -113,6 +113,7 @@ struct dw_mipi_csi2rx_device {
 
        enum v4l2_mbus_type bus_type;
        u32 lanes_num;
+       u64 enabled_streams;
 
        const struct dw_mipi_csi2rx_drvdata *drvdata;
 };
@@ -539,26 +540,33 @@ static int dw_mipi_csi2rx_enable_streams(struct v4l2_subdev *sd,
                                               DW_MIPI_CSI2RX_PAD_SRC,
                                               &streams_mask);
 
-       ret = pm_runtime_resume_and_get(dev);
-       if (ret)
-               goto err;
+       if (!csi2->enabled_streams) {
+               ret = pm_runtime_resume_and_get(dev);
+               if (ret)
+                       goto err;
 
-       ret = dw_mipi_csi2rx_start(csi2);
-       if (ret) {
-               dev_err(dev, "failed to enable CSI hardware\n");
-               goto err_pm_runtime_put;
+               ret = dw_mipi_csi2rx_start(csi2);
+               if (ret) {
+                       dev_err(dev, "failed to enable CSI hardware\n");
+                       goto err_pm_runtime_put;
+               }
        }
 
        ret = v4l2_subdev_enable_streams(remote_sd, remote_pad->index, mask);
        if (ret)
                goto err_csi_stop;
 
+       csi2->enabled_streams |= streams_mask;
+
        return 0;
 
 err_csi_stop:
-       dw_mipi_csi2rx_stop(csi2);
+       /* Stop CSI hardware if no streams are enabled */
+       if (!csi2->enabled_streams)
+               dw_mipi_csi2rx_stop(csi2);
 err_pm_runtime_put:
-       pm_runtime_put(dev);
+       if (!csi2->enabled_streams)
+               pm_runtime_put(dev);
 err:
        return ret;
 }
@@ -583,10 +591,15 @@ static int dw_mipi_csi2rx_disable_streams(struct v4l2_subdev *sd,
                                               &streams_mask);
 
        ret = v4l2_subdev_disable_streams(remote_sd, remote_pad->index, mask);
+       if (ret)
+               dev_err(dev, "failed to disable streams on remote subdev: %d\n", ret);
 
-       dw_mipi_csi2rx_stop(csi2);
+       csi2->enabled_streams &= ~streams_mask;
 
-       pm_runtime_put(dev);
+       if (!csi2->enabled_streams) {
+               dw_mipi_csi2rx_stop(csi2);
+               pm_runtime_put(dev);
+       }
 
        return ret;
 }