struct v4l2_subdev *source;
struct v4l2_subdev subdev;
struct ti_csi2rx_ctx ctx[TI_CSI2RX_MAX_CTX];
+ struct notifier_block pm_notifier;
u8 pix_per_clk;
/* Buffer to drain stale data from PSI-L endpoint */
struct {
return 0;
}
+static int ti_csi2rx_suspend(struct device *dev)
+{
+ struct ti_csi2rx_dev *csi = dev_get_drvdata(dev);
+ enum ti_csi2rx_dma_state state;
+ struct ti_csi2rx_ctx *ctx;
+ struct ti_csi2rx_dma *dma;
+ unsigned long flags = 0;
+ int i, ret = 0;
+
+ /* If device was not in use we can simply suspend */
+ if (pm_runtime_status_suspended(dev))
+ return 0;
+
+ /*
+ * If device is running, assert the pixel reset to cleanly stop any
+ * on-going streams before we suspend.
+ */
+ writel(0, csi->shim + SHIM_CNTL);
+
+ for (i = 0; i < csi->num_ctx; i++) {
+ ctx = &csi->ctx[i];
+ dma = &ctx->dma;
+
+ spin_lock_irqsave(&dma->lock, flags);
+ state = dma->state;
+ spin_unlock_irqrestore(&dma->lock, flags);
+
+ if (state != TI_CSI2RX_DMA_STOPPED) {
+ /* Disable source */
+ ret = v4l2_subdev_disable_streams(&csi->subdev,
+ TI_CSI2RX_PAD_FIRST_SOURCE + ctx->idx,
+ BIT(0));
+ if (ret)
+ dev_err(csi->dev, "Failed to stop subdev stream\n");
+ }
+
+ /* Stop any on-going streams */
+ writel(0, csi->shim + SHIM_DMACNTX(ctx->idx));
+
+ /* Drain DMA */
+ ti_csi2rx_drain_dma(ctx);
+
+ /* Terminate DMA */
+ ret = dmaengine_terminate_sync(ctx->dma.chan);
+ if (ret)
+ dev_err(csi->dev, "Failed to stop DMA\n");
+ }
+
+ return ret;
+}
+
+static int ti_csi2rx_resume(struct device *dev)
+{
+ struct ti_csi2rx_dev *csi = dev_get_drvdata(dev);
+ struct ti_csi2rx_ctx *ctx;
+ struct ti_csi2rx_dma *dma;
+ struct ti_csi2rx_buffer *buf;
+ unsigned long flags = 0;
+ unsigned int reg;
+ int i, ret = 0;
+
+ /* If device was not in use, we can simply wakeup */
+ if (pm_runtime_status_suspended(dev))
+ return 0;
+
+ /* If device was in use before, restore all the running streams */
+ reg = SHIM_CNTL_PIX_RST;
+ writel(reg, csi->shim + SHIM_CNTL);
+
+ for (i = 0; i < csi->num_ctx; i++) {
+ ctx = &csi->ctx[i];
+ dma = &ctx->dma;
+ spin_lock_irqsave(&dma->lock, flags);
+ if (dma->state != TI_CSI2RX_DMA_STOPPED) {
+ /* Re-submit all previously submitted buffers to DMA */
+ list_for_each_entry(buf, &ctx->dma.submitted, list) {
+ ti_csi2rx_start_dma(ctx, buf);
+ }
+ spin_unlock_irqrestore(&dma->lock, flags);
+
+ /* Restore stream config */
+ ti_csi2rx_setup_shim(ctx);
+
+ ret = v4l2_subdev_enable_streams(&csi->subdev,
+ TI_CSI2RX_PAD_FIRST_SOURCE + ctx->idx,
+ BIT(0));
+ if (ret)
+ dev_err(ctx->csi->dev, "Failed to start subdev\n");
+ } else {
+ spin_unlock_irqrestore(&dma->lock, flags);
+ }
+ }
+
+ return ret;
+}
+
+static int ti_csi2rx_pm_notifier(struct notifier_block *nb,
+ unsigned long action, void *data)
+{
+ struct ti_csi2rx_dev *csi =
+ container_of(nb, struct ti_csi2rx_dev, pm_notifier);
+
+ switch (action) {
+ case PM_HIBERNATION_PREPARE:
+ case PM_SUSPEND_PREPARE:
+ case PM_RESTORE_PREPARE:
+ ti_csi2rx_suspend(csi->dev);
+ break;
+ case PM_POST_SUSPEND:
+ case PM_POST_HIBERNATION:
+ case PM_POST_RESTORE:
+ ti_csi2rx_resume(csi->dev);
+ break;
+ }
+
+ return NOTIFY_DONE;
+}
+
static const struct dev_pm_ops ti_csi2rx_pm_ops = {
RUNTIME_PM_OPS(ti_csi2rx_runtime_suspend, ti_csi2rx_runtime_resume,
NULL)
goto err_notifier;
}
+ /*
+ * Use PM notifier instead of .suspend/.resume callbacks because the
+ * ordering of callbacks among camera pipeline devices (sensor, serdes,
+ * CSI bridge) cannot be enforced even with device links. The notifier
+ * is called when the system is fully functional, ensuring all
+ * dependencies are available when stopping/starting streams.
+ */
+ csi->pm_notifier.notifier_call = ti_csi2rx_pm_notifier;
+ ret = register_pm_notifier(&csi->pm_notifier);
+ if (ret) {
+ dev_err(csi->dev, "Failed to create PM notifier: %d\n", ret);
+ goto err_notifier;
+ }
+
return 0;
err_notifier:
ti_csi2rx_cleanup_ctx(&csi->ctx[i]);
ti_csi2rx_cleanup_notifier(csi);
+ unregister_pm_notifier(&csi->pm_notifier);
+
ti_csi2rx_cleanup_v4l2(csi);
dma_free_coherent(csi->dev, csi->drain.len, csi->drain.vaddr,
csi->drain.paddr);