Commit f4457ae7 authored by Chris Wilson's avatar Chris Wilson

drm/i915: Prevent leaking of -EIO from i915_wait_request()

Reporting -EIO from i915_wait_request() has proven very troublematic
over the years, with numerous hard-to-reproduce bugs cropping up in the
corner case of where a reset occurs and the code wasn't expecting such
an error.

If the we reset the GPU or have detected a hang and wish to reset the
GPU, the request is forcibly complete and the wait broken. Currently, we
report either -EAGAIN or -EIO in order for the caller to retreat and
restart the wait (if appropriate) after dropping and then reacquiring
the struct_mutex (essential to allow the GPU reset to proceed). However,
if we take the view that the request is complete (no further work will
be done on it by the GPU because it is dead and soon to be reset), then
we can proceed with the task at hand and then drop the struct_mutex
allowing the reset to occur. This transfers the burden of checking
whether it is safe to proceed to the caller, which in all but one
instance it is safe - completely eliminating the source of all spurious
-EIO.

Of note, we only have two API entry points where we expect that
userspace can observe an EIO. First is when submitting an execbuf, if
the GPU is terminally wedged, then the operation cannot succeed and an
-EIO is reported. Secondly, existing userspace uses the throttle ioctl
to detect an already wedged GPU before starting using HW acceleration
(or to confirm that the GPU is wedged after an error condition). So if
the GPU is wedged when the user calls throttle, also report -EIO.

v2: Split more carefully the change to i915_wait_request() and assorted
ABI from the reset handling.
v3: Add a couple of WARN_ON(EIO) to the interruptible modesetting code
so that we don't start to leak EIO there in future (and break our hang
resistant modesetting).
Signed-off-by: default avatarChris Wilson <chris@chris-wilson.co.uk>
Cc: Daniel Vetter <daniel.vetter@ffwll.ch>
Reviewed-by: default avatarDaniel Vetter <daniel.vetter@ffwll.ch>
Link: http://patchwork.freedesktop.org/patch/msgid/1460565315-7748-9-git-send-email-chris@chris-wilson.co.uk
Link: http://patchwork.freedesktop.org/patch/msgid/1460565315-7748-1-git-send-email-chris@chris-wilson.co.uk
parent f7e5838b
...@@ -3088,8 +3088,6 @@ i915_gem_find_active_request(struct intel_engine_cs *engine); ...@@ -3088,8 +3088,6 @@ i915_gem_find_active_request(struct intel_engine_cs *engine);
bool i915_gem_retire_requests(struct drm_device *dev); bool i915_gem_retire_requests(struct drm_device *dev);
void i915_gem_retire_requests_ring(struct intel_engine_cs *engine); void i915_gem_retire_requests_ring(struct intel_engine_cs *engine);
int __must_check i915_gem_check_wedge(struct i915_gpu_error *error,
bool interruptible);
static inline u32 i915_reset_counter(struct i915_gpu_error *error) static inline u32 i915_reset_counter(struct i915_gpu_error *error)
{ {
......
...@@ -206,11 +206,10 @@ i915_gem_object_put_pages_phys(struct drm_i915_gem_object *obj) ...@@ -206,11 +206,10 @@ i915_gem_object_put_pages_phys(struct drm_i915_gem_object *obj)
BUG_ON(obj->madv == __I915_MADV_PURGED); BUG_ON(obj->madv == __I915_MADV_PURGED);
ret = i915_gem_object_set_to_cpu_domain(obj, true); ret = i915_gem_object_set_to_cpu_domain(obj, true);
if (ret) { if (WARN_ON(ret)) {
/* In the event of a disaster, abandon all caches and /* In the event of a disaster, abandon all caches and
* hope for the best. * hope for the best.
*/ */
WARN_ON(ret != -EIO);
obj->base.read_domains = obj->base.write_domain = I915_GEM_DOMAIN_CPU; obj->base.read_domains = obj->base.write_domain = I915_GEM_DOMAIN_CPU;
} }
...@@ -1105,15 +1104,13 @@ i915_gem_pwrite_ioctl(struct drm_device *dev, void *data, ...@@ -1105,15 +1104,13 @@ i915_gem_pwrite_ioctl(struct drm_device *dev, void *data,
return ret; return ret;
} }
int static int
i915_gem_check_wedge(struct i915_gpu_error *error, i915_gem_check_wedge(unsigned reset_counter, bool interruptible)
bool interruptible)
{ {
if (i915_reset_in_progress_or_wedged(error)) { if (__i915_terminally_wedged(reset_counter))
/* Recovery complete, but the reset failed ... */
if (i915_terminally_wedged(error))
return -EIO; return -EIO;
if (__i915_reset_in_progress(reset_counter)) {
/* Non-interruptible callers can't handle -EAGAIN, hence return /* Non-interruptible callers can't handle -EAGAIN, hence return
* -EIO unconditionally for these. */ * -EIO unconditionally for these. */
if (!interruptible) if (!interruptible)
...@@ -1287,13 +1284,14 @@ int __i915_wait_request(struct drm_i915_gem_request *req, ...@@ -1287,13 +1284,14 @@ int __i915_wait_request(struct drm_i915_gem_request *req,
prepare_to_wait(&engine->irq_queue, &wait, state); prepare_to_wait(&engine->irq_queue, &wait, state);
/* We need to check whether any gpu reset happened in between /* We need to check whether any gpu reset happened in between
* the caller grabbing the seqno and now ... */ * the request being submitted and now. If a reset has occurred,
* the request is effectively complete (we either are in the
* process of or have discarded the rendering and completely
* reset the GPU. The results of the request are lost and we
* are free to continue on with the original operation.
*/
if (req->reset_counter != i915_reset_counter(&dev_priv->gpu_error)) { if (req->reset_counter != i915_reset_counter(&dev_priv->gpu_error)) {
/* ... but upgrade the -EAGAIN to an -EIO if the gpu ret = 0;
* is truely gone. */
ret = i915_gem_check_wedge(&dev_priv->gpu_error, interruptible);
if (ret == 0)
ret = -EAGAIN;
break; break;
} }
...@@ -2154,11 +2152,10 @@ i915_gem_object_put_pages_gtt(struct drm_i915_gem_object *obj) ...@@ -2154,11 +2152,10 @@ i915_gem_object_put_pages_gtt(struct drm_i915_gem_object *obj)
BUG_ON(obj->madv == __I915_MADV_PURGED); BUG_ON(obj->madv == __I915_MADV_PURGED);
ret = i915_gem_object_set_to_cpu_domain(obj, true); ret = i915_gem_object_set_to_cpu_domain(obj, true);
if (ret) { if (WARN_ON(ret)) {
/* In the event of a disaster, abandon all caches and /* In the event of a disaster, abandon all caches and
* hope for the best. * hope for the best.
*/ */
WARN_ON(ret != -EIO);
i915_gem_clflush_object(obj, true); i915_gem_clflush_object(obj, true);
obj->base.read_domains = obj->base.write_domain = I915_GEM_DOMAIN_CPU; obj->base.read_domains = obj->base.write_domain = I915_GEM_DOMAIN_CPU;
} }
...@@ -2729,8 +2726,11 @@ __i915_gem_request_alloc(struct intel_engine_cs *engine, ...@@ -2729,8 +2726,11 @@ __i915_gem_request_alloc(struct intel_engine_cs *engine,
*req_out = NULL; *req_out = NULL;
ret = i915_gem_check_wedge(&dev_priv->gpu_error, /* ABI: Before userspace accesses the GPU (e.g. execbuffer), report
dev_priv->mm.interruptible); * EIO if the GPU is already wedged, or EAGAIN to drop the struct_mutex
* and restart.
*/
ret = i915_gem_check_wedge(reset_counter, dev_priv->mm.interruptible);
if (ret) if (ret)
return ret; return ret;
...@@ -4165,9 +4165,9 @@ i915_gem_ring_throttle(struct drm_device *dev, struct drm_file *file) ...@@ -4165,9 +4165,9 @@ i915_gem_ring_throttle(struct drm_device *dev, struct drm_file *file)
if (ret) if (ret)
return ret; return ret;
ret = i915_gem_check_wedge(&dev_priv->gpu_error, false); /* ABI: return -EIO if already wedged */
if (ret) if (i915_terminally_wedged(&dev_priv->gpu_error))
return ret; return -EIO;
spin_lock(&file_priv->mm.lock); spin_lock(&file_priv->mm.lock);
list_for_each_entry(request, &file_priv->mm.request_list, client_list) { list_for_each_entry(request, &file_priv->mm.request_list, client_list) {
......
...@@ -112,10 +112,8 @@ static void cancel_userptr(struct work_struct *work) ...@@ -112,10 +112,8 @@ static void cancel_userptr(struct work_struct *work)
was_interruptible = dev_priv->mm.interruptible; was_interruptible = dev_priv->mm.interruptible;
dev_priv->mm.interruptible = false; dev_priv->mm.interruptible = false;
list_for_each_entry_safe(vma, tmp, &obj->vma_list, obj_link) { list_for_each_entry_safe(vma, tmp, &obj->vma_list, obj_link)
int ret = i915_vma_unbind(vma); WARN_ON(i915_vma_unbind(vma));
WARN_ON(ret && ret != -EIO);
}
WARN_ON(i915_gem_object_put_pages(obj)); WARN_ON(i915_gem_object_put_pages(obj));
dev_priv->mm.interruptible = was_interruptible; dev_priv->mm.interruptible = was_interruptible;
......
...@@ -13436,11 +13436,9 @@ static int intel_atomic_prepare_commit(struct drm_device *dev, ...@@ -13436,11 +13436,9 @@ static int intel_atomic_prepare_commit(struct drm_device *dev,
ret = __i915_wait_request(intel_plane_state->wait_req, ret = __i915_wait_request(intel_plane_state->wait_req,
true, NULL, NULL); true, NULL, NULL);
/* Swallow -EIO errors to allow updates during hw lockup. */
if (ret == -EIO)
ret = 0;
if (ret) { if (ret) {
/* Any hang should be swallowed by the wait */
WARN_ON(ret == -EIO);
mutex_lock(&dev->struct_mutex); mutex_lock(&dev->struct_mutex);
drm_atomic_helper_cleanup_planes(dev, state); drm_atomic_helper_cleanup_planes(dev, state);
mutex_unlock(&dev->struct_mutex); mutex_unlock(&dev->struct_mutex);
...@@ -13792,11 +13790,12 @@ intel_prepare_plane_fb(struct drm_plane *plane, ...@@ -13792,11 +13790,12 @@ intel_prepare_plane_fb(struct drm_plane *plane,
*/ */
if (needs_modeset(crtc_state)) if (needs_modeset(crtc_state))
ret = i915_gem_object_wait_rendering(old_obj, true); ret = i915_gem_object_wait_rendering(old_obj, true);
if (ret) {
/* Swallow -EIO errors to allow updates during hw lockup. */ /* GPU hangs should have been swallowed by the wait */
if (ret && ret != -EIO) WARN_ON(ret == -EIO);
return ret; return ret;
} }
}
/* For framebuffer backed by dmabuf, wait for fence */ /* For framebuffer backed by dmabuf, wait for fence */
if (obj && obj->base.dma_buf) { if (obj && obj->base.dma_buf) {
......
...@@ -1048,7 +1048,7 @@ void intel_logical_ring_stop(struct intel_engine_cs *engine) ...@@ -1048,7 +1048,7 @@ void intel_logical_ring_stop(struct intel_engine_cs *engine)
return; return;
ret = intel_engine_idle(engine); ret = intel_engine_idle(engine);
if (ret && !i915_reset_in_progress_or_wedged(&dev_priv->gpu_error)) if (ret)
DRM_ERROR("failed to quiesce %s whilst cleaning up: %d\n", DRM_ERROR("failed to quiesce %s whilst cleaning up: %d\n",
engine->name, ret); engine->name, ret);
......
...@@ -3184,8 +3184,7 @@ intel_stop_engine(struct intel_engine_cs *engine) ...@@ -3184,8 +3184,7 @@ intel_stop_engine(struct intel_engine_cs *engine)
return; return;
ret = intel_engine_idle(engine); ret = intel_engine_idle(engine);
if (ret && if (ret)
!i915_reset_in_progress_or_wedged(&to_i915(engine->dev)->gpu_error))
DRM_ERROR("failed to quiesce %s whilst cleaning up: %d\n", DRM_ERROR("failed to quiesce %s whilst cleaning up: %d\n",
engine->name, ret); engine->name, ret);
......
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