Commit 4d02e2de authored by Daniel Vetter's avatar Daniel Vetter Committed by Dave Airlie

drm: Per-plane locking

Turned out to be much simpler on top of my latest atomic stuff than
what I've feared. Some details:

- Drop the modeset_lock_all snakeoil in drm_plane_init. Same
  justification as for the equivalent change in drm_crtc_init done in

	commit d0fa1af4
	Author: Daniel Vetter <daniel.vetter@ffwll.ch>
	Date:   Mon Sep 8 09:02:49 2014 +0200

	    drm: Drop modeset locking from crtc init function

  Without these the drm_modeset_lock_init would fall over the exact
  same way.

- Since the atomic core code wraps the locking switching it to
  per-plane locks was a one-line change.

- For the legacy ioctls add a plane argument to the locking helper so
  that we can grab the right plane lock (cursor or primary). Since the
  universal cursor plane might not be there, or someone really crazy
  might forgoe the primary plane even accept NULL.

- Add some locking WARN_ON to the atomic helpers for good paranoid
  measure and to check that it all works out.

Tested on my exynos atomic hackfest with full lockdep checks and ww
backoff injection.

v2: I've forgotten about the load-detect code in i915.

v3: Thierry reported that in latest 3.18-rc vmwgfx doesn't compile any
more due to

commit 21e88620
Author: Rob Clark <robdclark@gmail.com>
Date:   Thu Oct 30 13:39:04 2014 -0400

    drm/vmwgfx: fix lock breakage

Rebased and fix this up.

Cc: Thierry Reding <thierry.reding@gmail.com>
Signed-off-by: default avatarDaniel Vetter <daniel.vetter@intel.com>
Reviewed-by: default avatarSean Paul <seanpaul@chromium.org>
Signed-off-by: default avatarDave Airlie <airlied@redhat.com>
parent 5ee3229c
...@@ -244,7 +244,7 @@ drm_atomic_get_plane_state(struct drm_atomic_state *state, ...@@ -244,7 +244,7 @@ drm_atomic_get_plane_state(struct drm_atomic_state *state,
* grab all crtc locks. Once we have per-plane locks we must update this * grab all crtc locks. Once we have per-plane locks we must update this
* to only take the plane mutex. * to only take the plane mutex.
*/ */
ret = drm_modeset_lock_all_crtcs(state->dev, state->acquire_ctx); ret = drm_modeset_lock(&plane->mutex, state->acquire_ctx);
if (ret) if (ret)
return ERR_PTR(ret); return ERR_PTR(ret);
......
...@@ -1006,6 +1006,8 @@ void drm_atomic_helper_commit_planes(struct drm_device *dev, ...@@ -1006,6 +1006,8 @@ void drm_atomic_helper_commit_planes(struct drm_device *dev,
if (!crtc) if (!crtc)
continue; continue;
WARN_ON(!drm_modeset_is_locked(&crtc->mutex));
funcs = crtc->helper_private; funcs = crtc->helper_private;
if (!funcs || !funcs->atomic_begin) if (!funcs || !funcs->atomic_begin)
...@@ -1021,6 +1023,8 @@ void drm_atomic_helper_commit_planes(struct drm_device *dev, ...@@ -1021,6 +1023,8 @@ void drm_atomic_helper_commit_planes(struct drm_device *dev,
if (!plane) if (!plane)
continue; continue;
WARN_ON(!drm_modeset_is_locked(&plane->mutex));
funcs = plane->helper_private; funcs = plane->helper_private;
if (!funcs || !funcs->atomic_update) if (!funcs || !funcs->atomic_update)
......
...@@ -1152,12 +1152,12 @@ int drm_universal_plane_init(struct drm_device *dev, struct drm_plane *plane, ...@@ -1152,12 +1152,12 @@ int drm_universal_plane_init(struct drm_device *dev, struct drm_plane *plane,
{ {
int ret; int ret;
drm_modeset_lock_all(dev);
ret = drm_mode_object_get(dev, &plane->base, DRM_MODE_OBJECT_PLANE); ret = drm_mode_object_get(dev, &plane->base, DRM_MODE_OBJECT_PLANE);
if (ret) if (ret)
goto out; goto out;
drm_modeset_lock_init(&plane->mutex);
plane->base.properties = &plane->properties; plane->base.properties = &plane->properties;
plane->dev = dev; plane->dev = dev;
plane->funcs = funcs; plane->funcs = funcs;
...@@ -1185,7 +1185,6 @@ int drm_universal_plane_init(struct drm_device *dev, struct drm_plane *plane, ...@@ -1185,7 +1185,6 @@ int drm_universal_plane_init(struct drm_device *dev, struct drm_plane *plane,
plane->type); plane->type);
out: out:
drm_modeset_unlock_all(dev);
return ret; return ret;
} }
...@@ -2809,7 +2808,7 @@ static int drm_mode_cursor_common(struct drm_device *dev, ...@@ -2809,7 +2808,7 @@ static int drm_mode_cursor_common(struct drm_device *dev,
* If this crtc has a universal cursor plane, call that plane's update * If this crtc has a universal cursor plane, call that plane's update
* handler rather than using legacy cursor handlers. * handler rather than using legacy cursor handlers.
*/ */
drm_modeset_lock_crtc(crtc); drm_modeset_lock_crtc(crtc, crtc->cursor);
if (crtc->cursor) { if (crtc->cursor) {
ret = drm_mode_cursor_universal(crtc, req, file_priv); ret = drm_mode_cursor_universal(crtc, req, file_priv);
goto out; goto out;
...@@ -4598,7 +4597,7 @@ int drm_mode_page_flip_ioctl(struct drm_device *dev, ...@@ -4598,7 +4597,7 @@ int drm_mode_page_flip_ioctl(struct drm_device *dev,
if (!crtc) if (!crtc)
return -ENOENT; return -ENOENT;
drm_modeset_lock_crtc(crtc); drm_modeset_lock_crtc(crtc, crtc->primary);
if (crtc->primary->fb == NULL) { if (crtc->primary->fb == NULL) {
/* The framebuffer is currently unbound, presumably /* The framebuffer is currently unbound, presumably
* due to a hotplug event, that userspace has not * due to a hotplug event, that userspace has not
......
...@@ -157,14 +157,20 @@ void drm_modeset_unlock_all(struct drm_device *dev) ...@@ -157,14 +157,20 @@ void drm_modeset_unlock_all(struct drm_device *dev)
EXPORT_SYMBOL(drm_modeset_unlock_all); EXPORT_SYMBOL(drm_modeset_unlock_all);
/** /**
* drm_modeset_lock_crtc - lock crtc with hidden acquire ctx * drm_modeset_lock_crtc - lock crtc with hidden acquire ctx for a plane update
* @crtc: drm crtc * @crtc: DRM CRTC
* @plane: DRM plane to be updated on @crtc
*
* This function locks the given crtc and plane (which should be either the
* primary or cursor plane) using a hidden acquire context. This is necessary so
* that drivers internally using the atomic interfaces can grab further locks
* with the lock acquire context.
* *
* This function locks the given crtc using a hidden acquire context. This is * Note that @plane can be NULL, e.g. when the cursor support hasn't yet been
* necessary so that drivers internally using the atomic interfaces can grab * converted to universal planes yet.
* further locks with the lock acquire context.
*/ */
void drm_modeset_lock_crtc(struct drm_crtc *crtc) void drm_modeset_lock_crtc(struct drm_crtc *crtc,
struct drm_plane *plane)
{ {
struct drm_modeset_acquire_ctx *ctx; struct drm_modeset_acquire_ctx *ctx;
int ret; int ret;
...@@ -180,6 +186,18 @@ void drm_modeset_lock_crtc(struct drm_crtc *crtc) ...@@ -180,6 +186,18 @@ void drm_modeset_lock_crtc(struct drm_crtc *crtc)
if (ret) if (ret)
goto fail; goto fail;
if (plane) {
ret = drm_modeset_lock(&plane->mutex, ctx);
if (ret)
goto fail;
if (plane->crtc) {
ret = drm_modeset_lock(&plane->crtc->mutex, ctx);
if (ret)
goto fail;
}
}
WARN_ON(crtc->acquire_ctx); WARN_ON(crtc->acquire_ctx);
/* now we hold the locks, so now that it is safe, stash the /* now we hold the locks, so now that it is safe, stash the
...@@ -437,15 +455,14 @@ void drm_modeset_unlock(struct drm_modeset_lock *lock) ...@@ -437,15 +455,14 @@ void drm_modeset_unlock(struct drm_modeset_lock *lock)
} }
EXPORT_SYMBOL(drm_modeset_unlock); EXPORT_SYMBOL(drm_modeset_unlock);
/* Temporary.. until we have sufficiently fine grained locking, there /* In some legacy codepaths it's convenient to just grab all the crtc and plane
* are a couple scenarios where it is convenient to grab all crtc locks. * related locks. */
* It is planned to remove this:
*/
int drm_modeset_lock_all_crtcs(struct drm_device *dev, int drm_modeset_lock_all_crtcs(struct drm_device *dev,
struct drm_modeset_acquire_ctx *ctx) struct drm_modeset_acquire_ctx *ctx)
{ {
struct drm_mode_config *config = &dev->mode_config; struct drm_mode_config *config = &dev->mode_config;
struct drm_crtc *crtc; struct drm_crtc *crtc;
struct drm_plane *plane;
int ret = 0; int ret = 0;
list_for_each_entry(crtc, &config->crtc_list, head) { list_for_each_entry(crtc, &config->crtc_list, head) {
...@@ -454,6 +471,12 @@ int drm_modeset_lock_all_crtcs(struct drm_device *dev, ...@@ -454,6 +471,12 @@ int drm_modeset_lock_all_crtcs(struct drm_device *dev,
return ret; return ret;
} }
list_for_each_entry(plane, &config->plane_list, head) {
ret = drm_modeset_lock(&plane->mutex, ctx);
if (ret)
return ret;
}
return 0; return 0;
} }
EXPORT_SYMBOL(drm_modeset_lock_all_crtcs); EXPORT_SYMBOL(drm_modeset_lock_all_crtcs);
...@@ -8728,6 +8728,9 @@ bool intel_get_load_detect_pipe(struct drm_connector *connector, ...@@ -8728,6 +8728,9 @@ bool intel_get_load_detect_pipe(struct drm_connector *connector,
crtc = encoder->crtc; crtc = encoder->crtc;
ret = drm_modeset_lock(&crtc->mutex, ctx); ret = drm_modeset_lock(&crtc->mutex, ctx);
if (ret)
goto fail_unlock;
ret = drm_modeset_lock(&crtc->primary->mutex, ctx);
if (ret) if (ret)
goto fail_unlock; goto fail_unlock;
...@@ -8765,6 +8768,9 @@ bool intel_get_load_detect_pipe(struct drm_connector *connector, ...@@ -8765,6 +8768,9 @@ bool intel_get_load_detect_pipe(struct drm_connector *connector,
} }
ret = drm_modeset_lock(&crtc->mutex, ctx); ret = drm_modeset_lock(&crtc->mutex, ctx);
if (ret)
goto fail_unlock;
ret = drm_modeset_lock(&crtc->primary->mutex, ctx);
if (ret) if (ret)
goto fail_unlock; goto fail_unlock;
intel_encoder->new_crtc = to_intel_crtc(crtc); intel_encoder->new_crtc = to_intel_crtc(crtc);
......
...@@ -252,7 +252,7 @@ int vmw_du_crtc_cursor_set(struct drm_crtc *crtc, struct drm_file *file_priv, ...@@ -252,7 +252,7 @@ int vmw_du_crtc_cursor_set(struct drm_crtc *crtc, struct drm_file *file_priv,
ret = 0; ret = 0;
out: out:
drm_modeset_unlock_all(dev_priv->dev); drm_modeset_unlock_all(dev_priv->dev);
drm_modeset_lock_crtc(crtc); drm_modeset_lock_crtc(crtc, crtc->cursor);
return ret; return ret;
} }
...@@ -281,7 +281,7 @@ int vmw_du_crtc_cursor_move(struct drm_crtc *crtc, int x, int y) ...@@ -281,7 +281,7 @@ int vmw_du_crtc_cursor_move(struct drm_crtc *crtc, int x, int y)
du->cursor_y + du->hotspot_y); du->cursor_y + du->hotspot_y);
drm_modeset_unlock_all(dev_priv->dev); drm_modeset_unlock_all(dev_priv->dev);
drm_modeset_lock_crtc(crtc); drm_modeset_lock_crtc(crtc, crtc->cursor);
return 0; return 0;
} }
......
...@@ -751,6 +751,8 @@ struct drm_plane { ...@@ -751,6 +751,8 @@ struct drm_plane {
struct drm_device *dev; struct drm_device *dev;
struct list_head head; struct list_head head;
struct drm_modeset_lock mutex;
struct drm_mode_object base; struct drm_mode_object base;
uint32_t possible_crtcs; uint32_t possible_crtcs;
......
...@@ -127,11 +127,13 @@ void drm_modeset_unlock(struct drm_modeset_lock *lock); ...@@ -127,11 +127,13 @@ void drm_modeset_unlock(struct drm_modeset_lock *lock);
struct drm_device; struct drm_device;
struct drm_crtc; struct drm_crtc;
struct drm_plane;
void drm_modeset_lock_all(struct drm_device *dev); void drm_modeset_lock_all(struct drm_device *dev);
int __drm_modeset_lock_all(struct drm_device *dev, bool trylock); int __drm_modeset_lock_all(struct drm_device *dev, bool trylock);
void drm_modeset_unlock_all(struct drm_device *dev); void drm_modeset_unlock_all(struct drm_device *dev);
void drm_modeset_lock_crtc(struct drm_crtc *crtc); void drm_modeset_lock_crtc(struct drm_crtc *crtc,
struct drm_plane *plane);
void drm_modeset_unlock_crtc(struct drm_crtc *crtc); void drm_modeset_unlock_crtc(struct drm_crtc *crtc);
void drm_warn_on_modeset_not_all_locked(struct drm_device *dev); void drm_warn_on_modeset_not_all_locked(struct drm_device *dev);
struct drm_modeset_acquire_ctx * struct drm_modeset_acquire_ctx *
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment