]> www.infradead.org Git - users/hch/dma-mapping.git/commitdiff
drm/amd/display: Correct cursor position on horizontal mirror
authorMartin Tsai <martin.tsai@amd.com>
Thu, 18 Aug 2022 08:01:24 +0000 (16:01 +0800)
committerAlex Deucher <alexander.deucher@amd.com>
Mon, 29 Aug 2022 21:59:30 +0000 (17:59 -0400)
[Why]
Incorrect cursor position will induce system hang on pipe split.

[How]
1.Handle horizontal mirror on rotation,
2.Correct cursor set on piep split.

Reviewed-by: Ariel Bernstein <Eric.Bernstein@amd.com>
Acked-by: Brian Chang <Brian.Chang@amd.com>
Signed-off-by: Martin Tsai <martin.tsai@amd.com>
Tested-by: Daniel Wheeler <daniel.wheeler@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
drivers/gpu/drm/amd/display/dc/dcn10/dcn10_dpp.c
drivers/gpu/drm/amd/display/dc/dcn10/dcn10_hubp.c
drivers/gpu/drm/amd/display/dc/dcn10/dcn10_hw_sequencer.c
drivers/gpu/drm/amd/display/dc/dcn20/dcn20_hubp.c

index db7ca4b0cdb9dd4165f9b8d1d53e9be2608ccc89..897f412f539e6502ac16e06dcbe3b1540212afef 100644 (file)
@@ -448,11 +448,12 @@ void dpp1_set_cursor_position(
                        src_y_offset = pos->y - pos->x_hotspot - param->viewport.y;
                }
        } else if (param->rotation == ROTATION_ANGLE_180) {
-               src_x_offset = pos->x - param->viewport.x;
+               if (!param->mirror)
+                       src_x_offset = pos->x - param->viewport.x;
+
                src_y_offset = pos->y - param->viewport.y;
        }
 
-
        if (src_x_offset >= (int)param->viewport.width)
                cur_en = 0;  /* not visible beyond right edge*/
 
index 564e061ccb589da01bf3e31bd8896521102712c1..52e201e9b091702f4b4b34539d307e546c9349d5 100644 (file)
@@ -1208,13 +1208,10 @@ void hubp1_cursor_set_position(
                        src_y_offset = pos->y - pos->x_hotspot - param->viewport.y;
                }
        } else if (param->rotation == ROTATION_ANGLE_180) {
-               src_x_offset = pos->x - param->viewport.x;
-               src_y_offset = pos->y - param->viewport.y;
-       }
+               if (!param->mirror)
+                       src_x_offset = pos->x - param->viewport.x;
 
-       if (param->mirror) {
-               x_hotspot = param->viewport.width - x_hotspot;
-               src_x_offset = param->viewport.x + param->viewport.width - src_x_offset;
+               src_y_offset = pos->y - param->viewport.y;
        }
 
        dst_x_offset = (src_x_offset >= 0) ? src_x_offset : 0;
index d3f6a2609c8c706bd922e73971fac09d41435c94..b92c14b9043fc32beac31e352a68fa9e86c1079e 100644 (file)
@@ -3470,8 +3470,7 @@ void dcn10_set_cursor_position(struct pipe_ctx *pipe_ctx)
                .rotation = pipe_ctx->plane_state->rotation,
                .mirror = pipe_ctx->plane_state->horizontal_mirror
        };
-       bool pipe_split_on = (pipe_ctx->top_pipe != NULL) ||
-               (pipe_ctx->bottom_pipe != NULL);
+       bool pipe_split_on = false;
        bool odm_combine_on = (pipe_ctx->next_odm_pipe != NULL) ||
                (pipe_ctx->prev_odm_pipe != NULL);
 
@@ -3480,6 +3479,13 @@ void dcn10_set_cursor_position(struct pipe_ctx *pipe_ctx)
        int x_pos = pos_cpy.x;
        int y_pos = pos_cpy.y;
 
+       if ((pipe_ctx->top_pipe != NULL) || (pipe_ctx->bottom_pipe != NULL)) {
+               if ((pipe_ctx->plane_state->src_rect.width != pipe_ctx->plane_res.scl_data.viewport.width) ||
+                       (pipe_ctx->plane_state->src_rect.height != pipe_ctx->plane_res.scl_data.viewport.height)) {
+                       pipe_split_on = true;
+               }
+       }
+
        /**
         * DC cursor is stream space, HW cursor is plane space and drawn
         * as part of the framebuffer.
@@ -3551,8 +3557,36 @@ void dcn10_set_cursor_position(struct pipe_ctx *pipe_ctx)
        if (pos_cpy.enable && dcn10_can_pipe_disable_cursor(pipe_ctx))
                pos_cpy.enable = false;
 
+
+       if (param.rotation == ROTATION_ANGLE_0) {
+               int viewport_width =
+                       pipe_ctx->plane_res.scl_data.viewport.width;
+               int viewport_x =
+                       pipe_ctx->plane_res.scl_data.viewport.x;
+
+               if (param.mirror) {
+                       if (pipe_split_on || odm_combine_on) {
+                               if (pos_cpy.x >= viewport_width + viewport_x) {
+                                       pos_cpy.x = 2 * viewport_width
+                                                       - pos_cpy.x + 2 * viewport_x;
+                               } else {
+                                       uint32_t temp_x = pos_cpy.x;
+
+                                       pos_cpy.x = 2 * viewport_x - pos_cpy.x;
+                                       if (temp_x >= viewport_x +
+                                               (int)hubp->curs_attr.width || pos_cpy.x
+                                               <= (int)hubp->curs_attr.width +
+                                               pipe_ctx->plane_state->src_rect.x) {
+                                               pos_cpy.x = temp_x + viewport_width;
+                                       }
+                               }
+                       } else {
+                               pos_cpy.x = viewport_width - pos_cpy.x + 2 * viewport_x;
+                       }
+               }
+       }
        // Swap axis and mirror horizontally
-       if (param.rotation == ROTATION_ANGLE_90) {
+       else if (param.rotation == ROTATION_ANGLE_90) {
                uint32_t temp_x = pos_cpy.x;
 
                pos_cpy.x = pipe_ctx->plane_res.scl_data.viewport.width -
@@ -3623,23 +3657,25 @@ void dcn10_set_cursor_position(struct pipe_ctx *pipe_ctx)
                int viewport_x =
                        pipe_ctx->plane_res.scl_data.viewport.x;
 
-               if (pipe_split_on || odm_combine_on) {
-                       if (pos_cpy.x >= viewport_width + viewport_x) {
-                               pos_cpy.x = 2 * viewport_width
-                                               - pos_cpy.x + 2 * viewport_x;
-                       } else {
-                               uint32_t temp_x = pos_cpy.x;
-
-                               pos_cpy.x = 2 * viewport_x - pos_cpy.x;
-                               if (temp_x >= viewport_x +
-                                       (int)hubp->curs_attr.width || pos_cpy.x
-                                       <= (int)hubp->curs_attr.width +
-                                       pipe_ctx->plane_state->src_rect.x) {
-                                       pos_cpy.x = temp_x + viewport_width;
+               if (!param.mirror) {
+                       if (pipe_split_on || odm_combine_on) {
+                               if (pos_cpy.x >= viewport_width + viewport_x) {
+                                       pos_cpy.x = 2 * viewport_width
+                                                       - pos_cpy.x + 2 * viewport_x;
+                               } else {
+                                       uint32_t temp_x = pos_cpy.x;
+
+                                       pos_cpy.x = 2 * viewport_x - pos_cpy.x;
+                                       if (temp_x >= viewport_x +
+                                               (int)hubp->curs_attr.width || pos_cpy.x
+                                               <= (int)hubp->curs_attr.width +
+                                               pipe_ctx->plane_state->src_rect.x) {
+                                               pos_cpy.x = temp_x + viewport_width;
+                                       }
                                }
+                       } else {
+                               pos_cpy.x = viewport_width - pos_cpy.x + 2 * viewport_x;
                        }
-               } else {
-                       pos_cpy.x = viewport_width - pos_cpy.x + 2 * viewport_x;
                }
 
                /**
index 9570c2118ccc73ae4ce3ffc32f7064c31cfa49a1..b1ec0e6f7f5877948c6b562cccea1ef65850245e 100644 (file)
@@ -987,13 +987,10 @@ void hubp2_cursor_set_position(
                        src_y_offset = pos->y - pos->x_hotspot - param->viewport.y;
                }
        } else if (param->rotation == ROTATION_ANGLE_180) {
-               src_x_offset = pos->x - param->viewport.x;
-               src_y_offset = pos->y - param->viewport.y;
-       }
+               if (!param->mirror)
+                       src_x_offset = pos->x - param->viewport.x;
 
-       if (param->mirror) {
-               x_hotspot = param->viewport.width - x_hotspot;
-               src_x_offset = param->viewport.x + param->viewport.width - src_x_offset;
+               src_y_offset = pos->y - param->viewport.y;
        }
 
        dst_x_offset = (src_x_offset >= 0) ? src_x_offset : 0;