Commit be62acb4 authored by Mika Kuoppala's avatar Mika Kuoppala Committed by Daniel Vetter

drm/i915: ban badly behaving contexts

Now when we have mechanism in place to track which context
was guilty of hanging the gpu, it is possible to punish
for bad behaviour.

If context has recently submitted a faulty batchbuffers guilty of
gpu hang and submits another batch which hangs gpu in quick
succession, ban it permanently. If ctx is banned, no more
batchbuffers will be queued for execution.

There is no need for global wedge machinery anymore and
it would be unwise to wedge the whole gpu if we have multiple
hanging batches queued for execution. Instead just ban
the guilty ones and carry on.

v2: Store guilty ban status bool in gpu_error instead of pointers
    that might become danling before hang is declared.

v3: Use return value for banned status instead of stashing state
    into gpu_error (Chris Wilson)

v4: - rebase on top of fixed hang stats api
    - add define for ban period
    - rename commit and improve commit msg

v5: - rely context banning instead of wedging the gpu
    - beautification and fix for ban calculation (Chris)
Signed-off-by: default avatarMika Kuoppala <mika.kuoppala@intel.com>
Reviewed-by: default avatarChris Wilson <chris@chris-wilson.co.uk>
Signed-off-by: default avatarDaniel Vetter <daniel.vetter@ffwll.ch>
parent bf13e81b
...@@ -719,24 +719,19 @@ int i915_reset(struct drm_device *dev) ...@@ -719,24 +719,19 @@ int i915_reset(struct drm_device *dev)
simulated = dev_priv->gpu_error.stop_rings != 0; simulated = dev_priv->gpu_error.stop_rings != 0;
if (!simulated && get_seconds() - dev_priv->gpu_error.last_reset < 5) { ret = intel_gpu_reset(dev);
DRM_ERROR("GPU hanging too fast, declaring wedged!\n");
ret = -ENODEV; /* Also reset the gpu hangman. */
} else { if (simulated) {
ret = intel_gpu_reset(dev); DRM_INFO("Simulated gpu hang, resetting stop_rings\n");
dev_priv->gpu_error.stop_rings = 0;
/* Also reset the gpu hangman. */ if (ret == -ENODEV) {
if (simulated) { DRM_ERROR("Reset not implemented, but ignoring "
DRM_INFO("Simulated gpu hang, resetting stop_rings\n"); "error for simulated gpu hangs\n");
dev_priv->gpu_error.stop_rings = 0; ret = 0;
if (ret == -ENODEV) { }
DRM_ERROR("Reset not implemented, but ignoring "
"error for simulated gpu hangs\n");
ret = 0;
}
} else
dev_priv->gpu_error.last_reset = get_seconds();
} }
if (ret) { if (ret) {
DRM_ERROR("Failed to reset chip.\n"); DRM_ERROR("Failed to reset chip.\n");
mutex_unlock(&dev->struct_mutex); mutex_unlock(&dev->struct_mutex);
......
...@@ -586,6 +586,12 @@ struct i915_ctx_hang_stats { ...@@ -586,6 +586,12 @@ struct i915_ctx_hang_stats {
/* This context had batch active when hang was declared */ /* This context had batch active when hang was declared */
unsigned batch_active; unsigned batch_active;
/* Time when this context was last blamed for a GPU reset */
unsigned long guilty_ts;
/* This context is banned to submit more work */
bool banned;
}; };
/* This must match up with the value previously used for execbuf2.rsvd1. */ /* This must match up with the value previously used for execbuf2.rsvd1. */
...@@ -987,6 +993,9 @@ struct i915_gpu_error { ...@@ -987,6 +993,9 @@ struct i915_gpu_error {
/* For hangcheck timer */ /* For hangcheck timer */
#define DRM_I915_HANGCHECK_PERIOD 1500 /* in ms */ #define DRM_I915_HANGCHECK_PERIOD 1500 /* in ms */
#define DRM_I915_HANGCHECK_JIFFIES msecs_to_jiffies(DRM_I915_HANGCHECK_PERIOD) #define DRM_I915_HANGCHECK_JIFFIES msecs_to_jiffies(DRM_I915_HANGCHECK_PERIOD)
/* Hang gpu twice in this window and your context gets banned */
#define DRM_I915_CTX_BAN_PERIOD DIV_ROUND_UP(8*DRM_I915_HANGCHECK_PERIOD, 1000)
struct timer_list hangcheck_timer; struct timer_list hangcheck_timer;
/* For reset and error_state handling. */ /* For reset and error_state handling. */
...@@ -995,8 +1004,6 @@ struct i915_gpu_error { ...@@ -995,8 +1004,6 @@ struct i915_gpu_error {
struct drm_i915_error_state *first_error; struct drm_i915_error_state *first_error;
struct work_struct work; struct work_struct work;
unsigned long last_reset;
/** /**
* State variable and reset counter controlling the reset flow * State variable and reset counter controlling the reset flow
* *
......
...@@ -2188,6 +2188,21 @@ static bool i915_request_guilty(struct drm_i915_gem_request *request, ...@@ -2188,6 +2188,21 @@ static bool i915_request_guilty(struct drm_i915_gem_request *request,
return false; return false;
} }
static bool i915_context_is_banned(const struct i915_ctx_hang_stats *hs)
{
const unsigned long elapsed = get_seconds() - hs->guilty_ts;
if (hs->banned)
return true;
if (elapsed <= DRM_I915_CTX_BAN_PERIOD) {
DRM_ERROR("context hanging too fast, declaring banned!\n");
return true;
}
return false;
}
static void i915_set_reset_status(struct intel_ring_buffer *ring, static void i915_set_reset_status(struct intel_ring_buffer *ring,
struct drm_i915_gem_request *request, struct drm_i915_gem_request *request,
u32 acthd) u32 acthd)
...@@ -2224,10 +2239,13 @@ static void i915_set_reset_status(struct intel_ring_buffer *ring, ...@@ -2224,10 +2239,13 @@ static void i915_set_reset_status(struct intel_ring_buffer *ring,
hs = &request->file_priv->hang_stats; hs = &request->file_priv->hang_stats;
if (hs) { if (hs) {
if (guilty) if (guilty) {
hs->banned = i915_context_is_banned(hs);
hs->batch_active++; hs->batch_active++;
else hs->guilty_ts = get_seconds();
} else {
hs->batch_pending++; hs->batch_pending++;
}
} }
} }
......
...@@ -929,6 +929,7 @@ i915_gem_do_execbuffer(struct drm_device *dev, void *data, ...@@ -929,6 +929,7 @@ i915_gem_do_execbuffer(struct drm_device *dev, void *data,
struct drm_i915_gem_object *batch_obj; struct drm_i915_gem_object *batch_obj;
struct drm_clip_rect *cliprects = NULL; struct drm_clip_rect *cliprects = NULL;
struct intel_ring_buffer *ring; struct intel_ring_buffer *ring;
struct i915_ctx_hang_stats *hs;
u32 ctx_id = i915_execbuffer2_get_context_id(*args); u32 ctx_id = i915_execbuffer2_get_context_id(*args);
u32 exec_start, exec_len; u32 exec_start, exec_len;
u32 mask, flags; u32 mask, flags;
...@@ -1118,6 +1119,17 @@ i915_gem_do_execbuffer(struct drm_device *dev, void *data, ...@@ -1118,6 +1119,17 @@ i915_gem_do_execbuffer(struct drm_device *dev, void *data,
if (ret) if (ret)
goto err; goto err;
hs = i915_gem_context_get_hang_stats(dev, file, ctx_id);
if (IS_ERR(hs)) {
ret = PTR_ERR(hs);
goto err;
}
if (hs->banned) {
ret = -EIO;
goto err;
}
ret = i915_switch_context(ring, file, ctx_id); ret = i915_switch_context(ring, file, ctx_id);
if (ret) if (ret)
goto err; goto err;
......
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