mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-15 13:15:57 +00:00
drm/plane: Make framebuffer refcounting the responsibility of setplane_internal callers
lock_all_ctx in setplane_internal may return -EINTR, and __setplane_internal could return -EDEADLK. Making more special cases for fb would make the code even harder to read, so the easiest solution is not taking over the fb refcount, and making callers responsible for dropping the ref. Bugzilla: https://bugs.freedesktop.org/show_bug.cgi?id=102707 Fixes: 13736ba3b38b ("drm/legacy: Convert setplane ioctl locking to interruptible.") Testcase: kms_atomic_interruptible Signed-off-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com> Link: https://patchwork.freedesktop.org/patch/msgid/20171220093545.613-2-maarten.lankhorst@linux.intel.com Reviewed-by: Daniel Vetter <daniel.vetter@ffwll.ch>
This commit is contained in:
parent
2c08cd7c20
commit
ce0769e0ea
@ -558,11 +558,10 @@ int drm_plane_check_pixel_format(const struct drm_plane *plane, u32 format)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* setplane_internal - setplane handler for internal callers
|
* __setplane_internal - setplane handler for internal callers
|
||||||
*
|
*
|
||||||
* Note that we assume an extra reference has already been taken on fb. If the
|
* This function will take a reference on the new fb for the plane
|
||||||
* update fails, this reference will be dropped before return; if it succeeds,
|
* on success.
|
||||||
* the previous framebuffer (if any) will be unreferenced instead.
|
|
||||||
*
|
*
|
||||||
* src_{x,y,w,h} are provided in 16.16 fixed point format
|
* src_{x,y,w,h} are provided in 16.16 fixed point format
|
||||||
*/
|
*/
|
||||||
@ -630,14 +629,12 @@ static int __setplane_internal(struct drm_plane *plane,
|
|||||||
if (!ret) {
|
if (!ret) {
|
||||||
plane->crtc = crtc;
|
plane->crtc = crtc;
|
||||||
plane->fb = fb;
|
plane->fb = fb;
|
||||||
fb = NULL;
|
drm_framebuffer_get(plane->fb);
|
||||||
} else {
|
} else {
|
||||||
plane->old_fb = NULL;
|
plane->old_fb = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
out:
|
out:
|
||||||
if (fb)
|
|
||||||
drm_framebuffer_put(fb);
|
|
||||||
if (plane->old_fb)
|
if (plane->old_fb)
|
||||||
drm_framebuffer_put(plane->old_fb);
|
drm_framebuffer_put(plane->old_fb);
|
||||||
plane->old_fb = NULL;
|
plane->old_fb = NULL;
|
||||||
@ -685,6 +682,7 @@ int drm_mode_setplane(struct drm_device *dev, void *data,
|
|||||||
struct drm_plane *plane;
|
struct drm_plane *plane;
|
||||||
struct drm_crtc *crtc = NULL;
|
struct drm_crtc *crtc = NULL;
|
||||||
struct drm_framebuffer *fb = NULL;
|
struct drm_framebuffer *fb = NULL;
|
||||||
|
int ret;
|
||||||
|
|
||||||
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
@ -717,15 +715,16 @@ int drm_mode_setplane(struct drm_device *dev, void *data,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
ret = setplane_internal(plane, crtc, fb,
|
||||||
* setplane_internal will take care of deref'ing either the old or new
|
plane_req->crtc_x, plane_req->crtc_y,
|
||||||
* framebuffer depending on success.
|
plane_req->crtc_w, plane_req->crtc_h,
|
||||||
*/
|
plane_req->src_x, plane_req->src_y,
|
||||||
return setplane_internal(plane, crtc, fb,
|
plane_req->src_w, plane_req->src_h);
|
||||||
plane_req->crtc_x, plane_req->crtc_y,
|
|
||||||
plane_req->crtc_w, plane_req->crtc_h,
|
if (fb)
|
||||||
plane_req->src_x, plane_req->src_y,
|
drm_framebuffer_put(fb);
|
||||||
plane_req->src_w, plane_req->src_h);
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int drm_mode_cursor_universal(struct drm_crtc *crtc,
|
static int drm_mode_cursor_universal(struct drm_crtc *crtc,
|
||||||
@ -788,13 +787,12 @@ static int drm_mode_cursor_universal(struct drm_crtc *crtc,
|
|||||||
src_h = fb->height << 16;
|
src_h = fb->height << 16;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* setplane_internal will take care of deref'ing either the old or new
|
|
||||||
* framebuffer depending on success.
|
|
||||||
*/
|
|
||||||
ret = __setplane_internal(crtc->cursor, crtc, fb,
|
ret = __setplane_internal(crtc->cursor, crtc, fb,
|
||||||
crtc_x, crtc_y, crtc_w, crtc_h,
|
crtc_x, crtc_y, crtc_w, crtc_h,
|
||||||
0, 0, src_w, src_h, ctx);
|
0, 0, src_w, src_h, ctx);
|
||||||
|
|
||||||
|
if (fb)
|
||||||
|
drm_framebuffer_put(fb);
|
||||||
|
|
||||||
/* Update successful; save new cursor position, if necessary */
|
/* Update successful; save new cursor position, if necessary */
|
||||||
if (ret == 0 && req->flags & DRM_MODE_CURSOR_MOVE) {
|
if (ret == 0 && req->flags & DRM_MODE_CURSOR_MOVE) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user