diff options
author | Chris Wilson <chris@chris-wilson.co.uk> | 2010-10-17 04:45:41 -0400 |
---|---|---|
committer | Chris Wilson <chris@chris-wilson.co.uk> | 2010-10-19 04:20:23 -0400 |
commit | 1d7cfea152cae6159aa30ceae38c3eaf13ea083c (patch) | |
tree | f9d8b2536555c3a5af75d461f16d7374af988cf8 | |
parent | 139d363bcf2d995a72694ddd2b8665af6cb7fb54 (diff) |
drm/i915: Do interrupible mutex lock first to avoid locking for unreference
One of the primarily consumers of the i915 driver is X, a large signal
driven application. Frequently when writing into the buffers, there is a
pending signal which causes us not to take the interruptible lock but
then we need to take that same lock around the object unreference. By
rearranging the code to do the interruptible lock as the first check, we
can avoid the frequent additional locking around the unreference.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
-rw-r--r-- | drivers/gpu/drm/i915/i915_gem.c | 206 |
1 files changed, 93 insertions, 113 deletions
diff --git a/drivers/gpu/drm/i915/i915_gem.c b/drivers/gpu/drm/i915/i915_gem.c index efc6a4e3b1d2..34a07fc20513 100644 --- a/drivers/gpu/drm/i915/i915_gem.c +++ b/drivers/gpu/drm/i915/i915_gem.c | |||
@@ -547,16 +547,16 @@ i915_gem_pread_ioctl(struct drm_device *dev, void *data, | |||
547 | struct drm_i915_gem_object *obj_priv; | 547 | struct drm_i915_gem_object *obj_priv; |
548 | int ret = 0; | 548 | int ret = 0; |
549 | 549 | ||
550 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
551 | if (obj == NULL) | ||
552 | return -ENOENT; | ||
553 | obj_priv = to_intel_bo(obj); | ||
554 | |||
555 | ret = i915_mutex_lock_interruptible(dev); | 550 | ret = i915_mutex_lock_interruptible(dev); |
556 | if (ret) { | 551 | if (ret) |
557 | drm_gem_object_unreference_unlocked(obj); | ||
558 | return ret; | 552 | return ret; |
553 | |||
554 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
555 | if (obj == NULL) { | ||
556 | ret = -ENOENT; | ||
557 | goto unlock; | ||
559 | } | 558 | } |
559 | obj_priv = to_intel_bo(obj); | ||
560 | 560 | ||
561 | /* Bounds check source. */ | 561 | /* Bounds check source. */ |
562 | if (args->offset > obj->size || args->size > obj->size - args->offset) { | 562 | if (args->offset > obj->size || args->size > obj->size - args->offset) { |
@@ -601,6 +601,7 @@ out_put: | |||
601 | i915_gem_object_put_pages(obj); | 601 | i915_gem_object_put_pages(obj); |
602 | out: | 602 | out: |
603 | drm_gem_object_unreference(obj); | 603 | drm_gem_object_unreference(obj); |
604 | unlock: | ||
604 | mutex_unlock(&dev->struct_mutex); | 605 | mutex_unlock(&dev->struct_mutex); |
605 | return ret; | 606 | return ret; |
606 | } | 607 | } |
@@ -982,16 +983,17 @@ i915_gem_pwrite_ioctl(struct drm_device *dev, void *data, | |||
982 | struct drm_i915_gem_object *obj_priv; | 983 | struct drm_i915_gem_object *obj_priv; |
983 | int ret = 0; | 984 | int ret = 0; |
984 | 985 | ||
985 | obj = drm_gem_object_lookup(dev, file, args->handle); | ||
986 | if (obj == NULL) | ||
987 | return -ENOENT; | ||
988 | obj_priv = to_intel_bo(obj); | ||
989 | |||
990 | ret = i915_mutex_lock_interruptible(dev); | 986 | ret = i915_mutex_lock_interruptible(dev); |
991 | if (ret) { | 987 | if (ret) |
992 | drm_gem_object_unreference_unlocked(obj); | ||
993 | return ret; | 988 | return ret; |
989 | |||
990 | obj = drm_gem_object_lookup(dev, file, args->handle); | ||
991 | if (obj == NULL) { | ||
992 | ret = -ENOENT; | ||
993 | goto unlock; | ||
994 | } | 994 | } |
995 | obj_priv = to_intel_bo(obj); | ||
996 | |||
995 | 997 | ||
996 | /* Bounds check destination. */ | 998 | /* Bounds check destination. */ |
997 | if (args->offset > obj->size || args->size > obj->size - args->offset) { | 999 | if (args->offset > obj->size || args->size > obj->size - args->offset) { |
@@ -1062,6 +1064,7 @@ out_put: | |||
1062 | 1064 | ||
1063 | out: | 1065 | out: |
1064 | drm_gem_object_unreference(obj); | 1066 | drm_gem_object_unreference(obj); |
1067 | unlock: | ||
1065 | mutex_unlock(&dev->struct_mutex); | 1068 | mutex_unlock(&dev->struct_mutex); |
1066 | return ret; | 1069 | return ret; |
1067 | } | 1070 | } |
@@ -1098,16 +1101,16 @@ i915_gem_set_domain_ioctl(struct drm_device *dev, void *data, | |||
1098 | if (write_domain != 0 && read_domains != write_domain) | 1101 | if (write_domain != 0 && read_domains != write_domain) |
1099 | return -EINVAL; | 1102 | return -EINVAL; |
1100 | 1103 | ||
1101 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
1102 | if (obj == NULL) | ||
1103 | return -ENOENT; | ||
1104 | obj_priv = to_intel_bo(obj); | ||
1105 | |||
1106 | ret = i915_mutex_lock_interruptible(dev); | 1104 | ret = i915_mutex_lock_interruptible(dev); |
1107 | if (ret) { | 1105 | if (ret) |
1108 | drm_gem_object_unreference_unlocked(obj); | ||
1109 | return ret; | 1106 | return ret; |
1107 | |||
1108 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
1109 | if (obj == NULL) { | ||
1110 | ret = -ENOENT; | ||
1111 | goto unlock; | ||
1110 | } | 1112 | } |
1113 | obj_priv = to_intel_bo(obj); | ||
1111 | 1114 | ||
1112 | intel_mark_busy(dev, obj); | 1115 | intel_mark_busy(dev, obj); |
1113 | 1116 | ||
@@ -1139,6 +1142,7 @@ i915_gem_set_domain_ioctl(struct drm_device *dev, void *data, | |||
1139 | list_move_tail(&obj_priv->list, &dev_priv->mm.inactive_list); | 1142 | list_move_tail(&obj_priv->list, &dev_priv->mm.inactive_list); |
1140 | 1143 | ||
1141 | drm_gem_object_unreference(obj); | 1144 | drm_gem_object_unreference(obj); |
1145 | unlock: | ||
1142 | mutex_unlock(&dev->struct_mutex); | 1146 | mutex_unlock(&dev->struct_mutex); |
1143 | return ret; | 1147 | return ret; |
1144 | } | 1148 | } |
@@ -1157,14 +1161,14 @@ i915_gem_sw_finish_ioctl(struct drm_device *dev, void *data, | |||
1157 | if (!(dev->driver->driver_features & DRIVER_GEM)) | 1161 | if (!(dev->driver->driver_features & DRIVER_GEM)) |
1158 | return -ENODEV; | 1162 | return -ENODEV; |
1159 | 1163 | ||
1160 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
1161 | if (obj == NULL) | ||
1162 | return -ENOENT; | ||
1163 | |||
1164 | ret = i915_mutex_lock_interruptible(dev); | 1164 | ret = i915_mutex_lock_interruptible(dev); |
1165 | if (ret) { | 1165 | if (ret) |
1166 | drm_gem_object_unreference_unlocked(obj); | ||
1167 | return ret; | 1166 | return ret; |
1167 | |||
1168 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
1169 | if (obj == NULL) { | ||
1170 | ret = -ENOENT; | ||
1171 | goto unlock; | ||
1168 | } | 1172 | } |
1169 | 1173 | ||
1170 | /* Pinned buffers may be scanout, so flush the cache */ | 1174 | /* Pinned buffers may be scanout, so flush the cache */ |
@@ -1172,6 +1176,7 @@ i915_gem_sw_finish_ioctl(struct drm_device *dev, void *data, | |||
1172 | i915_gem_object_flush_cpu_write_domain(obj); | 1176 | i915_gem_object_flush_cpu_write_domain(obj); |
1173 | 1177 | ||
1174 | drm_gem_object_unreference(obj); | 1178 | drm_gem_object_unreference(obj); |
1179 | unlock: | ||
1175 | mutex_unlock(&dev->struct_mutex); | 1180 | mutex_unlock(&dev->struct_mutex); |
1176 | return ret; | 1181 | return ret; |
1177 | } | 1182 | } |
@@ -1469,33 +1474,27 @@ i915_gem_mmap_gtt_ioctl(struct drm_device *dev, void *data, | |||
1469 | if (!(dev->driver->driver_features & DRIVER_GEM)) | 1474 | if (!(dev->driver->driver_features & DRIVER_GEM)) |
1470 | return -ENODEV; | 1475 | return -ENODEV; |
1471 | 1476 | ||
1472 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
1473 | if (obj == NULL) | ||
1474 | return -ENOENT; | ||
1475 | |||
1476 | ret = i915_mutex_lock_interruptible(dev); | 1477 | ret = i915_mutex_lock_interruptible(dev); |
1477 | if (ret) { | 1478 | if (ret) |
1478 | drm_gem_object_unreference_unlocked(obj); | ||
1479 | return ret; | 1479 | return ret; |
1480 | } | ||
1481 | 1480 | ||
1481 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
1482 | if (obj == NULL) { | ||
1483 | ret = -ENOENT; | ||
1484 | goto unlock; | ||
1485 | } | ||
1482 | obj_priv = to_intel_bo(obj); | 1486 | obj_priv = to_intel_bo(obj); |
1483 | 1487 | ||
1484 | if (obj_priv->madv != I915_MADV_WILLNEED) { | 1488 | if (obj_priv->madv != I915_MADV_WILLNEED) { |
1485 | DRM_ERROR("Attempting to mmap a purgeable buffer\n"); | 1489 | DRM_ERROR("Attempting to mmap a purgeable buffer\n"); |
1486 | drm_gem_object_unreference(obj); | 1490 | ret = -EINVAL; |
1487 | mutex_unlock(&dev->struct_mutex); | 1491 | goto out; |
1488 | return -EINVAL; | ||
1489 | } | 1492 | } |
1490 | 1493 | ||
1491 | |||
1492 | if (!obj_priv->mmap_offset) { | 1494 | if (!obj_priv->mmap_offset) { |
1493 | ret = i915_gem_create_mmap_offset(obj); | 1495 | ret = i915_gem_create_mmap_offset(obj); |
1494 | if (ret) { | 1496 | if (ret) |
1495 | drm_gem_object_unreference(obj); | 1497 | goto out; |
1496 | mutex_unlock(&dev->struct_mutex); | ||
1497 | return ret; | ||
1498 | } | ||
1499 | } | 1498 | } |
1500 | 1499 | ||
1501 | args->offset = obj_priv->mmap_offset; | 1500 | args->offset = obj_priv->mmap_offset; |
@@ -1506,17 +1505,15 @@ i915_gem_mmap_gtt_ioctl(struct drm_device *dev, void *data, | |||
1506 | */ | 1505 | */ |
1507 | if (!obj_priv->agp_mem) { | 1506 | if (!obj_priv->agp_mem) { |
1508 | ret = i915_gem_object_bind_to_gtt(obj, 0); | 1507 | ret = i915_gem_object_bind_to_gtt(obj, 0); |
1509 | if (ret) { | 1508 | if (ret) |
1510 | drm_gem_object_unreference(obj); | 1509 | goto out; |
1511 | mutex_unlock(&dev->struct_mutex); | ||
1512 | return ret; | ||
1513 | } | ||
1514 | } | 1510 | } |
1515 | 1511 | ||
1512 | out: | ||
1516 | drm_gem_object_unreference(obj); | 1513 | drm_gem_object_unreference(obj); |
1514 | unlock: | ||
1517 | mutex_unlock(&dev->struct_mutex); | 1515 | mutex_unlock(&dev->struct_mutex); |
1518 | 1516 | return ret; | |
1519 | return 0; | ||
1520 | } | 1517 | } |
1521 | 1518 | ||
1522 | static void | 1519 | static void |
@@ -4100,44 +4097,36 @@ i915_gem_pin_ioctl(struct drm_device *dev, void *data, | |||
4100 | struct drm_i915_gem_object *obj_priv; | 4097 | struct drm_i915_gem_object *obj_priv; |
4101 | int ret; | 4098 | int ret; |
4102 | 4099 | ||
4100 | ret = i915_mutex_lock_interruptible(dev); | ||
4101 | if (ret) | ||
4102 | return ret; | ||
4103 | |||
4103 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | 4104 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); |
4104 | if (obj == NULL) { | 4105 | if (obj == NULL) { |
4105 | DRM_ERROR("Bad handle in i915_gem_pin_ioctl(): %d\n", | 4106 | ret = -ENOENT; |
4106 | args->handle); | 4107 | goto unlock; |
4107 | return -ENOENT; | ||
4108 | } | 4108 | } |
4109 | obj_priv = to_intel_bo(obj); | 4109 | obj_priv = to_intel_bo(obj); |
4110 | 4110 | ||
4111 | ret = i915_mutex_lock_interruptible(dev); | ||
4112 | if (ret) { | ||
4113 | drm_gem_object_unreference_unlocked(obj); | ||
4114 | return ret; | ||
4115 | } | ||
4116 | |||
4117 | if (obj_priv->madv != I915_MADV_WILLNEED) { | 4111 | if (obj_priv->madv != I915_MADV_WILLNEED) { |
4118 | DRM_ERROR("Attempting to pin a purgeable buffer\n"); | 4112 | DRM_ERROR("Attempting to pin a purgeable buffer\n"); |
4119 | drm_gem_object_unreference(obj); | 4113 | ret = -EINVAL; |
4120 | mutex_unlock(&dev->struct_mutex); | 4114 | goto out; |
4121 | return -EINVAL; | ||
4122 | } | 4115 | } |
4123 | 4116 | ||
4124 | if (obj_priv->pin_filp != NULL && obj_priv->pin_filp != file_priv) { | 4117 | if (obj_priv->pin_filp != NULL && obj_priv->pin_filp != file_priv) { |
4125 | DRM_ERROR("Already pinned in i915_gem_pin_ioctl(): %d\n", | 4118 | DRM_ERROR("Already pinned in i915_gem_pin_ioctl(): %d\n", |
4126 | args->handle); | 4119 | args->handle); |
4127 | drm_gem_object_unreference(obj); | 4120 | ret = -EINVAL; |
4128 | mutex_unlock(&dev->struct_mutex); | 4121 | goto out; |
4129 | return -EINVAL; | ||
4130 | } | 4122 | } |
4131 | 4123 | ||
4132 | obj_priv->user_pin_count++; | 4124 | obj_priv->user_pin_count++; |
4133 | obj_priv->pin_filp = file_priv; | 4125 | obj_priv->pin_filp = file_priv; |
4134 | if (obj_priv->user_pin_count == 1) { | 4126 | if (obj_priv->user_pin_count == 1) { |
4135 | ret = i915_gem_object_pin(obj, args->alignment); | 4127 | ret = i915_gem_object_pin(obj, args->alignment); |
4136 | if (ret != 0) { | 4128 | if (ret) |
4137 | drm_gem_object_unreference(obj); | 4129 | goto out; |
4138 | mutex_unlock(&dev->struct_mutex); | ||
4139 | return ret; | ||
4140 | } | ||
4141 | } | 4130 | } |
4142 | 4131 | ||
4143 | /* XXX - flush the CPU caches for pinned objects | 4132 | /* XXX - flush the CPU caches for pinned objects |
@@ -4145,10 +4134,11 @@ i915_gem_pin_ioctl(struct drm_device *dev, void *data, | |||
4145 | */ | 4134 | */ |
4146 | i915_gem_object_flush_cpu_write_domain(obj); | 4135 | i915_gem_object_flush_cpu_write_domain(obj); |
4147 | args->offset = obj_priv->gtt_offset; | 4136 | args->offset = obj_priv->gtt_offset; |
4137 | out: | ||
4148 | drm_gem_object_unreference(obj); | 4138 | drm_gem_object_unreference(obj); |
4139 | unlock: | ||
4149 | mutex_unlock(&dev->struct_mutex); | 4140 | mutex_unlock(&dev->struct_mutex); |
4150 | 4141 | return ret; | |
4151 | return 0; | ||
4152 | } | 4142 | } |
4153 | 4143 | ||
4154 | int | 4144 | int |
@@ -4160,27 +4150,22 @@ i915_gem_unpin_ioctl(struct drm_device *dev, void *data, | |||
4160 | struct drm_i915_gem_object *obj_priv; | 4150 | struct drm_i915_gem_object *obj_priv; |
4161 | int ret; | 4151 | int ret; |
4162 | 4152 | ||
4153 | ret = i915_mutex_lock_interruptible(dev); | ||
4154 | if (ret) | ||
4155 | return ret; | ||
4156 | |||
4163 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | 4157 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); |
4164 | if (obj == NULL) { | 4158 | if (obj == NULL) { |
4165 | DRM_ERROR("Bad handle in i915_gem_unpin_ioctl(): %d\n", | 4159 | ret = -ENOENT; |
4166 | args->handle); | 4160 | goto unlock; |
4167 | return -ENOENT; | ||
4168 | } | 4161 | } |
4169 | |||
4170 | obj_priv = to_intel_bo(obj); | 4162 | obj_priv = to_intel_bo(obj); |
4171 | 4163 | ||
4172 | ret = i915_mutex_lock_interruptible(dev); | ||
4173 | if (ret) { | ||
4174 | drm_gem_object_unreference_unlocked(obj); | ||
4175 | return ret; | ||
4176 | } | ||
4177 | |||
4178 | if (obj_priv->pin_filp != file_priv) { | 4164 | if (obj_priv->pin_filp != file_priv) { |
4179 | DRM_ERROR("Not pinned by caller in i915_gem_pin_ioctl(): %d\n", | 4165 | DRM_ERROR("Not pinned by caller in i915_gem_pin_ioctl(): %d\n", |
4180 | args->handle); | 4166 | args->handle); |
4181 | drm_gem_object_unreference(obj); | 4167 | ret = -EINVAL; |
4182 | mutex_unlock(&dev->struct_mutex); | 4168 | goto out; |
4183 | return -EINVAL; | ||
4184 | } | 4169 | } |
4185 | obj_priv->user_pin_count--; | 4170 | obj_priv->user_pin_count--; |
4186 | if (obj_priv->user_pin_count == 0) { | 4171 | if (obj_priv->user_pin_count == 0) { |
@@ -4188,9 +4173,11 @@ i915_gem_unpin_ioctl(struct drm_device *dev, void *data, | |||
4188 | i915_gem_object_unpin(obj); | 4173 | i915_gem_object_unpin(obj); |
4189 | } | 4174 | } |
4190 | 4175 | ||
4176 | out: | ||
4191 | drm_gem_object_unreference(obj); | 4177 | drm_gem_object_unreference(obj); |
4178 | unlock: | ||
4192 | mutex_unlock(&dev->struct_mutex); | 4179 | mutex_unlock(&dev->struct_mutex); |
4193 | return 0; | 4180 | return ret; |
4194 | } | 4181 | } |
4195 | 4182 | ||
4196 | int | 4183 | int |
@@ -4202,25 +4189,22 @@ i915_gem_busy_ioctl(struct drm_device *dev, void *data, | |||
4202 | struct drm_i915_gem_object *obj_priv; | 4189 | struct drm_i915_gem_object *obj_priv; |
4203 | int ret; | 4190 | int ret; |
4204 | 4191 | ||
4205 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
4206 | if (obj == NULL) { | ||
4207 | DRM_ERROR("Bad handle in i915_gem_busy_ioctl(): %d\n", | ||
4208 | args->handle); | ||
4209 | return -ENOENT; | ||
4210 | } | ||
4211 | |||
4212 | ret = i915_mutex_lock_interruptible(dev); | 4192 | ret = i915_mutex_lock_interruptible(dev); |
4213 | if (ret) { | 4193 | if (ret) |
4214 | drm_gem_object_unreference_unlocked(obj); | ||
4215 | return ret; | 4194 | return ret; |
4195 | |||
4196 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | ||
4197 | if (obj == NULL) { | ||
4198 | ret = -ENOENT; | ||
4199 | goto unlock; | ||
4216 | } | 4200 | } |
4201 | obj_priv = to_intel_bo(obj); | ||
4217 | 4202 | ||
4218 | /* Count all active objects as busy, even if they are currently not used | 4203 | /* Count all active objects as busy, even if they are currently not used |
4219 | * by the gpu. Users of this interface expect objects to eventually | 4204 | * by the gpu. Users of this interface expect objects to eventually |
4220 | * become non-busy without any further actions, therefore emit any | 4205 | * become non-busy without any further actions, therefore emit any |
4221 | * necessary flushes here. | 4206 | * necessary flushes here. |
4222 | */ | 4207 | */ |
4223 | obj_priv = to_intel_bo(obj); | ||
4224 | args->busy = obj_priv->active; | 4208 | args->busy = obj_priv->active; |
4225 | if (args->busy) { | 4209 | if (args->busy) { |
4226 | /* Unconditionally flush objects, even when the gpu still uses this | 4210 | /* Unconditionally flush objects, even when the gpu still uses this |
@@ -4244,8 +4228,9 @@ i915_gem_busy_ioctl(struct drm_device *dev, void *data, | |||
4244 | } | 4228 | } |
4245 | 4229 | ||
4246 | drm_gem_object_unreference(obj); | 4230 | drm_gem_object_unreference(obj); |
4231 | unlock: | ||
4247 | mutex_unlock(&dev->struct_mutex); | 4232 | mutex_unlock(&dev->struct_mutex); |
4248 | return 0; | 4233 | return ret; |
4249 | } | 4234 | } |
4250 | 4235 | ||
4251 | int | 4236 | int |
@@ -4272,26 +4257,20 @@ i915_gem_madvise_ioctl(struct drm_device *dev, void *data, | |||
4272 | return -EINVAL; | 4257 | return -EINVAL; |
4273 | } | 4258 | } |
4274 | 4259 | ||
4260 | ret = i915_mutex_lock_interruptible(dev); | ||
4261 | if (ret) | ||
4262 | return ret; | ||
4263 | |||
4275 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); | 4264 | obj = drm_gem_object_lookup(dev, file_priv, args->handle); |
4276 | if (obj == NULL) { | 4265 | if (obj == NULL) { |
4277 | DRM_ERROR("Bad handle in i915_gem_madvise_ioctl(): %d\n", | 4266 | ret = -ENOENT; |
4278 | args->handle); | 4267 | goto unlock; |
4279 | return -ENOENT; | ||
4280 | } | 4268 | } |
4281 | obj_priv = to_intel_bo(obj); | 4269 | obj_priv = to_intel_bo(obj); |
4282 | 4270 | ||
4283 | ret = i915_mutex_lock_interruptible(dev); | ||
4284 | if (ret) { | ||
4285 | drm_gem_object_unreference_unlocked(obj); | ||
4286 | return ret; | ||
4287 | } | ||
4288 | |||
4289 | if (obj_priv->pin_count) { | 4271 | if (obj_priv->pin_count) { |
4290 | drm_gem_object_unreference(obj); | 4272 | ret = -EINVAL; |
4291 | mutex_unlock(&dev->struct_mutex); | 4273 | goto out; |
4292 | |||
4293 | DRM_ERROR("Attempted i915_gem_madvise_ioctl() on a pinned object\n"); | ||
4294 | return -EINVAL; | ||
4295 | } | 4274 | } |
4296 | 4275 | ||
4297 | if (obj_priv->madv != __I915_MADV_PURGED) | 4276 | if (obj_priv->madv != __I915_MADV_PURGED) |
@@ -4304,10 +4283,11 @@ i915_gem_madvise_ioctl(struct drm_device *dev, void *data, | |||
4304 | 4283 | ||
4305 | args->retained = obj_priv->madv != __I915_MADV_PURGED; | 4284 | args->retained = obj_priv->madv != __I915_MADV_PURGED; |
4306 | 4285 | ||
4286 | out: | ||
4307 | drm_gem_object_unreference(obj); | 4287 | drm_gem_object_unreference(obj); |
4288 | unlock: | ||
4308 | mutex_unlock(&dev->struct_mutex); | 4289 | mutex_unlock(&dev->struct_mutex); |
4309 | 4290 | return ret; | |
4310 | return 0; | ||
4311 | } | 4291 | } |
4312 | 4292 | ||
4313 | struct drm_gem_object * i915_gem_alloc_object(struct drm_device *dev, | 4293 | struct drm_gem_object * i915_gem_alloc_object(struct drm_device *dev, |