On 5/26/26 08:17, Melissa Wen wrote:
Only allow updates on colorops that are part of an active pipeline, i.e.
check if a colorop belongs to the color pipeline of a plane in its
current, new or old state. If not, reject the state change of this
inactive colorop. Performing this check later in drm_atomic_check_only()
to remove the ordering dependency that would exist if done at the time
of colorop property setting. Userspace is allowed to change colorops of
an active color pipeline, or when activating or deactivating its
pipeline in the same commit. However, changes in inactive color pipeline
is not allowed.

Suggested-by: Chaitanya Kumar Borah <[email protected]>
Signed-off-by: Melissa Wen <[email protected]>
---
  drivers/gpu/drm/drm_atomic.c | 59 ++++++++++++++++++++++++++++++++++++
  1 file changed, 59 insertions(+)

diff --git a/drivers/gpu/drm/drm_atomic.c b/drivers/gpu/drm/drm_atomic.c
index 4fb3a23e862a..a0549435954b 100644
--- a/drivers/gpu/drm/drm_atomic.c
+++ b/drivers/gpu/drm/drm_atomic.c
@@ -865,6 +865,54 @@ drm_atomic_add_pipeline_colorops(struct drm_atomic_commit 
*state,
        return 0;
  }
+/**
+ * drm_atomic_colorop_check - check new colorop state
+ * @new_colorop_state: new colorop state to check
+ *
+ * Ensure that the colorop in @new_colorop_state belongs to an active color
+ * pipeline, i.e. it's in the chain of colorops set to the color_pipeline
+ * property of current, old or new plane state.
+ *
+ * Returns: 0 on success, -EINVAL otherwise.
+ */
+static int drm_atomic_colorop_check(const struct drm_colorop_state 
*new_colorop_state)
+{
+       struct drm_atomic_commit *state = new_colorop_state->state;
+       struct drm_plane *plane = new_colorop_state->colorop->plane;
+       struct drm_plane_state *new_plane_state, *old_plane_state;
+       struct drm_colorop *colorop;
+
+       new_plane_state = drm_atomic_get_new_plane_state(state, plane);
+       old_plane_state = drm_atomic_get_old_plane_state(state, plane);
+
+       /* No changes in the plane state. Check current-committed plane state */
+       if (!new_plane_state) {
+               for (colorop = plane->state->color_pipeline; colorop; colorop = 
colorop->next)
+                       if (colorop == new_colorop_state->colorop)
+                               return 0;
+               return -EINVAL;
+       }
+
+       if (WARN_ON(!old_plane_state)) return -EINVAL;

return should be in a new line.

+
+       /* Check if the colorop is active in the new plane state */
+       for (colorop = new_plane_state->color_pipeline; colorop; colorop = 
colorop->next)
+               if (colorop == new_colorop_state->colorop)
+                       return 0;
+
+       /* Same color pipeline as new; no point walking old. Colorop isn't 
active */
+       if (new_plane_state->color_pipeline == old_plane_state->color_pipeline)
+               return -EINVAL;
+
+       /* Check if the colorop was active in the old plane state */
+       for (colorop = old_plane_state->color_pipeline; colorop; colorop = 
colorop->next)
+               if (colorop == new_colorop_state->colorop)
+                       return 0;
+
+       /* Colorop is not part of an active color pipeline. */
+       return -EINVAL;
+}
+
  static void drm_atomic_colorop_print_state(struct drm_printer *p,
                                           const struct drm_colorop_state 
*state)
  {
@@ -1714,6 +1762,8 @@ int drm_atomic_check_only(struct drm_atomic_commit *state)
        struct drm_plane *plane;
        struct drm_plane_state *old_plane_state;
        struct drm_plane_state *new_plane_state;
+       struct drm_colorop *colorop;
+       struct drm_colorop_state *new_colorop_state;
        struct drm_crtc *crtc;
        struct drm_crtc_state *old_crtc_state;
        struct drm_crtc_state *new_crtc_state;
@@ -1730,6 +1780,15 @@ int drm_atomic_check_only(struct drm_atomic_commit 
*state)
                        requested_crtc |= drm_crtc_mask(crtc);
        }
+ for_each_new_colorop_in_state(state, colorop, new_colorop_state, i) {
+               ret = drm_atomic_colorop_check(new_colorop_state);
+               if (ret) {
+                       drm_dbg_atomic(dev, "[COLOROP:%d:%d] is not part of an 
active color pipeline.\n",
+                                      colorop->base.id, colorop->type);
+                       return ret;
+               }
+       }
+
        for_each_oldnew_plane_in_state(state, plane, old_plane_state, 
new_plane_state, i) {
                ret = drm_atomic_plane_check(old_plane_state, new_plane_state);
                if (ret) {

Reply via email to