aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/gpu/drm/drm_gem.c
diff options
context:
space:
mode:
authorChris Wilson <chris@chris-wilson.co.uk>2009-02-11 09:26:28 -0500
committerDave Airlie <airlied@redhat.com>2009-02-19 21:21:10 -0500
commit8d59bae5d9aae10ab230561519bfb97962509bcb (patch)
tree0be3a5e9cbb512e6ec933a8e74f59958c7ee8018 /drivers/gpu/drm/drm_gem.c
parenta35f2e2b83a789e189a501ebd722bc9a1310eb05 (diff)
drm: Do not leak a new reference for flink() on an existing name
The name table should only hold a single reference, so avoid leaking additional references for secondary calls to flink(). Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk> Signed-off-by: Eric Anholt <eric@anholt.net> Signed-off-by: Dave Airlie <airlied@linux.ie>
Diffstat (limited to 'drivers/gpu/drm/drm_gem.c')
-rw-r--r--drivers/gpu/drm/drm_gem.c33
1 files changed, 16 insertions, 17 deletions
diff --git a/drivers/gpu/drm/drm_gem.c b/drivers/gpu/drm/drm_gem.c
index e5a8ebf9a662..5dad6b9d0dec 100644
--- a/drivers/gpu/drm/drm_gem.c
+++ b/drivers/gpu/drm/drm_gem.c
@@ -301,27 +301,25 @@ again:
301 } 301 }
302 302
303 spin_lock(&dev->object_name_lock); 303 spin_lock(&dev->object_name_lock);
304 if (obj->name) { 304 if (!obj->name) {
305 args->name = obj->name; 305 ret = idr_get_new_above(&dev->object_name_idr, obj, 1,
306 &obj->name);
307 args->name = (uint64_t) obj->name;
306 spin_unlock(&dev->object_name_lock); 308 spin_unlock(&dev->object_name_lock);
307 return 0;
308 }
309 ret = idr_get_new_above(&dev->object_name_idr, obj, 1,
310 &obj->name);
311 spin_unlock(&dev->object_name_lock);
312 if (ret == -EAGAIN)
313 goto again;
314 309
315 if (ret != 0) 310 if (ret == -EAGAIN)
316 goto err; 311 goto again;
317 312
318 /* 313 if (ret != 0)
319 * Leave the reference from the lookup around as the 314 goto err;
320 * name table now holds one
321 */
322 args->name = (uint64_t) obj->name;
323 315
324 return 0; 316 /* Allocate a reference for the name table. */
317 drm_gem_object_reference(obj);
318 } else {
319 args->name = (uint64_t) obj->name;
320 spin_unlock(&dev->object_name_lock);
321 ret = 0;
322 }
325 323
326err: 324err:
327 mutex_lock(&dev->struct_mutex); 325 mutex_lock(&dev->struct_mutex);
@@ -452,6 +450,7 @@ drm_gem_object_handle_free(struct kref *kref)
452 spin_lock(&dev->object_name_lock); 450 spin_lock(&dev->object_name_lock);
453 if (obj->name) { 451 if (obj->name) {
454 idr_remove(&dev->object_name_idr, obj->name); 452 idr_remove(&dev->object_name_idr, obj->name);
453 obj->name = 0;
455 spin_unlock(&dev->object_name_lock); 454 spin_unlock(&dev->object_name_lock);
456 /* 455 /*
457 * The object name held a reference to this object, drop 456 * The object name held a reference to this object, drop