|
@@ -37,6 +37,40 @@
|
|
|
#include <drm/drm_edid.h>
|
|
|
#include <drm/drm_fourcc.h>
|
|
|
|
|
|
+/**
|
|
|
+ * drm_modeset_lock_all - take all modeset locks
|
|
|
+ * @dev: drm device
|
|
|
+ *
|
|
|
+ * This function takes all modeset locks, suitable where a more fine-grained
|
|
|
+ * scheme isn't (yet) implemented.
|
|
|
+ */
|
|
|
+void drm_modeset_lock_all(struct drm_device *dev)
|
|
|
+{
|
|
|
+ struct drm_crtc *crtc;
|
|
|
+
|
|
|
+ mutex_lock(&dev->mode_config.mutex);
|
|
|
+
|
|
|
+ list_for_each_entry(crtc, &dev->mode_config.crtc_list, head)
|
|
|
+ mutex_lock_nest_lock(&crtc->mutex, &dev->mode_config.mutex);
|
|
|
+}
|
|
|
+EXPORT_SYMBOL(drm_modeset_lock_all);
|
|
|
+
|
|
|
+/**
|
|
|
+ * drm_modeset_unlock_all - drop all modeset locks
|
|
|
+ * @dev: device
|
|
|
+ */
|
|
|
+void drm_modeset_unlock_all(struct drm_device *dev)
|
|
|
+{
|
|
|
+ struct drm_crtc *crtc;
|
|
|
+
|
|
|
+ list_for_each_entry(crtc, &dev->mode_config.crtc_list, head)
|
|
|
+ mutex_unlock(&crtc->mutex);
|
|
|
+
|
|
|
+ mutex_unlock(&dev->mode_config.mutex);
|
|
|
+}
|
|
|
+
|
|
|
+EXPORT_SYMBOL(drm_modeset_unlock_all);
|
|
|
+
|
|
|
/* Avoid boilerplate. I'm tired of typing. */
|
|
|
#define DRM_ENUM_NAME_FN(fnname, list) \
|
|
|
char *fnname(int val) \
|
|
@@ -203,12 +237,10 @@ char *drm_get_connector_status_name(enum drm_connector_status status)
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * drm_mode_object_get - allocate a new identifier
|
|
|
+ * drm_mode_object_get - allocate a new modeset identifier
|
|
|
* @dev: DRM device
|
|
|
- * @ptr: object pointer, used to generate unique ID
|
|
|
- * @type: object type
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
+ * @obj: object pointer, used to generate unique ID
|
|
|
+ * @obj_type: object type
|
|
|
*
|
|
|
* Create a unique identifier based on @ptr in @dev's identifier space. Used
|
|
|
* for tracking modes, CRTCs and connectors.
|
|
@@ -231,24 +263,27 @@ again:
|
|
|
|
|
|
mutex_lock(&dev->mode_config.idr_mutex);
|
|
|
ret = idr_get_new_above(&dev->mode_config.crtc_idr, obj, 1, &new_id);
|
|
|
+
|
|
|
+ if (!ret) {
|
|
|
+ /*
|
|
|
+ * Set up the object linking under the protection of the idr
|
|
|
+ * lock so that other users can't see inconsistent state.
|
|
|
+ */
|
|
|
+ obj->id = new_id;
|
|
|
+ obj->type = obj_type;
|
|
|
+ }
|
|
|
mutex_unlock(&dev->mode_config.idr_mutex);
|
|
|
+
|
|
|
if (ret == -EAGAIN)
|
|
|
goto again;
|
|
|
- else if (ret)
|
|
|
- return ret;
|
|
|
|
|
|
- obj->id = new_id;
|
|
|
- obj->type = obj_type;
|
|
|
- return 0;
|
|
|
+ return ret;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * drm_mode_object_put - free an identifer
|
|
|
+ * drm_mode_object_put - free a modeset identifer
|
|
|
* @dev: DRM device
|
|
|
- * @id: ID to free
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold DRM mode_config lock.
|
|
|
+ * @object: object to free
|
|
|
*
|
|
|
* Free @id from @dev's unique identifier pool.
|
|
|
*/
|
|
@@ -260,11 +295,24 @@ static void drm_mode_object_put(struct drm_device *dev,
|
|
|
mutex_unlock(&dev->mode_config.idr_mutex);
|
|
|
}
|
|
|
|
|
|
+/**
|
|
|
+ * drm_mode_object_find - look up a drm object with static lifetime
|
|
|
+ * @dev: drm device
|
|
|
+ * @id: id of the mode object
|
|
|
+ * @type: type of the mode object
|
|
|
+ *
|
|
|
+ * Note that framebuffers cannot be looked up with this functions - since those
|
|
|
+ * are reference counted, they need special treatment.
|
|
|
+ */
|
|
|
struct drm_mode_object *drm_mode_object_find(struct drm_device *dev,
|
|
|
uint32_t id, uint32_t type)
|
|
|
{
|
|
|
struct drm_mode_object *obj = NULL;
|
|
|
|
|
|
+ /* Framebuffers are reference counted and need their own lookup
|
|
|
+ * function.*/
|
|
|
+ WARN_ON(type == DRM_MODE_OBJECT_FB);
|
|
|
+
|
|
|
mutex_lock(&dev->mode_config.idr_mutex);
|
|
|
obj = idr_find(&dev->mode_config.crtc_idr, id);
|
|
|
if (!obj || (obj->type != type) || (obj->id != id))
|
|
@@ -278,13 +326,18 @@ EXPORT_SYMBOL(drm_mode_object_find);
|
|
|
/**
|
|
|
* drm_framebuffer_init - initialize a framebuffer
|
|
|
* @dev: DRM device
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold mode config lock.
|
|
|
+ * @fb: framebuffer to be initialized
|
|
|
+ * @funcs: ... with these functions
|
|
|
*
|
|
|
* Allocates an ID for the framebuffer's parent mode object, sets its mode
|
|
|
* functions & device file and adds it to the master fd list.
|
|
|
*
|
|
|
+ * IMPORTANT:
|
|
|
+ * This functions publishes the fb and makes it available for concurrent access
|
|
|
+ * by other users. Which means by this point the fb _must_ be fully set up -
|
|
|
+ * since all the fb attributes are invariant over its lifetime, no further
|
|
|
+ * locking but only correct reference counting is required.
|
|
|
+ *
|
|
|
* RETURNS:
|
|
|
* Zero on success, error code on failure.
|
|
|
*/
|
|
@@ -293,16 +346,23 @@ int drm_framebuffer_init(struct drm_device *dev, struct drm_framebuffer *fb,
|
|
|
{
|
|
|
int ret;
|
|
|
|
|
|
+ mutex_lock(&dev->mode_config.fb_lock);
|
|
|
kref_init(&fb->refcount);
|
|
|
+ INIT_LIST_HEAD(&fb->filp_head);
|
|
|
+ fb->dev = dev;
|
|
|
+ fb->funcs = funcs;
|
|
|
|
|
|
ret = drm_mode_object_get(dev, &fb->base, DRM_MODE_OBJECT_FB);
|
|
|
if (ret)
|
|
|
- return ret;
|
|
|
+ goto out;
|
|
|
+
|
|
|
+ /* Grab the idr reference. */
|
|
|
+ drm_framebuffer_reference(fb);
|
|
|
|
|
|
- fb->dev = dev;
|
|
|
- fb->funcs = funcs;
|
|
|
dev->mode_config.num_fb++;
|
|
|
list_add(&fb->head, &dev->mode_config.fb_list);
|
|
|
+out:
|
|
|
+ mutex_unlock(&dev->mode_config.fb_lock);
|
|
|
|
|
|
return 0;
|
|
|
}
|
|
@@ -315,23 +375,63 @@ static void drm_framebuffer_free(struct kref *kref)
|
|
|
fb->funcs->destroy(fb);
|
|
|
}
|
|
|
|
|
|
+static struct drm_framebuffer *__drm_framebuffer_lookup(struct drm_device *dev,
|
|
|
+ uint32_t id)
|
|
|
+{
|
|
|
+ struct drm_mode_object *obj = NULL;
|
|
|
+ struct drm_framebuffer *fb;
|
|
|
+
|
|
|
+ mutex_lock(&dev->mode_config.idr_mutex);
|
|
|
+ obj = idr_find(&dev->mode_config.crtc_idr, id);
|
|
|
+ if (!obj || (obj->type != DRM_MODE_OBJECT_FB) || (obj->id != id))
|
|
|
+ fb = NULL;
|
|
|
+ else
|
|
|
+ fb = obj_to_fb(obj);
|
|
|
+ mutex_unlock(&dev->mode_config.idr_mutex);
|
|
|
+
|
|
|
+ return fb;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * drm_framebuffer_lookup - look up a drm framebuffer and grab a reference
|
|
|
+ * @dev: drm device
|
|
|
+ * @id: id of the fb object
|
|
|
+ *
|
|
|
+ * If successful, this grabs an additional reference to the framebuffer -
|
|
|
+ * callers need to make sure to eventually unreference the returned framebuffer
|
|
|
+ * again.
|
|
|
+ */
|
|
|
+struct drm_framebuffer *drm_framebuffer_lookup(struct drm_device *dev,
|
|
|
+ uint32_t id)
|
|
|
+{
|
|
|
+ struct drm_framebuffer *fb;
|
|
|
+
|
|
|
+ mutex_lock(&dev->mode_config.fb_lock);
|
|
|
+ fb = __drm_framebuffer_lookup(dev, id);
|
|
|
+ if (fb)
|
|
|
+ kref_get(&fb->refcount);
|
|
|
+ mutex_unlock(&dev->mode_config.fb_lock);
|
|
|
+
|
|
|
+ return fb;
|
|
|
+}
|
|
|
+EXPORT_SYMBOL(drm_framebuffer_lookup);
|
|
|
+
|
|
|
/**
|
|
|
* drm_framebuffer_unreference - unref a framebuffer
|
|
|
+ * @fb: framebuffer to unref
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold mode config lock.
|
|
|
+ * This functions decrements the fb's refcount and frees it if it drops to zero.
|
|
|
*/
|
|
|
void drm_framebuffer_unreference(struct drm_framebuffer *fb)
|
|
|
{
|
|
|
- struct drm_device *dev = fb->dev;
|
|
|
DRM_DEBUG("FB ID: %d\n", fb->base.id);
|
|
|
- WARN_ON(!mutex_is_locked(&dev->mode_config.mutex));
|
|
|
kref_put(&fb->refcount, drm_framebuffer_free);
|
|
|
}
|
|
|
EXPORT_SYMBOL(drm_framebuffer_unreference);
|
|
|
|
|
|
/**
|
|
|
* drm_framebuffer_reference - incr the fb refcnt
|
|
|
+ * @fb: framebuffer
|
|
|
*/
|
|
|
void drm_framebuffer_reference(struct drm_framebuffer *fb)
|
|
|
{
|
|
@@ -340,29 +440,74 @@ void drm_framebuffer_reference(struct drm_framebuffer *fb)
|
|
|
}
|
|
|
EXPORT_SYMBOL(drm_framebuffer_reference);
|
|
|
|
|
|
+static void drm_framebuffer_free_bug(struct kref *kref)
|
|
|
+{
|
|
|
+ BUG();
|
|
|
+}
|
|
|
+
|
|
|
+static void __drm_framebuffer_unreference(struct drm_framebuffer *fb)
|
|
|
+{
|
|
|
+ DRM_DEBUG("FB ID: %d\n", fb->base.id);
|
|
|
+ kref_put(&fb->refcount, drm_framebuffer_free_bug);
|
|
|
+}
|
|
|
+
|
|
|
+/* dev->mode_config.fb_lock must be held! */
|
|
|
+static void __drm_framebuffer_unregister(struct drm_device *dev,
|
|
|
+ struct drm_framebuffer *fb)
|
|
|
+{
|
|
|
+ mutex_lock(&dev->mode_config.idr_mutex);
|
|
|
+ idr_remove(&dev->mode_config.crtc_idr, fb->base.id);
|
|
|
+ mutex_unlock(&dev->mode_config.idr_mutex);
|
|
|
+
|
|
|
+ fb->base.id = 0;
|
|
|
+
|
|
|
+ __drm_framebuffer_unreference(fb);
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * drm_framebuffer_unregister_private - unregister a private fb from the lookup idr
|
|
|
+ * @fb: fb to unregister
|
|
|
+ *
|
|
|
+ * Drivers need to call this when cleaning up driver-private framebuffers, e.g.
|
|
|
+ * those used for fbdev. Note that the caller must hold a reference of it's own,
|
|
|
+ * i.e. the object may not be destroyed through this call (since it'll lead to a
|
|
|
+ * locking inversion).
|
|
|
+ */
|
|
|
+void drm_framebuffer_unregister_private(struct drm_framebuffer *fb)
|
|
|
+{
|
|
|
+ struct drm_device *dev = fb->dev;
|
|
|
+
|
|
|
+ mutex_lock(&dev->mode_config.fb_lock);
|
|
|
+ /* Mark fb as reaped and drop idr ref. */
|
|
|
+ __drm_framebuffer_unregister(dev, fb);
|
|
|
+ mutex_unlock(&dev->mode_config.fb_lock);
|
|
|
+}
|
|
|
+EXPORT_SYMBOL(drm_framebuffer_unregister_private);
|
|
|
+
|
|
|
/**
|
|
|
* drm_framebuffer_cleanup - remove a framebuffer object
|
|
|
* @fb: framebuffer to remove
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold mode config lock.
|
|
|
+ * Cleanup references to a user-created framebuffer. This function is intended
|
|
|
+ * to be used from the drivers ->destroy callback.
|
|
|
+ *
|
|
|
+ * Note that this function does not remove the fb from active usuage - if it is
|
|
|
+ * still used anywhere, hilarity can ensue since userspace could call getfb on
|
|
|
+ * the id and get back -EINVAL. Obviously no concern at driver unload time.
|
|
|
*
|
|
|
- * Scans all the CRTCs in @dev's mode_config. If they're using @fb, removes
|
|
|
- * it, setting it to NULL.
|
|
|
+ * Also, the framebuffer will not be removed from the lookup idr - for
|
|
|
+ * user-created framebuffers this will happen in in the rmfb ioctl. For
|
|
|
+ * driver-private objects (e.g. for fbdev) drivers need to explicitly call
|
|
|
+ * drm_framebuffer_unregister_private.
|
|
|
*/
|
|
|
void drm_framebuffer_cleanup(struct drm_framebuffer *fb)
|
|
|
{
|
|
|
struct drm_device *dev = fb->dev;
|
|
|
- /*
|
|
|
- * This could be moved to drm_framebuffer_remove(), but for
|
|
|
- * debugging is nice to keep around the list of fb's that are
|
|
|
- * no longer associated w/ a drm_file but are not unreferenced
|
|
|
- * yet. (i915 and omapdrm have debugfs files which will show
|
|
|
- * this.)
|
|
|
- */
|
|
|
- drm_mode_object_put(dev, &fb->base);
|
|
|
+
|
|
|
+ mutex_lock(&dev->mode_config.fb_lock);
|
|
|
list_del(&fb->head);
|
|
|
dev->mode_config.num_fb--;
|
|
|
+ mutex_unlock(&dev->mode_config.fb_lock);
|
|
|
}
|
|
|
EXPORT_SYMBOL(drm_framebuffer_cleanup);
|
|
|
|
|
@@ -370,11 +515,13 @@ EXPORT_SYMBOL(drm_framebuffer_cleanup);
|
|
|
* drm_framebuffer_remove - remove and unreference a framebuffer object
|
|
|
* @fb: framebuffer to remove
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold mode config lock.
|
|
|
- *
|
|
|
* Scans all the CRTCs and planes in @dev's mode_config. If they're
|
|
|
- * using @fb, removes it, setting it to NULL.
|
|
|
+ * using @fb, removes it, setting it to NULL. Then drops the reference to the
|
|
|
+ * passed-in framebuffer. Might take the modeset locks.
|
|
|
+ *
|
|
|
+ * Note that this function optimizes the cleanup away if the caller holds the
|
|
|
+ * last reference to the framebuffer. It is also guaranteed to not take the
|
|
|
+ * modeset locks in this case.
|
|
|
*/
|
|
|
void drm_framebuffer_remove(struct drm_framebuffer *fb)
|
|
|
{
|
|
@@ -384,33 +531,53 @@ void drm_framebuffer_remove(struct drm_framebuffer *fb)
|
|
|
struct drm_mode_set set;
|
|
|
int ret;
|
|
|
|
|
|
- /* remove from any CRTC */
|
|
|
- list_for_each_entry(crtc, &dev->mode_config.crtc_list, head) {
|
|
|
- if (crtc->fb == fb) {
|
|
|
- /* should turn off the crtc */
|
|
|
- memset(&set, 0, sizeof(struct drm_mode_set));
|
|
|
- set.crtc = crtc;
|
|
|
- set.fb = NULL;
|
|
|
- ret = crtc->funcs->set_config(&set);
|
|
|
- if (ret)
|
|
|
- DRM_ERROR("failed to reset crtc %p when fb was deleted\n", crtc);
|
|
|
+ WARN_ON(!list_empty(&fb->filp_head));
|
|
|
+
|
|
|
+ /*
|
|
|
+ * drm ABI mandates that we remove any deleted framebuffers from active
|
|
|
+ * useage. But since most sane clients only remove framebuffers they no
|
|
|
+ * longer need, try to optimize this away.
|
|
|
+ *
|
|
|
+ * Since we're holding a reference ourselves, observing a refcount of 1
|
|
|
+ * means that we're the last holder and can skip it. Also, the refcount
|
|
|
+ * can never increase from 1 again, so we don't need any barriers or
|
|
|
+ * locks.
|
|
|
+ *
|
|
|
+ * Note that userspace could try to race with use and instate a new
|
|
|
+ * usage _after_ we've cleared all current ones. End result will be an
|
|
|
+ * in-use fb with fb-id == 0. Userspace is allowed to shoot its own foot
|
|
|
+ * in this manner.
|
|
|
+ */
|
|
|
+ if (atomic_read(&fb->refcount.refcount) > 1) {
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
+ /* remove from any CRTC */
|
|
|
+ list_for_each_entry(crtc, &dev->mode_config.crtc_list, head) {
|
|
|
+ if (crtc->fb == fb) {
|
|
|
+ /* should turn off the crtc */
|
|
|
+ memset(&set, 0, sizeof(struct drm_mode_set));
|
|
|
+ set.crtc = crtc;
|
|
|
+ set.fb = NULL;
|
|
|
+ ret = drm_mode_set_config_internal(&set);
|
|
|
+ if (ret)
|
|
|
+ DRM_ERROR("failed to reset crtc %p when fb was deleted\n", crtc);
|
|
|
+ }
|
|
|
}
|
|
|
- }
|
|
|
|
|
|
- list_for_each_entry(plane, &dev->mode_config.plane_list, head) {
|
|
|
- if (plane->fb == fb) {
|
|
|
- /* should turn off the crtc */
|
|
|
- ret = plane->funcs->disable_plane(plane);
|
|
|
- if (ret)
|
|
|
- DRM_ERROR("failed to disable plane with busy fb\n");
|
|
|
- /* disconnect the plane from the fb and crtc: */
|
|
|
- plane->fb = NULL;
|
|
|
- plane->crtc = NULL;
|
|
|
+ list_for_each_entry(plane, &dev->mode_config.plane_list, head) {
|
|
|
+ if (plane->fb == fb) {
|
|
|
+ /* should turn off the crtc */
|
|
|
+ ret = plane->funcs->disable_plane(plane);
|
|
|
+ if (ret)
|
|
|
+ DRM_ERROR("failed to disable plane with busy fb\n");
|
|
|
+ /* disconnect the plane from the fb and crtc: */
|
|
|
+ __drm_framebuffer_unreference(plane->fb);
|
|
|
+ plane->fb = NULL;
|
|
|
+ plane->crtc = NULL;
|
|
|
+ }
|
|
|
}
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
}
|
|
|
|
|
|
- list_del(&fb->filp_head);
|
|
|
-
|
|
|
drm_framebuffer_unreference(fb);
|
|
|
}
|
|
|
EXPORT_SYMBOL(drm_framebuffer_remove);
|
|
@@ -421,9 +588,6 @@ EXPORT_SYMBOL(drm_framebuffer_remove);
|
|
|
* @crtc: CRTC object to init
|
|
|
* @funcs: callbacks for the new CRTC
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Takes mode_config lock.
|
|
|
- *
|
|
|
* Inits a new object created as base part of an driver crtc object.
|
|
|
*
|
|
|
* RETURNS:
|
|
@@ -438,7 +602,9 @@ int drm_crtc_init(struct drm_device *dev, struct drm_crtc *crtc,
|
|
|
crtc->funcs = funcs;
|
|
|
crtc->invert_dimensions = false;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
+ mutex_init(&crtc->mutex);
|
|
|
+ mutex_lock_nest_lock(&crtc->mutex, &dev->mode_config.mutex);
|
|
|
|
|
|
ret = drm_mode_object_get(dev, &crtc->base, DRM_MODE_OBJECT_CRTC);
|
|
|
if (ret)
|
|
@@ -450,7 +616,7 @@ int drm_crtc_init(struct drm_device *dev, struct drm_crtc *crtc,
|
|
|
dev->mode_config.num_crtc++;
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
|
|
|
return ret;
|
|
|
}
|
|
@@ -460,9 +626,6 @@ EXPORT_SYMBOL(drm_crtc_init);
|
|
|
* drm_crtc_cleanup - Cleans up the core crtc usage.
|
|
|
* @crtc: CRTC to cleanup
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold mode config lock.
|
|
|
- *
|
|
|
* Cleanup @crtc. Removes from drm modesetting space
|
|
|
* does NOT free object, caller does that.
|
|
|
*/
|
|
@@ -484,9 +647,6 @@ EXPORT_SYMBOL(drm_crtc_cleanup);
|
|
|
* @connector: connector the new mode
|
|
|
* @mode: mode data
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold mode config lock.
|
|
|
- *
|
|
|
* Add @mode to @connector's mode list for later use.
|
|
|
*/
|
|
|
void drm_mode_probed_add(struct drm_connector *connector,
|
|
@@ -501,9 +661,6 @@ EXPORT_SYMBOL(drm_mode_probed_add);
|
|
|
* @connector: connector list to modify
|
|
|
* @mode: mode to remove
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold mode config lock.
|
|
|
- *
|
|
|
* Remove @mode from @connector's mode list, then free it.
|
|
|
*/
|
|
|
void drm_mode_remove(struct drm_connector *connector,
|
|
@@ -519,10 +676,7 @@ EXPORT_SYMBOL(drm_mode_remove);
|
|
|
* @dev: DRM device
|
|
|
* @connector: the connector to init
|
|
|
* @funcs: callbacks for this connector
|
|
|
- * @name: user visible name of the connector
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @connector_type: user visible type of the connector
|
|
|
*
|
|
|
* Initialises a preallocated connector. Connectors should be
|
|
|
* subclassed as part of driver connector objects.
|
|
@@ -537,7 +691,7 @@ int drm_connector_init(struct drm_device *dev,
|
|
|
{
|
|
|
int ret;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
|
|
|
ret = drm_mode_object_get(dev, &connector->base, DRM_MODE_OBJECT_CONNECTOR);
|
|
|
if (ret)
|
|
@@ -567,7 +721,7 @@ int drm_connector_init(struct drm_device *dev,
|
|
|
dev->mode_config.dpms_property, 0);
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
|
|
|
return ret;
|
|
|
}
|
|
@@ -577,9 +731,6 @@ EXPORT_SYMBOL(drm_connector_init);
|
|
|
* drm_connector_cleanup - cleans up an initialised connector
|
|
|
* @connector: connector to cleanup
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
- *
|
|
|
* Cleans up the connector but doesn't free the object.
|
|
|
*/
|
|
|
void drm_connector_cleanup(struct drm_connector *connector)
|
|
@@ -596,11 +747,9 @@ void drm_connector_cleanup(struct drm_connector *connector)
|
|
|
list_for_each_entry_safe(mode, t, &connector->user_modes, head)
|
|
|
drm_mode_remove(connector, mode);
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
drm_mode_object_put(dev, &connector->base);
|
|
|
list_del(&connector->head);
|
|
|
dev->mode_config.num_connector--;
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
}
|
|
|
EXPORT_SYMBOL(drm_connector_cleanup);
|
|
|
|
|
@@ -622,7 +771,7 @@ int drm_encoder_init(struct drm_device *dev,
|
|
|
{
|
|
|
int ret;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
|
|
|
ret = drm_mode_object_get(dev, &encoder->base, DRM_MODE_OBJECT_ENCODER);
|
|
|
if (ret)
|
|
@@ -636,7 +785,7 @@ int drm_encoder_init(struct drm_device *dev,
|
|
|
dev->mode_config.num_encoder++;
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
|
|
|
return ret;
|
|
|
}
|
|
@@ -645,11 +794,11 @@ EXPORT_SYMBOL(drm_encoder_init);
|
|
|
void drm_encoder_cleanup(struct drm_encoder *encoder)
|
|
|
{
|
|
|
struct drm_device *dev = encoder->dev;
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
drm_mode_object_put(dev, &encoder->base);
|
|
|
list_del(&encoder->head);
|
|
|
dev->mode_config.num_encoder--;
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
}
|
|
|
EXPORT_SYMBOL(drm_encoder_cleanup);
|
|
|
|
|
@@ -661,7 +810,7 @@ int drm_plane_init(struct drm_device *dev, struct drm_plane *plane,
|
|
|
{
|
|
|
int ret;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
|
|
|
ret = drm_mode_object_get(dev, &plane->base, DRM_MODE_OBJECT_PLANE);
|
|
|
if (ret)
|
|
@@ -695,7 +844,7 @@ int drm_plane_init(struct drm_device *dev, struct drm_plane *plane,
|
|
|
}
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
|
|
|
return ret;
|
|
|
}
|
|
@@ -705,7 +854,7 @@ void drm_plane_cleanup(struct drm_plane *plane)
|
|
|
{
|
|
|
struct drm_device *dev = plane->dev;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
kfree(plane->format_types);
|
|
|
drm_mode_object_put(dev, &plane->base);
|
|
|
/* if not added to a list, it must be a private plane */
|
|
@@ -713,7 +862,7 @@ void drm_plane_cleanup(struct drm_plane *plane)
|
|
|
list_del(&plane->head);
|
|
|
dev->mode_config.num_plane--;
|
|
|
}
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
}
|
|
|
EXPORT_SYMBOL(drm_plane_cleanup);
|
|
|
|
|
@@ -721,9 +870,6 @@ EXPORT_SYMBOL(drm_plane_cleanup);
|
|
|
* drm_mode_create - create a new display mode
|
|
|
* @dev: DRM device
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold DRM mode_config lock.
|
|
|
- *
|
|
|
* Create a new drm_display_mode, give it an ID, and return it.
|
|
|
*
|
|
|
* RETURNS:
|
|
@@ -751,9 +897,6 @@ EXPORT_SYMBOL(drm_mode_create);
|
|
|
* @dev: DRM device
|
|
|
* @mode: mode to remove
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold mode config lock.
|
|
|
- *
|
|
|
* Free @mode's unique identifier, then free it.
|
|
|
*/
|
|
|
void drm_mode_destroy(struct drm_device *dev, struct drm_display_mode *mode)
|
|
@@ -978,16 +1121,19 @@ EXPORT_SYMBOL(drm_mode_create_dirty_info_property);
|
|
|
* drm_mode_config_init - initialize DRM mode_configuration structure
|
|
|
* @dev: DRM device
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * None, should happen single threaded at init time.
|
|
|
- *
|
|
|
* Initialize @dev's mode_config structure, used for tracking the graphics
|
|
|
* configuration of @dev.
|
|
|
+ *
|
|
|
+ * Since this initializes the modeset locks, no locking is possible. Which is no
|
|
|
+ * problem, since this should happen single threaded at init time. It is the
|
|
|
+ * driver's problem to ensure this guarantee.
|
|
|
+ *
|
|
|
*/
|
|
|
void drm_mode_config_init(struct drm_device *dev)
|
|
|
{
|
|
|
mutex_init(&dev->mode_config.mutex);
|
|
|
mutex_init(&dev->mode_config.idr_mutex);
|
|
|
+ mutex_init(&dev->mode_config.fb_lock);
|
|
|
INIT_LIST_HEAD(&dev->mode_config.fb_list);
|
|
|
INIT_LIST_HEAD(&dev->mode_config.crtc_list);
|
|
|
INIT_LIST_HEAD(&dev->mode_config.connector_list);
|
|
@@ -997,9 +1143,9 @@ void drm_mode_config_init(struct drm_device *dev)
|
|
|
INIT_LIST_HEAD(&dev->mode_config.plane_list);
|
|
|
idr_init(&dev->mode_config.crtc_idr);
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
drm_mode_create_standard_connector_properties(dev);
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
|
|
|
/* Just to be sure */
|
|
|
dev->mode_config.num_fb = 0;
|
|
@@ -1057,12 +1203,13 @@ EXPORT_SYMBOL(drm_mode_group_init_legacy_group);
|
|
|
* drm_mode_config_cleanup - free up DRM mode_config info
|
|
|
* @dev: DRM device
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Caller must hold mode config lock.
|
|
|
- *
|
|
|
* Free up all the connectors and CRTCs associated with this DRM device, then
|
|
|
* free up the framebuffers and associated buffer objects.
|
|
|
*
|
|
|
+ * Note that since this /should/ happen single-threaded at driver/device
|
|
|
+ * teardown time, no locking is required. It's the driver's job to ensure that
|
|
|
+ * this guarantee actually holds true.
|
|
|
+ *
|
|
|
* FIXME: cleanup any dangling user buffer objects too
|
|
|
*/
|
|
|
void drm_mode_config_cleanup(struct drm_device *dev)
|
|
@@ -1089,6 +1236,15 @@ void drm_mode_config_cleanup(struct drm_device *dev)
|
|
|
drm_property_destroy(dev, property);
|
|
|
}
|
|
|
|
|
|
+ /*
|
|
|
+ * Single-threaded teardown context, so it's not required to grab the
|
|
|
+ * fb_lock to protect against concurrent fb_list access. Contrary, it
|
|
|
+ * would actually deadlock with the drm_framebuffer_cleanup function.
|
|
|
+ *
|
|
|
+ * Also, if there are any framebuffers left, that's a driver leak now,
|
|
|
+ * so politely WARN about this.
|
|
|
+ */
|
|
|
+ WARN_ON(!list_empty(&dev->mode_config.fb_list));
|
|
|
list_for_each_entry_safe(fb, fbt, &dev->mode_config.fb_list, head) {
|
|
|
drm_framebuffer_remove(fb);
|
|
|
}
|
|
@@ -1112,9 +1268,6 @@ EXPORT_SYMBOL(drm_mode_config_cleanup);
|
|
|
* @out: drm_mode_modeinfo struct to return to the user
|
|
|
* @in: drm_display_mode to use
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * None.
|
|
|
- *
|
|
|
* Convert a drm_display_mode into a drm_mode_modeinfo structure to return to
|
|
|
* the user.
|
|
|
*/
|
|
@@ -1151,9 +1304,6 @@ static void drm_crtc_convert_to_umode(struct drm_mode_modeinfo *out,
|
|
|
* @out: drm_display_mode to return to the user
|
|
|
* @in: drm_mode_modeinfo to use
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * None.
|
|
|
- *
|
|
|
* Convert a drm_mode_modeinfo into a drm_display_mode structure to return to
|
|
|
* the caller.
|
|
|
*
|
|
@@ -1188,13 +1338,9 @@ static int drm_crtc_convert_umode(struct drm_display_mode *out,
|
|
|
|
|
|
/**
|
|
|
* drm_mode_getresources - get graphics configuration
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* Construct a set of configuration description structures and return
|
|
|
* them to the user, including CRTC, connector and framebuffer configuration.
|
|
@@ -1228,8 +1374,8 @@ int drm_mode_getresources(struct drm_device *dev, void *data,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
|
|
|
+ mutex_lock(&file_priv->fbs_lock);
|
|
|
/*
|
|
|
* For the non-control nodes we need to limit the list of resources
|
|
|
* by IDs in the group list for this node
|
|
@@ -1237,6 +1383,23 @@ int drm_mode_getresources(struct drm_device *dev, void *data,
|
|
|
list_for_each(lh, &file_priv->fbs)
|
|
|
fb_count++;
|
|
|
|
|
|
+ /* handle this in 4 parts */
|
|
|
+ /* FBs */
|
|
|
+ if (card_res->count_fbs >= fb_count) {
|
|
|
+ copied = 0;
|
|
|
+ fb_id = (uint32_t __user *)(unsigned long)card_res->fb_id_ptr;
|
|
|
+ list_for_each_entry(fb, &file_priv->fbs, filp_head) {
|
|
|
+ if (put_user(fb->base.id, fb_id + copied)) {
|
|
|
+ mutex_unlock(&file_priv->fbs_lock);
|
|
|
+ return -EFAULT;
|
|
|
+ }
|
|
|
+ copied++;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ card_res->count_fbs = fb_count;
|
|
|
+ mutex_unlock(&file_priv->fbs_lock);
|
|
|
+
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
mode_group = &file_priv->master->minor->mode_group;
|
|
|
if (file_priv->master->minor->type == DRM_MINOR_CONTROL) {
|
|
|
|
|
@@ -1260,21 +1423,6 @@ int drm_mode_getresources(struct drm_device *dev, void *data,
|
|
|
card_res->max_width = dev->mode_config.max_width;
|
|
|
card_res->min_width = dev->mode_config.min_width;
|
|
|
|
|
|
- /* handle this in 4 parts */
|
|
|
- /* FBs */
|
|
|
- if (card_res->count_fbs >= fb_count) {
|
|
|
- copied = 0;
|
|
|
- fb_id = (uint32_t __user *)(unsigned long)card_res->fb_id_ptr;
|
|
|
- list_for_each_entry(fb, &file_priv->fbs, filp_head) {
|
|
|
- if (put_user(fb->base.id, fb_id + copied)) {
|
|
|
- ret = -EFAULT;
|
|
|
- goto out;
|
|
|
- }
|
|
|
- copied++;
|
|
|
- }
|
|
|
- }
|
|
|
- card_res->count_fbs = fb_count;
|
|
|
-
|
|
|
/* CRTCs */
|
|
|
if (card_res->count_crtcs >= crtc_count) {
|
|
|
copied = 0;
|
|
@@ -1370,19 +1518,15 @@ int drm_mode_getresources(struct drm_device *dev, void *data,
|
|
|
card_res->count_connectors, card_res->count_encoders);
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* drm_mode_getcrtc - get CRTC configuration
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* Construct a CRTC configuration structure to return to the user.
|
|
|
*
|
|
@@ -1402,7 +1546,7 @@ int drm_mode_getcrtc(struct drm_device *dev,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
|
|
|
obj = drm_mode_object_find(dev, crtc_resp->crtc_id,
|
|
|
DRM_MODE_OBJECT_CRTC);
|
|
@@ -1430,19 +1574,15 @@ int drm_mode_getcrtc(struct drm_device *dev,
|
|
|
}
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* drm_mode_getconnector - get connector configuration
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* Construct a connector configuration structure to return to the user.
|
|
|
*
|
|
@@ -1575,6 +1715,7 @@ int drm_mode_getconnector(struct drm_device *dev, void *data,
|
|
|
|
|
|
out:
|
|
|
mutex_unlock(&dev->mode_config.mutex);
|
|
|
+
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -1589,7 +1730,7 @@ int drm_mode_getencoder(struct drm_device *dev, void *data,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
obj = drm_mode_object_find(dev, enc_resp->encoder_id,
|
|
|
DRM_MODE_OBJECT_ENCODER);
|
|
|
if (!obj) {
|
|
@@ -1608,7 +1749,7 @@ int drm_mode_getencoder(struct drm_device *dev, void *data,
|
|
|
enc_resp->possible_clones = encoder->possible_clones;
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -1618,9 +1759,6 @@ out:
|
|
|
* @data: ioctl data
|
|
|
* @file_priv: DRM file info
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
- *
|
|
|
* Return an plane count and set of IDs.
|
|
|
*/
|
|
|
int drm_mode_getplane_res(struct drm_device *dev, void *data,
|
|
@@ -1635,7 +1773,7 @@ int drm_mode_getplane_res(struct drm_device *dev, void *data,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
config = &dev->mode_config;
|
|
|
|
|
|
/*
|
|
@@ -1657,7 +1795,7 @@ int drm_mode_getplane_res(struct drm_device *dev, void *data,
|
|
|
plane_resp->count_planes = config->num_plane;
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -1667,9 +1805,6 @@ out:
|
|
|
* @data: ioctl data
|
|
|
* @file_priv: DRM file info
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
- *
|
|
|
* Return plane info, including formats supported, gamma size, any
|
|
|
* current fb, etc.
|
|
|
*/
|
|
@@ -1685,7 +1820,7 @@ int drm_mode_getplane(struct drm_device *dev, void *data,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
obj = drm_mode_object_find(dev, plane_resp->plane_id,
|
|
|
DRM_MODE_OBJECT_PLANE);
|
|
|
if (!obj) {
|
|
@@ -1725,7 +1860,7 @@ int drm_mode_getplane(struct drm_device *dev, void *data,
|
|
|
plane_resp->count_format_types = plane->format_count;
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -1733,10 +1868,7 @@ out:
|
|
|
* drm_mode_setplane - set up or tear down an plane
|
|
|
* @dev: DRM device
|
|
|
* @data: ioctl data*
|
|
|
- * @file_prive: DRM file info
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @file_priv: DRM file info
|
|
|
*
|
|
|
* Set plane info, including placement, fb, scaling, and other factors.
|
|
|
* Or pass a NULL fb to disable.
|
|
@@ -1748,7 +1880,7 @@ int drm_mode_setplane(struct drm_device *dev, void *data,
|
|
|
struct drm_mode_object *obj;
|
|
|
struct drm_plane *plane;
|
|
|
struct drm_crtc *crtc;
|
|
|
- struct drm_framebuffer *fb;
|
|
|
+ struct drm_framebuffer *fb = NULL, *old_fb = NULL;
|
|
|
int ret = 0;
|
|
|
unsigned int fb_width, fb_height;
|
|
|
int i;
|
|
@@ -1756,8 +1888,6 @@ int drm_mode_setplane(struct drm_device *dev, void *data,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
-
|
|
|
/*
|
|
|
* First, find the plane, crtc, and fb objects. If not available,
|
|
|
* we don't bother to call the driver.
|
|
@@ -1767,16 +1897,18 @@ int drm_mode_setplane(struct drm_device *dev, void *data,
|
|
|
if (!obj) {
|
|
|
DRM_DEBUG_KMS("Unknown plane ID %d\n",
|
|
|
plane_req->plane_id);
|
|
|
- ret = -ENOENT;
|
|
|
- goto out;
|
|
|
+ return -ENOENT;
|
|
|
}
|
|
|
plane = obj_to_plane(obj);
|
|
|
|
|
|
/* No fb means shut it down */
|
|
|
if (!plane_req->fb_id) {
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
+ old_fb = plane->fb;
|
|
|
plane->funcs->disable_plane(plane);
|
|
|
plane->crtc = NULL;
|
|
|
plane->fb = NULL;
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
goto out;
|
|
|
}
|
|
|
|
|
@@ -1790,15 +1922,13 @@ int drm_mode_setplane(struct drm_device *dev, void *data,
|
|
|
}
|
|
|
crtc = obj_to_crtc(obj);
|
|
|
|
|
|
- obj = drm_mode_object_find(dev, plane_req->fb_id,
|
|
|
- DRM_MODE_OBJECT_FB);
|
|
|
- if (!obj) {
|
|
|
+ fb = drm_framebuffer_lookup(dev, plane_req->fb_id);
|
|
|
+ if (!fb) {
|
|
|
DRM_DEBUG_KMS("Unknown framebuffer ID %d\n",
|
|
|
plane_req->fb_id);
|
|
|
ret = -ENOENT;
|
|
|
goto out;
|
|
|
}
|
|
|
- fb = obj_to_fb(obj);
|
|
|
|
|
|
/* Check whether this plane supports the fb pixel format. */
|
|
|
for (i = 0; i < plane->format_count; i++)
|
|
@@ -1844,31 +1974,62 @@ int drm_mode_setplane(struct drm_device *dev, void *data,
|
|
|
goto out;
|
|
|
}
|
|
|
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
ret = plane->funcs->update_plane(plane, crtc, fb,
|
|
|
plane_req->crtc_x, plane_req->crtc_y,
|
|
|
plane_req->crtc_w, plane_req->crtc_h,
|
|
|
plane_req->src_x, plane_req->src_y,
|
|
|
plane_req->src_w, plane_req->src_h);
|
|
|
if (!ret) {
|
|
|
+ old_fb = plane->fb;
|
|
|
+ fb = NULL;
|
|
|
plane->crtc = crtc;
|
|
|
plane->fb = fb;
|
|
|
}
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ if (fb)
|
|
|
+ drm_framebuffer_unreference(fb);
|
|
|
+ if (old_fb)
|
|
|
+ drm_framebuffer_unreference(old_fb);
|
|
|
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * drm_mode_setcrtc - set CRTC configuration
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
+ * drm_mode_set_config_internal - helper to call ->set_config
|
|
|
+ * @set: modeset config to set
|
|
|
*
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * This is a little helper to wrap internal calls to the ->set_config driver
|
|
|
+ * interface. The only thing it adds is correct refcounting dance.
|
|
|
+ */
|
|
|
+int drm_mode_set_config_internal(struct drm_mode_set *set)
|
|
|
+{
|
|
|
+ struct drm_crtc *crtc = set->crtc;
|
|
|
+ struct drm_framebuffer *fb, *old_fb;
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ old_fb = crtc->fb;
|
|
|
+ fb = set->fb;
|
|
|
+
|
|
|
+ ret = crtc->funcs->set_config(set);
|
|
|
+ if (ret == 0) {
|
|
|
+ if (old_fb)
|
|
|
+ drm_framebuffer_unreference(old_fb);
|
|
|
+ if (fb)
|
|
|
+ drm_framebuffer_reference(fb);
|
|
|
+ }
|
|
|
+
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+EXPORT_SYMBOL(drm_mode_set_config_internal);
|
|
|
+
|
|
|
+/**
|
|
|
+ * drm_mode_setcrtc - set CRTC configuration
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* Build a new CRTC configuration based on user request.
|
|
|
*
|
|
@@ -1899,7 +2060,7 @@ int drm_mode_setcrtc(struct drm_device *dev, void *data,
|
|
|
if (crtc_req->x > INT_MAX || crtc_req->y > INT_MAX)
|
|
|
return -ERANGE;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
obj = drm_mode_object_find(dev, crtc_req->crtc_id,
|
|
|
DRM_MODE_OBJECT_CRTC);
|
|
|
if (!obj) {
|
|
@@ -1921,16 +2082,16 @@ int drm_mode_setcrtc(struct drm_device *dev, void *data,
|
|
|
goto out;
|
|
|
}
|
|
|
fb = crtc->fb;
|
|
|
+ /* Make refcounting symmetric with the lookup path. */
|
|
|
+ drm_framebuffer_reference(fb);
|
|
|
} else {
|
|
|
- obj = drm_mode_object_find(dev, crtc_req->fb_id,
|
|
|
- DRM_MODE_OBJECT_FB);
|
|
|
- if (!obj) {
|
|
|
+ fb = drm_framebuffer_lookup(dev, crtc_req->fb_id);
|
|
|
+ if (!fb) {
|
|
|
DRM_DEBUG_KMS("Unknown FB ID%d\n",
|
|
|
crtc_req->fb_id);
|
|
|
ret = -EINVAL;
|
|
|
goto out;
|
|
|
}
|
|
|
- fb = obj_to_fb(obj);
|
|
|
}
|
|
|
|
|
|
mode = drm_mode_create(dev);
|
|
@@ -2027,12 +2188,15 @@ int drm_mode_setcrtc(struct drm_device *dev, void *data,
|
|
|
set.connectors = connector_set;
|
|
|
set.num_connectors = crtc_req->count_connectors;
|
|
|
set.fb = fb;
|
|
|
- ret = crtc->funcs->set_config(&set);
|
|
|
+ ret = drm_mode_set_config_internal(&set);
|
|
|
|
|
|
out:
|
|
|
+ if (fb)
|
|
|
+ drm_framebuffer_unreference(fb);
|
|
|
+
|
|
|
kfree(connector_set);
|
|
|
drm_mode_destroy(dev, mode);
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -2050,15 +2214,14 @@ int drm_mode_cursor_ioctl(struct drm_device *dev,
|
|
|
if (!req->flags || (~DRM_MODE_CURSOR_FLAGS & req->flags))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
obj = drm_mode_object_find(dev, req->crtc_id, DRM_MODE_OBJECT_CRTC);
|
|
|
if (!obj) {
|
|
|
DRM_DEBUG_KMS("Unknown CRTC ID %d\n", req->crtc_id);
|
|
|
- ret = -EINVAL;
|
|
|
- goto out;
|
|
|
+ return -EINVAL;
|
|
|
}
|
|
|
crtc = obj_to_crtc(obj);
|
|
|
|
|
|
+ mutex_lock(&crtc->mutex);
|
|
|
if (req->flags & DRM_MODE_CURSOR_BO) {
|
|
|
if (!crtc->funcs->cursor_set) {
|
|
|
ret = -ENXIO;
|
|
@@ -2078,7 +2241,8 @@ int drm_mode_cursor_ioctl(struct drm_device *dev,
|
|
|
}
|
|
|
}
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ mutex_unlock(&crtc->mutex);
|
|
|
+
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -2120,13 +2284,9 @@ EXPORT_SYMBOL(drm_mode_legacy_fb_format);
|
|
|
|
|
|
/**
|
|
|
* drm_mode_addfb - add an FB to the graphics configuration
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* Add a new FB to the specified CRTC, given a user request.
|
|
|
*
|
|
@@ -2161,24 +2321,19 @@ int drm_mode_addfb(struct drm_device *dev,
|
|
|
if ((config->min_height > r.height) || (r.height > config->max_height))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
-
|
|
|
- /* TODO check buffer is sufficiently large */
|
|
|
- /* TODO setup destructor callback */
|
|
|
-
|
|
|
fb = dev->mode_config.funcs->fb_create(dev, file_priv, &r);
|
|
|
if (IS_ERR(fb)) {
|
|
|
DRM_DEBUG_KMS("could not create framebuffer\n");
|
|
|
- ret = PTR_ERR(fb);
|
|
|
- goto out;
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
+ return PTR_ERR(fb);
|
|
|
}
|
|
|
|
|
|
+ mutex_lock(&file_priv->fbs_lock);
|
|
|
or->fb_id = fb->base.id;
|
|
|
list_add(&fb->filp_head, &file_priv->fbs);
|
|
|
DRM_DEBUG_KMS("[FB:%d]\n", fb->base.id);
|
|
|
+ mutex_unlock(&file_priv->fbs_lock);
|
|
|
|
|
|
-out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -2304,13 +2459,9 @@ static int framebuffer_check(const struct drm_mode_fb_cmd2 *r)
|
|
|
|
|
|
/**
|
|
|
* drm_mode_addfb2 - add an FB to the graphics configuration
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* Add a new FB to the specified CRTC, given a user request with format.
|
|
|
*
|
|
@@ -2350,33 +2501,28 @@ int drm_mode_addfb2(struct drm_device *dev,
|
|
|
if (ret)
|
|
|
return ret;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
-
|
|
|
fb = dev->mode_config.funcs->fb_create(dev, file_priv, r);
|
|
|
if (IS_ERR(fb)) {
|
|
|
DRM_DEBUG_KMS("could not create framebuffer\n");
|
|
|
- ret = PTR_ERR(fb);
|
|
|
- goto out;
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
+ return PTR_ERR(fb);
|
|
|
}
|
|
|
|
|
|
+ mutex_lock(&file_priv->fbs_lock);
|
|
|
r->fb_id = fb->base.id;
|
|
|
list_add(&fb->filp_head, &file_priv->fbs);
|
|
|
DRM_DEBUG_KMS("[FB:%d]\n", fb->base.id);
|
|
|
+ mutex_unlock(&file_priv->fbs_lock);
|
|
|
+
|
|
|
|
|
|
-out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* drm_mode_rmfb - remove an FB from the configuration
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* Remove the FB specified by the user.
|
|
|
*
|
|
@@ -2388,50 +2534,49 @@ out:
|
|
|
int drm_mode_rmfb(struct drm_device *dev,
|
|
|
void *data, struct drm_file *file_priv)
|
|
|
{
|
|
|
- struct drm_mode_object *obj;
|
|
|
struct drm_framebuffer *fb = NULL;
|
|
|
struct drm_framebuffer *fbl = NULL;
|
|
|
uint32_t *id = data;
|
|
|
- int ret = 0;
|
|
|
int found = 0;
|
|
|
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
- obj = drm_mode_object_find(dev, *id, DRM_MODE_OBJECT_FB);
|
|
|
- /* TODO check that we really get a framebuffer back. */
|
|
|
- if (!obj) {
|
|
|
- ret = -EINVAL;
|
|
|
- goto out;
|
|
|
- }
|
|
|
- fb = obj_to_fb(obj);
|
|
|
+ mutex_lock(&file_priv->fbs_lock);
|
|
|
+ mutex_lock(&dev->mode_config.fb_lock);
|
|
|
+ fb = __drm_framebuffer_lookup(dev, *id);
|
|
|
+ if (!fb)
|
|
|
+ goto fail_lookup;
|
|
|
|
|
|
list_for_each_entry(fbl, &file_priv->fbs, filp_head)
|
|
|
if (fb == fbl)
|
|
|
found = 1;
|
|
|
+ if (!found)
|
|
|
+ goto fail_lookup;
|
|
|
|
|
|
- if (!found) {
|
|
|
- ret = -EINVAL;
|
|
|
- goto out;
|
|
|
- }
|
|
|
+ /* Mark fb as reaped, we still have a ref from fpriv->fbs. */
|
|
|
+ __drm_framebuffer_unregister(dev, fb);
|
|
|
+
|
|
|
+ list_del_init(&fb->filp_head);
|
|
|
+ mutex_unlock(&dev->mode_config.fb_lock);
|
|
|
+ mutex_unlock(&file_priv->fbs_lock);
|
|
|
|
|
|
drm_framebuffer_remove(fb);
|
|
|
|
|
|
-out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
- return ret;
|
|
|
+ return 0;
|
|
|
+
|
|
|
+fail_lookup:
|
|
|
+ mutex_unlock(&dev->mode_config.fb_lock);
|
|
|
+ mutex_unlock(&file_priv->fbs_lock);
|
|
|
+
|
|
|
+ return -EINVAL;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* drm_mode_getfb - get FB info
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* Lookup the FB given its ID and return info about it.
|
|
|
*
|
|
@@ -2444,30 +2589,28 @@ int drm_mode_getfb(struct drm_device *dev,
|
|
|
void *data, struct drm_file *file_priv)
|
|
|
{
|
|
|
struct drm_mode_fb_cmd *r = data;
|
|
|
- struct drm_mode_object *obj;
|
|
|
struct drm_framebuffer *fb;
|
|
|
- int ret = 0;
|
|
|
+ int ret;
|
|
|
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
- obj = drm_mode_object_find(dev, r->fb_id, DRM_MODE_OBJECT_FB);
|
|
|
- if (!obj) {
|
|
|
- ret = -EINVAL;
|
|
|
- goto out;
|
|
|
- }
|
|
|
- fb = obj_to_fb(obj);
|
|
|
+ fb = drm_framebuffer_lookup(dev, r->fb_id);
|
|
|
+ if (!fb)
|
|
|
+ return -EINVAL;
|
|
|
|
|
|
r->height = fb->height;
|
|
|
r->width = fb->width;
|
|
|
r->depth = fb->depth;
|
|
|
r->bpp = fb->bits_per_pixel;
|
|
|
r->pitch = fb->pitches[0];
|
|
|
- fb->funcs->create_handle(fb, file_priv, &r->handle);
|
|
|
+ if (fb->funcs->create_handle)
|
|
|
+ ret = fb->funcs->create_handle(fb, file_priv, &r->handle);
|
|
|
+ else
|
|
|
+ ret = -ENODEV;
|
|
|
+
|
|
|
+ drm_framebuffer_unreference(fb);
|
|
|
|
|
|
-out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -2477,7 +2620,6 @@ int drm_mode_dirtyfb_ioctl(struct drm_device *dev,
|
|
|
struct drm_clip_rect __user *clips_ptr;
|
|
|
struct drm_clip_rect *clips = NULL;
|
|
|
struct drm_mode_fb_dirty_cmd *r = data;
|
|
|
- struct drm_mode_object *obj;
|
|
|
struct drm_framebuffer *fb;
|
|
|
unsigned flags;
|
|
|
int num_clips;
|
|
@@ -2486,13 +2628,9 @@ int drm_mode_dirtyfb_ioctl(struct drm_device *dev,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
- obj = drm_mode_object_find(dev, r->fb_id, DRM_MODE_OBJECT_FB);
|
|
|
- if (!obj) {
|
|
|
- ret = -EINVAL;
|
|
|
- goto out_err1;
|
|
|
- }
|
|
|
- fb = obj_to_fb(obj);
|
|
|
+ fb = drm_framebuffer_lookup(dev, r->fb_id);
|
|
|
+ if (!fb)
|
|
|
+ return -EINVAL;
|
|
|
|
|
|
num_clips = r->num_clips;
|
|
|
clips_ptr = (struct drm_clip_rect __user *)(unsigned long)r->clips_ptr;
|
|
@@ -2530,27 +2668,26 @@ int drm_mode_dirtyfb_ioctl(struct drm_device *dev,
|
|
|
}
|
|
|
|
|
|
if (fb->funcs->dirty) {
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
ret = fb->funcs->dirty(fb, file_priv, flags, r->color,
|
|
|
clips, num_clips);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
} else {
|
|
|
ret = -ENOSYS;
|
|
|
- goto out_err2;
|
|
|
}
|
|
|
|
|
|
out_err2:
|
|
|
kfree(clips);
|
|
|
out_err1:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_framebuffer_unreference(fb);
|
|
|
+
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
|
|
|
/**
|
|
|
* drm_fb_release - remove and free the FBs on this file
|
|
|
- * @filp: file * from the ioctl
|
|
|
- *
|
|
|
- * LOCKING:
|
|
|
- * Takes mode config lock.
|
|
|
+ * @priv: drm file for the ioctl
|
|
|
*
|
|
|
* Destroy all the FBs associated with @filp.
|
|
|
*
|
|
@@ -2564,11 +2701,20 @@ void drm_fb_release(struct drm_file *priv)
|
|
|
struct drm_device *dev = priv->minor->dev;
|
|
|
struct drm_framebuffer *fb, *tfb;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ mutex_lock(&priv->fbs_lock);
|
|
|
list_for_each_entry_safe(fb, tfb, &priv->fbs, filp_head) {
|
|
|
+
|
|
|
+ mutex_lock(&dev->mode_config.fb_lock);
|
|
|
+ /* Mark fb as reaped, we still have a ref from fpriv->fbs. */
|
|
|
+ __drm_framebuffer_unregister(dev, fb);
|
|
|
+ mutex_unlock(&dev->mode_config.fb_lock);
|
|
|
+
|
|
|
+ list_del_init(&fb->filp_head);
|
|
|
+
|
|
|
+ /* This will also drop the fpriv->fbs reference. */
|
|
|
drm_framebuffer_remove(fb);
|
|
|
}
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ mutex_unlock(&priv->fbs_lock);
|
|
|
}
|
|
|
|
|
|
/**
|
|
@@ -2660,10 +2806,9 @@ EXPORT_SYMBOL(drm_mode_detachmode_crtc);
|
|
|
|
|
|
/**
|
|
|
* drm_fb_attachmode - Attach a user mode to an connector
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* This attaches a user specified mode to an connector.
|
|
|
* Called by the user via ioctl.
|
|
@@ -2684,7 +2829,7 @@ int drm_mode_attachmode_ioctl(struct drm_device *dev,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
|
|
|
obj = drm_mode_object_find(dev, mode_cmd->connector_id, DRM_MODE_OBJECT_CONNECTOR);
|
|
|
if (!obj) {
|
|
@@ -2708,17 +2853,16 @@ int drm_mode_attachmode_ioctl(struct drm_device *dev,
|
|
|
|
|
|
drm_mode_attachmode(dev, connector, mode);
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
|
|
|
/**
|
|
|
* drm_fb_detachmode - Detach a user specified mode from an connector
|
|
|
- * @inode: inode from the ioctl
|
|
|
- * @filp: file * from the ioctl
|
|
|
- * @cmd: cmd from ioctl
|
|
|
- * @arg: arg from ioctl
|
|
|
+ * @dev: drm device for the ioctl
|
|
|
+ * @data: data pointer for the ioctl
|
|
|
+ * @file_priv: drm file for the ioctl call
|
|
|
*
|
|
|
* Called by the user via ioctl.
|
|
|
*
|
|
@@ -2738,7 +2882,7 @@ int drm_mode_detachmode_ioctl(struct drm_device *dev,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
|
|
|
obj = drm_mode_object_find(dev, mode_cmd->connector_id, DRM_MODE_OBJECT_CONNECTOR);
|
|
|
if (!obj) {
|
|
@@ -2755,7 +2899,7 @@ int drm_mode_detachmode_ioctl(struct drm_device *dev,
|
|
|
|
|
|
ret = drm_mode_detachmode(dev, connector, &mode);
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -3001,7 +3145,7 @@ int drm_mode_getproperty_ioctl(struct drm_device *dev,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
obj = drm_mode_object_find(dev, out_resp->prop_id, DRM_MODE_OBJECT_PROPERTY);
|
|
|
if (!obj) {
|
|
|
ret = -EINVAL;
|
|
@@ -3079,7 +3223,7 @@ int drm_mode_getproperty_ioctl(struct drm_device *dev,
|
|
|
out_resp->count_enum_blobs = blob_count;
|
|
|
}
|
|
|
done:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -3130,7 +3274,7 @@ int drm_mode_getblob_ioctl(struct drm_device *dev,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
obj = drm_mode_object_find(dev, out_resp->blob_id, DRM_MODE_OBJECT_BLOB);
|
|
|
if (!obj) {
|
|
|
ret = -EINVAL;
|
|
@@ -3148,7 +3292,7 @@ int drm_mode_getblob_ioctl(struct drm_device *dev,
|
|
|
out_resp->length = blob->length;
|
|
|
|
|
|
done:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -3290,7 +3434,7 @@ int drm_mode_obj_get_properties_ioctl(struct drm_device *dev, void *data,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
|
|
|
obj = drm_mode_object_find(dev, arg->obj_id, arg->obj_type);
|
|
|
if (!obj) {
|
|
@@ -3327,7 +3471,7 @@ int drm_mode_obj_get_properties_ioctl(struct drm_device *dev, void *data,
|
|
|
}
|
|
|
arg->count_props = props_count;
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -3344,7 +3488,7 @@ int drm_mode_obj_set_property_ioctl(struct drm_device *dev, void *data,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
|
|
|
arg_obj = drm_mode_object_find(dev, arg->obj_id, arg->obj_type);
|
|
|
if (!arg_obj)
|
|
@@ -3382,7 +3526,7 @@ int drm_mode_obj_set_property_ioctl(struct drm_device *dev, void *data,
|
|
|
}
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -3444,7 +3588,7 @@ int drm_mode_gamma_set_ioctl(struct drm_device *dev,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
obj = drm_mode_object_find(dev, crtc_lut->crtc_id, DRM_MODE_OBJECT_CRTC);
|
|
|
if (!obj) {
|
|
|
ret = -EINVAL;
|
|
@@ -3485,7 +3629,7 @@ int drm_mode_gamma_set_ioctl(struct drm_device *dev,
|
|
|
crtc->funcs->gamma_set(crtc, r_base, g_base, b_base, 0, crtc->gamma_size);
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
|
|
|
}
|
|
@@ -3503,7 +3647,7 @@ int drm_mode_gamma_get_ioctl(struct drm_device *dev,
|
|
|
if (!drm_core_check_feature(dev, DRIVER_MODESET))
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_lock_all(dev);
|
|
|
obj = drm_mode_object_find(dev, crtc_lut->crtc_id, DRM_MODE_OBJECT_CRTC);
|
|
|
if (!obj) {
|
|
|
ret = -EINVAL;
|
|
@@ -3536,7 +3680,7 @@ int drm_mode_gamma_get_ioctl(struct drm_device *dev,
|
|
|
goto out;
|
|
|
}
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ drm_modeset_unlock_all(dev);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -3546,7 +3690,7 @@ int drm_mode_page_flip_ioctl(struct drm_device *dev,
|
|
|
struct drm_mode_crtc_page_flip *page_flip = data;
|
|
|
struct drm_mode_object *obj;
|
|
|
struct drm_crtc *crtc;
|
|
|
- struct drm_framebuffer *fb;
|
|
|
+ struct drm_framebuffer *fb = NULL, *old_fb = NULL;
|
|
|
struct drm_pending_vblank_event *e = NULL;
|
|
|
unsigned long flags;
|
|
|
int hdisplay, vdisplay;
|
|
@@ -3556,12 +3700,12 @@ int drm_mode_page_flip_ioctl(struct drm_device *dev,
|
|
|
page_flip->reserved != 0)
|
|
|
return -EINVAL;
|
|
|
|
|
|
- mutex_lock(&dev->mode_config.mutex);
|
|
|
obj = drm_mode_object_find(dev, page_flip->crtc_id, DRM_MODE_OBJECT_CRTC);
|
|
|
if (!obj)
|
|
|
- goto out;
|
|
|
+ return -EINVAL;
|
|
|
crtc = obj_to_crtc(obj);
|
|
|
|
|
|
+ mutex_lock(&crtc->mutex);
|
|
|
if (crtc->fb == NULL) {
|
|
|
/* The framebuffer is currently unbound, presumably
|
|
|
* due to a hotplug event, that userspace has not
|
|
@@ -3574,10 +3718,9 @@ int drm_mode_page_flip_ioctl(struct drm_device *dev,
|
|
|
if (crtc->funcs->page_flip == NULL)
|
|
|
goto out;
|
|
|
|
|
|
- obj = drm_mode_object_find(dev, page_flip->fb_id, DRM_MODE_OBJECT_FB);
|
|
|
- if (!obj)
|
|
|
+ fb = drm_framebuffer_lookup(dev, page_flip->fb_id);
|
|
|
+ if (!fb)
|
|
|
goto out;
|
|
|
- fb = obj_to_fb(obj);
|
|
|
|
|
|
hdisplay = crtc->mode.hdisplay;
|
|
|
vdisplay = crtc->mode.vdisplay;
|
|
@@ -3623,6 +3766,7 @@ int drm_mode_page_flip_ioctl(struct drm_device *dev,
|
|
|
(void (*) (struct drm_pending_event *)) kfree;
|
|
|
}
|
|
|
|
|
|
+ old_fb = crtc->fb;
|
|
|
ret = crtc->funcs->page_flip(crtc, fb, e);
|
|
|
if (ret) {
|
|
|
if (page_flip->flags & DRM_MODE_PAGE_FLIP_EVENT) {
|
|
@@ -3631,10 +3775,20 @@ int drm_mode_page_flip_ioctl(struct drm_device *dev,
|
|
|
spin_unlock_irqrestore(&dev->event_lock, flags);
|
|
|
kfree(e);
|
|
|
}
|
|
|
+ /* Keep the old fb, don't unref it. */
|
|
|
+ old_fb = NULL;
|
|
|
+ } else {
|
|
|
+ /* Unref only the old framebuffer. */
|
|
|
+ fb = NULL;
|
|
|
}
|
|
|
|
|
|
out:
|
|
|
- mutex_unlock(&dev->mode_config.mutex);
|
|
|
+ if (fb)
|
|
|
+ drm_framebuffer_unreference(fb);
|
|
|
+ if (old_fb)
|
|
|
+ drm_framebuffer_unreference(old_fb);
|
|
|
+ mutex_unlock(&crtc->mutex);
|
|
|
+
|
|
|
return ret;
|
|
|
}
|
|
|
|