[PATCH 10/10] drm/imx: ipuv3-plane: use drm_plane_helper_check_state, clipped coordinates
Ying Liu
gnuiyl at gmail.com
Thu Oct 20 08:51:30 UTC 2016
On Thu, Oct 20, 2016 at 4:41 PM, Ville Syrjälä
<ville.syrjala at linux.intel.com> wrote:
> On Thu, Oct 20, 2016 at 03:53:46PM +0800, Ying Liu wrote:
>> On Wed, Oct 19, 2016 at 6:21 PM, Philipp Zabel <p.zabel at pengutronix.de> wrote:
>> > Use drm_plane_helper_check_state to clip raw user coordinates to crtc
>> > bounds. This checks for full plane coverage and scaling already, so
>> > we can drop some custom checks. Use the clipped coordinates everywhere.
>> >
>> > Suggested-by: Ville Syrjälä <ville.syrjala at linux.intel.com>
>> > Signed-off-by: Philipp Zabel <p.zabel at pengutronix.de>
>> > ---
>> > drivers/gpu/drm/imx/ipuv3-plane.c | 78 ++++++++++++++++++---------------------
>> > 1 file changed, 36 insertions(+), 42 deletions(-)
>> >
>> > diff --git a/drivers/gpu/drm/imx/ipuv3-plane.c b/drivers/gpu/drm/imx/ipuv3-plane.c
>> > index 6a97e396..f0ce899 100644
>> > --- a/drivers/gpu/drm/imx/ipuv3-plane.c
>> > +++ b/drivers/gpu/drm/imx/ipuv3-plane.c
>> > @@ -70,8 +70,8 @@ drm_plane_state_to_eba(struct drm_plane_state *state)
>> > {
>> > struct drm_framebuffer *fb = state->fb;
>> > struct drm_gem_cma_object *cma_obj;
>> > - int x = state->src_x >> 16;
>> > - int y = state->src_y >> 16;
>> > + int x = state->src.x1 >> 16;
>> > + int y = state->src.y1 >> 16;
>> >
>> > cma_obj = drm_fb_cma_get_gem_obj(fb, 0);
>> > BUG_ON(!cma_obj);
>> > @@ -86,8 +86,8 @@ drm_plane_state_to_ubo(struct drm_plane_state *state)
>> > struct drm_framebuffer *fb = state->fb;
>> > struct drm_gem_cma_object *cma_obj;
>> > unsigned long eba = drm_plane_state_to_eba(state);
>> > - int x = state->src_x >> 16;
>> > - int y = state->src_y >> 16;
>> > + int x = state->src.x1 >> 16;
>> > + int y = state->src.y1 >> 16;
>> >
>> > cma_obj = drm_fb_cma_get_gem_obj(fb, 1);
>> > BUG_ON(!cma_obj);
>> > @@ -105,8 +105,8 @@ drm_plane_state_to_vbo(struct drm_plane_state *state)
>> > struct drm_framebuffer *fb = state->fb;
>> > struct drm_gem_cma_object *cma_obj;
>> > unsigned long eba = drm_plane_state_to_eba(state);
>> > - int x = state->src_x >> 16;
>> > - int y = state->src_y >> 16;
>> > + int x = state->src.x1 >> 16;
>> > + int y = state->src.y1 >> 16;
>> >
>> > cma_obj = drm_fb_cma_get_gem_obj(fb, 2);
>> > BUG_ON(!cma_obj);
>> > @@ -218,7 +218,10 @@ static int ipu_plane_atomic_check(struct drm_plane *plane,
>> > struct drm_framebuffer *fb = state->fb;
>> > struct drm_framebuffer *old_fb = old_state->fb;
>> > unsigned long eba, ubo, vbo, old_ubo, old_vbo;
>> > + bool can_position = (plane->type == DRM_PLANE_TYPE_OVERLAY);
>> > + struct drm_rect clip;
>> > int hsub, vsub;
>> > + int ret;
>> >
>> > /* Ok to disable */
>> > if (!fb)
>> > @@ -232,44 +235,35 @@ static int ipu_plane_atomic_check(struct drm_plane *plane,
>> > if (WARN_ON(!crtc_state))
>> > return -EINVAL;
>> >
>> > + clip.x1 = 0;
>> > + clip.y1 = 0;
>> > + clip.x2 = crtc_state->adjusted_mode.hdisplay;
>> > + clip.y2 = crtc_state->adjusted_mode.vdisplay;
>> > + ret = drm_plane_helper_check_state(state, &clip,
>> > + DRM_PLANE_HELPER_NO_SCALING,
>> > + DRM_PLANE_HELPER_NO_SCALING,
>> > + can_position, true);
>> > + if (ret)
>> > + return ret;
>>
>> Does the clip thing potentially change the user's request by force?
>> For example, the user request an unreasonable big resolution.
>
> The user is allowed to ask for destination coordinates extending outside
> the crtc dimensions. This will chop off the parts that aren't visible,
> and it will chop off the corresponding areas of the source as well.
How about returning -EINVAL in this case which stands for
an atomic check failure?
>
>>
>> > +
>> > /* CRTC should be enabled */
>> > if (!crtc_state->enable)
>> > return -EINVAL;
>> >
>> > - /* no scaling */
>> > - if (state->src_w >> 16 != state->crtc_w ||
>> > - state->src_h >> 16 != state->crtc_h)
>> > - return -EINVAL;
>> > -
>> > switch (plane->type) {
>> > case DRM_PLANE_TYPE_PRIMARY:
>> > - /* full plane doesn't support partial off screen */
>> > - if (state->crtc_x || state->crtc_y ||
>> > - state->crtc_w != crtc_state->adjusted_mode.hdisplay ||
>> > - state->crtc_h != crtc_state->adjusted_mode.vdisplay)
>> > - return -EINVAL;
>> > -
>> > /* full plane minimum width is 13 pixels */
>> > - if (state->crtc_w < 13)
>> > + if (drm_rect_width(&state->dst) < 13)
>> > return -EINVAL;
>> > break;
>> > case DRM_PLANE_TYPE_OVERLAY:
>> > - if (state->crtc_x < 0 || state->crtc_y < 0)
>> > - return -EINVAL;
>>
>> How does drm_plane_helper_check_state() cover this check?
>>
>> > -
>> > - if (state->crtc_x + state->crtc_w >
>> > - crtc_state->adjusted_mode.hdisplay)
>> > - return -EINVAL;
>> > - if (state->crtc_y + state->crtc_h >
>> > - crtc_state->adjusted_mode.vdisplay)
>> > - return -EINVAL;
>>
>> And these?
>>
>> Regards,
>> Liu Ying
>>
>> > break;
>> > default:
>> > - dev_warn(dev, "Unsupported plane type\n");
>> > + dev_warn(dev, "Unsupported plane type %d\n", plane->type);
>> > return -EINVAL;
>> > }
>> >
>> > - if (state->crtc_h < 2)
>> > + if (drm_rect_height(&state->dst) < 2)
>> > return -EINVAL;
>> >
>> > /*
>> > @@ -279,9 +273,8 @@ static int ipu_plane_atomic_check(struct drm_plane *plane,
>> > * callback. The planes will be reenabled in plane's ->atomic_update
>> > * callback.
>> > */
>> > - if (old_fb && (state->src_w != old_state->src_w ||
>> > - state->src_h != old_state->src_h ||
>> > - fb->pixel_format != old_fb->pixel_format))
>> > + if (old_fb && (!drm_rect_equals(&state->dst, &old_state->dst) ||
>> > + fb->pixel_format != old_fb->pixel_format))
>> > crtc_state->mode_changed = true;
>> >
>> > eba = drm_plane_state_to_eba(state);
>> > @@ -350,8 +343,8 @@ static int ipu_plane_atomic_check(struct drm_plane *plane,
>> > */
>> > hsub = drm_format_horz_chroma_subsampling(fb->pixel_format);
>> > vsub = drm_format_vert_chroma_subsampling(fb->pixel_format);
>> > - if (((state->src_x >> 16) & (hsub - 1)) ||
>> > - ((state->src_y >> 16) & (vsub - 1)))
>> > + if (((state->src.x1 >> 16) & (hsub - 1)) ||
>> > + ((state->src.y1 >> 16) & (vsub - 1)))
>> > return -EINVAL;
>> > }
>> >
>> > @@ -395,8 +388,8 @@ static void ipu_plane_atomic_update(struct drm_plane *plane,
>> > ics = ipu_drm_fourcc_to_colorspace(state->fb->pixel_format);
>> > ipu_dp_setup_channel(ipu_plane->dp, ics,
>> > IPUV3_COLORSPACE_UNKNOWN);
>> > - ipu_dp_set_window_pos(ipu_plane->dp, state->crtc_x,
>> > - state->crtc_y);
>> > + ipu_dp_set_window_pos(ipu_plane->dp,
>> > + state->dst.x1, state->dst.y1);
>> > /* Enable local alpha on partial plane */
>> > switch (state->fb->pixel_format) {
>> > case DRM_FORMAT_ARGB1555:
>> > @@ -416,11 +409,12 @@ static void ipu_plane_atomic_update(struct drm_plane *plane,
>> > }
>> > }
>> >
>> > - ipu_dmfc_config_wait4eot(ipu_plane->dmfc, state->crtc_w);
>> > + ipu_dmfc_config_wait4eot(ipu_plane->dmfc, drm_rect_width(&state->dst));
>> >
>> > ipu_cpmem_zero(ipu_plane->ipu_ch);
>> > - ipu_cpmem_set_resolution(ipu_plane->ipu_ch, state->src_w >> 16,
>> > - state->src_h >> 16);
>> > + ipu_cpmem_set_resolution(ipu_plane->ipu_ch,
>> > + drm_rect_width(&state->src) >> 16,
>> > + drm_rect_height(&state->src) >> 16);
>> > ipu_cpmem_set_fmt(ipu_plane->ipu_ch, state->fb->pixel_format);
>> > ipu_cpmem_set_high_priority(ipu_plane->ipu_ch);
>> > ipu_idmac_set_double_buffer(ipu_plane->ipu_ch, 1);
>> > @@ -444,7 +438,7 @@ static void ipu_plane_atomic_update(struct drm_plane *plane,
>> >
>> > dev_dbg(ipu_plane->base.dev->dev,
>> > "phy = %lu %lu %lu, x = %d, y = %d", eba, ubo, vbo,
>> > - state->src_x >> 16, state->src_y >> 16);
>> > + state->src.x1 >> 16, state->src.y1 >> 16);
>> > break;
>> > case DRM_FORMAT_NV12:
>> > case DRM_FORMAT_NV16:
>> > @@ -455,11 +449,11 @@ static void ipu_plane_atomic_update(struct drm_plane *plane,
>> >
>> > dev_dbg(ipu_plane->base.dev->dev,
>> > "phy = %lu %lu, x = %d, y = %d", eba, ubo,
>> > - state->src_x >> 16, state->src_y >> 16);
>> > + state->src.x1 >> 16, state->src.y1 >> 16);
>> > break;
>> > default:
>> > dev_dbg(ipu_plane->base.dev->dev, "phys = %lu, x = %d, y = %d",
>> > - eba, state->src_x >> 16, state->src_y >> 16);
>> > + eba, state->src.x1 >> 16, state->src.y1 >> 16);
>> > break;
>> > }
>> > ipu_cpmem_set_buffer(ipu_plane->ipu_ch, 0, eba);
>> > --
>> > 2.9.3
>> >
>
> --
> Ville Syrjälä
> Intel OTC
More information about the dri-devel
mailing list