]> asedeno.scripts.mit.edu Git - linux.git/commitdiff
drm/i915: Skip context_barrier emission for unused contexts
authorChris Wilson <chris@chris-wilson.co.uk>
Tue, 4 Jun 2019 15:24:08 +0000 (16:24 +0100)
committerChris Wilson <chris@chris-wilson.co.uk>
Thu, 6 Jun 2019 12:35:16 +0000 (13:35 +0100)
The intent was to skip unused HW contexts by checking ce->state.
However, this only works for execlists where the ppGTT pointers is
stored inside the HW context. For gen7, the ppGTT is alongside the
logical state and must be updated on all active engines but, crucially,
only on active engines. As we need different checks, and to keep
context_barrier_task() agnostic, pass in the predicate.

Bugzilla: https://bugs.freedesktop.org/show_bug.cgi?id=110836
Fixes: 62c8e423450d ("drm/i915: Skip unused contexts for context_barrier_task()")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190604152408.24468-1-chris@chris-wilson.co.uk
drivers/gpu/drm/i915/gem/i915_gem_context.c
drivers/gpu/drm/i915/gem/selftests/i915_gem_context.c

index 6cac1c144c796552e26d2f3f2334be4549fce89a..dd9aa77e38ae5f97f728812edc7a3a3c59f634d0 100644 (file)
@@ -915,6 +915,7 @@ static void cb_retire(struct i915_active *base)
 I915_SELFTEST_DECLARE(static intel_engine_mask_t context_barrier_inject_fault);
 static int context_barrier_task(struct i915_gem_context *ctx,
                                intel_engine_mask_t engines,
+                               bool (*skip)(struct intel_context *ce, void *data),
                                int (*emit)(struct i915_request *rq, void *data),
                                void (*task)(void *data),
                                void *data)
@@ -944,7 +945,10 @@ static int context_barrier_task(struct i915_gem_context *ctx,
                        break;
                }
 
-               if (!(ce->engine->mask & engines) || !ce->state)
+               if (!(ce->engine->mask & engines))
+                       continue;
+
+               if (skip && skip(ce, data))
                        continue;
 
                rq = intel_context_create_request(ce);
@@ -1071,6 +1075,14 @@ static int emit_ppgtt_update(struct i915_request *rq, void *data)
        return 0;
 }
 
+static bool skip_ppgtt_update(struct intel_context *ce, void *data)
+{
+       if (HAS_LOGICAL_RING_CONTEXTS(ce->engine->i915))
+               return !ce->state;
+       else
+               return !atomic_read(&ce->pin_count);
+}
+
 static int set_ppgtt(struct drm_i915_file_private *file_priv,
                     struct i915_gem_context *ctx,
                     struct drm_i915_gem_context_param *args)
@@ -1118,6 +1130,7 @@ static int set_ppgtt(struct drm_i915_file_private *file_priv,
         * only indirectly through the context.
         */
        err = context_barrier_task(ctx, ALL_ENGINES,
+                                  skip_ppgtt_update,
                                   emit_ppgtt_update,
                                   set_ppgtt_barrier,
                                   old);
index 1bc3b8026400d20fb08be3cce070b1df0efd6288..41105f6ed206ffd8f902a3d83e24a9233240a113 100644 (file)
@@ -1619,6 +1619,11 @@ __engine_name(struct drm_i915_private *i915, intel_engine_mask_t engines)
        return "none";
 }
 
+static bool skip_unused_engines(struct intel_context *ce, void *data)
+{
+       return !ce->state;
+}
+
 static void mock_barrier_task(void *data)
 {
        unsigned int *counter = data;
@@ -1651,7 +1656,7 @@ static int mock_context_barrier(void *arg)
 
        counter = 0;
        err = context_barrier_task(ctx, 0,
-                                  NULL, mock_barrier_task, &counter);
+                                  NULL, NULL, mock_barrier_task, &counter);
        if (err) {
                pr_err("Failed at line %d, err=%d\n", __LINE__, err);
                goto out;
@@ -1664,7 +1669,10 @@ static int mock_context_barrier(void *arg)
 
        counter = 0;
        err = context_barrier_task(ctx, ALL_ENGINES,
-                                  NULL, mock_barrier_task, &counter);
+                                  skip_unused_engines,
+                                  NULL,
+                                  mock_barrier_task,
+                                  &counter);
        if (err) {
                pr_err("Failed at line %d, err=%d\n", __LINE__, err);
                goto out;
@@ -1685,7 +1693,7 @@ static int mock_context_barrier(void *arg)
        counter = 0;
        context_barrier_inject_fault = BIT(RCS0);
        err = context_barrier_task(ctx, ALL_ENGINES,
-                                  NULL, mock_barrier_task, &counter);
+                                  NULL, NULL, mock_barrier_task, &counter);
        context_barrier_inject_fault = 0;
        if (err == -ENXIO)
                err = 0;
@@ -1700,7 +1708,10 @@ static int mock_context_barrier(void *arg)
 
        counter = 0;
        err = context_barrier_task(ctx, ALL_ENGINES,
-                                  NULL, mock_barrier_task, &counter);
+                                  skip_unused_engines,
+                                  NULL,
+                                  mock_barrier_task,
+                                  &counter);
        if (err) {
                pr_err("Failed at line %d, err=%d\n", __LINE__, err);
                goto out;