[Intel-gfx] [PATCH v9 31/70] drm/i915: Fix workarounds selftest, part 1
Daniel Vetter
daniel at ffwll.ch
Wed Mar 24 16:16:46 UTC 2021
On Tue, Mar 23, 2021 at 04:50:20PM +0100, Maarten Lankhorst wrote:
> pin_map needs the ww lock, so ensure we pin both before submission.
>
> Signed-off-by: Maarten Lankhorst <maarten.lankhorst at linux.intel.com>
> Reviewed-by: Thomas Hellström <thomas.hellstrom at linux.intel.com>
Another one where I picked an older version:
https://lore.kernel.org/intel-gfx/20210128162612.927917-32-maarten.lankhorst@linux.intel.com/
Cheers, Daniel
> ---
> drivers/gpu/drm/i915/gem/i915_gem_object.h | 3 +
> drivers/gpu/drm/i915/gem/i915_gem_pages.c | 12 +++
> .../gpu/drm/i915/gt/selftest_workarounds.c | 95 +++++++++++++------
> 3 files changed, 80 insertions(+), 30 deletions(-)
>
> diff --git a/drivers/gpu/drm/i915/gem/i915_gem_object.h b/drivers/gpu/drm/i915/gem/i915_gem_object.h
> index 6c3f75adb53c..983f2d4b2a85 100644
> --- a/drivers/gpu/drm/i915/gem/i915_gem_object.h
> +++ b/drivers/gpu/drm/i915/gem/i915_gem_object.h
> @@ -437,6 +437,9 @@ void i915_gem_object_writeback(struct drm_i915_gem_object *obj);
> void *__must_check i915_gem_object_pin_map(struct drm_i915_gem_object *obj,
> enum i915_map_type type);
>
> +void *__must_check i915_gem_object_pin_map_unlocked(struct drm_i915_gem_object *obj,
> + enum i915_map_type type);
> +
> void __i915_gem_object_flush_map(struct drm_i915_gem_object *obj,
> unsigned long offset,
> unsigned long size);
> diff --git a/drivers/gpu/drm/i915/gem/i915_gem_pages.c b/drivers/gpu/drm/i915/gem/i915_gem_pages.c
> index e947d4c0da1f..a24617af3c93 100644
> --- a/drivers/gpu/drm/i915/gem/i915_gem_pages.c
> +++ b/drivers/gpu/drm/i915/gem/i915_gem_pages.c
> @@ -400,6 +400,18 @@ void *i915_gem_object_pin_map(struct drm_i915_gem_object *obj,
> goto out_unlock;
> }
>
> +void *i915_gem_object_pin_map_unlocked(struct drm_i915_gem_object *obj,
> + enum i915_map_type type)
> +{
> + void *ret;
> +
> + i915_gem_object_lock(obj, NULL);
> + ret = i915_gem_object_pin_map(obj, type);
> + i915_gem_object_unlock(obj);
> +
> + return ret;
> +}
> +
> void __i915_gem_object_flush_map(struct drm_i915_gem_object *obj,
> unsigned long offset,
> unsigned long size)
> diff --git a/drivers/gpu/drm/i915/gt/selftest_workarounds.c b/drivers/gpu/drm/i915/gt/selftest_workarounds.c
> index de6136bd10ac..a508614b2fd5 100644
> --- a/drivers/gpu/drm/i915/gt/selftest_workarounds.c
> +++ b/drivers/gpu/drm/i915/gt/selftest_workarounds.c
> @@ -103,15 +103,13 @@ read_nonprivs(struct intel_context *ce, struct i915_vma *result)
> int err;
> int i;
>
> - rq = intel_context_create_request(ce);
> + rq = i915_request_create(ce);
> if (IS_ERR(rq))
> return rq;
>
> - i915_vma_lock(result);
> err = i915_request_await_object(rq, result->obj, true);
> if (err == 0)
> err = i915_vma_move_to_active(result, rq, EXEC_OBJECT_WRITE);
> - i915_vma_unlock(result);
> if (err)
> goto err_rq;
>
> @@ -176,10 +174,11 @@ static int check_whitelist(struct intel_context *ce)
> u32 *vaddr;
> int i;
>
> - result = __vm_create_scratch_for_read(&engine->gt->ggtt->vm, PAGE_SIZE);
> + result = __vm_create_scratch_for_read_pinned(&engine->gt->ggtt->vm, PAGE_SIZE);
> if (IS_ERR(result))
> return PTR_ERR(result);
>
> + i915_gem_object_lock(result->obj, NULL);
> vaddr = i915_gem_object_pin_map(result->obj, I915_MAP_WB);
> if (IS_ERR(vaddr)) {
> err = PTR_ERR(vaddr);
> @@ -219,6 +218,8 @@ static int check_whitelist(struct intel_context *ce)
> out_map:
> i915_gem_object_unpin_map(result->obj);
> out_put:
> + i915_vma_unpin(result);
> + i915_gem_object_unlock(result->obj);
> i915_vma_put(result);
> return err;
> }
> @@ -279,10 +280,14 @@ static int check_whitelist_across_reset(struct intel_engine_cs *engine,
> if (IS_ERR(ce))
> return PTR_ERR(ce);
>
> - err = igt_spinner_init(&spin, engine->gt);
> + err = intel_context_pin(ce);
> if (err)
> goto out_ctx;
>
> + err = igt_spinner_init(&spin, engine->gt);
> + if (err)
> + goto out_unpin;
> +
> err = check_whitelist(ce);
> if (err) {
> pr_err("Invalid whitelist *before* %s reset!\n", name);
> @@ -315,6 +320,13 @@ static int check_whitelist_across_reset(struct intel_engine_cs *engine,
> err = PTR_ERR(tmp);
> goto out_spin;
> }
> + err = intel_context_pin(tmp);
> + if (err) {
> + intel_context_put(tmp);
> + goto out_spin;
> + }
> +
> + intel_context_unpin(ce);
> intel_context_put(ce);
> ce = tmp;
>
> @@ -327,6 +339,8 @@ static int check_whitelist_across_reset(struct intel_engine_cs *engine,
>
> out_spin:
> igt_spinner_fini(&spin);
> +out_unpin:
> + intel_context_unpin(ce);
> out_ctx:
> intel_context_put(ce);
> return err;
> @@ -475,6 +489,7 @@ static int check_dirty_whitelist(struct intel_context *ce)
>
> for (i = 0; i < engine->whitelist.count; i++) {
> u32 reg = i915_mmio_reg_offset(engine->whitelist.list[i].reg);
> + struct i915_gem_ww_ctx ww;
> u64 addr = scratch->node.start;
> struct i915_request *rq;
> u32 srm, lrm, rsvd;
> @@ -490,6 +505,29 @@ static int check_dirty_whitelist(struct intel_context *ce)
>
> ro_reg = ro_register(reg);
>
> + i915_gem_ww_ctx_init(&ww, false);
> +retry:
> + cs = NULL;
> + err = i915_gem_object_lock(scratch->obj, &ww);
> + if (!err)
> + err = i915_gem_object_lock(batch->obj, &ww);
> + if (!err)
> + err = intel_context_pin_ww(ce, &ww);
> + if (err)
> + goto out;
> +
> + cs = i915_gem_object_pin_map(batch->obj, I915_MAP_WC);
> + if (IS_ERR(cs)) {
> + err = PTR_ERR(cs);
> + goto out_ctx;
> + }
> +
> + results = i915_gem_object_pin_map(scratch->obj, I915_MAP_WB);
> + if (IS_ERR(results)) {
> + err = PTR_ERR(results);
> + goto out_unmap_batch;
> + }
> +
> /* Clear non priv flags */
> reg &= RING_FORCE_TO_NONPRIV_ADDRESS_MASK;
>
> @@ -501,12 +539,6 @@ static int check_dirty_whitelist(struct intel_context *ce)
> pr_debug("%s: Writing garbage to %x\n",
> engine->name, reg);
>
> - cs = i915_gem_object_pin_map(batch->obj, I915_MAP_WC);
> - if (IS_ERR(cs)) {
> - err = PTR_ERR(cs);
> - goto out_batch;
> - }
> -
> /* SRM original */
> *cs++ = srm;
> *cs++ = reg;
> @@ -553,11 +585,12 @@ static int check_dirty_whitelist(struct intel_context *ce)
> i915_gem_object_flush_map(batch->obj);
> i915_gem_object_unpin_map(batch->obj);
> intel_gt_chipset_flush(engine->gt);
> + cs = NULL;
>
> - rq = intel_context_create_request(ce);
> + rq = i915_request_create(ce);
> if (IS_ERR(rq)) {
> err = PTR_ERR(rq);
> - goto out_batch;
> + goto out_unmap_scratch;
> }
>
> if (engine->emit_init_breadcrumb) { /* Be nice if we hang */
> @@ -566,20 +599,16 @@ static int check_dirty_whitelist(struct intel_context *ce)
> goto err_request;
> }
>
> - i915_vma_lock(batch);
> err = i915_request_await_object(rq, batch->obj, false);
> if (err == 0)
> err = i915_vma_move_to_active(batch, rq, 0);
> - i915_vma_unlock(batch);
> if (err)
> goto err_request;
>
> - i915_vma_lock(scratch);
> err = i915_request_await_object(rq, scratch->obj, true);
> if (err == 0)
> err = i915_vma_move_to_active(scratch, rq,
> EXEC_OBJECT_WRITE);
> - i915_vma_unlock(scratch);
> if (err)
> goto err_request;
>
> @@ -595,13 +624,7 @@ static int check_dirty_whitelist(struct intel_context *ce)
> pr_err("%s: Futzing %x timedout; cancelling test\n",
> engine->name, reg);
> intel_gt_set_wedged(engine->gt);
> - goto out_batch;
> - }
> -
> - results = i915_gem_object_pin_map(scratch->obj, I915_MAP_WB);
> - if (IS_ERR(results)) {
> - err = PTR_ERR(results);
> - goto out_batch;
> + goto out_unmap_scratch;
> }
>
> GEM_BUG_ON(values[ARRAY_SIZE(values) - 1] != 0xffffffff);
> @@ -612,7 +635,7 @@ static int check_dirty_whitelist(struct intel_context *ce)
> pr_err("%s: Unable to write to whitelisted register %x\n",
> engine->name, reg);
> err = -EINVAL;
> - goto out_unpin;
> + goto out_unmap_scratch;
> }
> } else {
> rsvd = 0;
> @@ -678,15 +701,27 @@ static int check_dirty_whitelist(struct intel_context *ce)
>
> err = -EINVAL;
> }
> -out_unpin:
> +out_unmap_scratch:
> i915_gem_object_unpin_map(scratch->obj);
> +out_unmap_batch:
> + if (cs)
> + i915_gem_object_unpin_map(batch->obj);
> +out_ctx:
> + intel_context_unpin(ce);
> +out:
> + if (err == -EDEADLK) {
> + err = i915_gem_ww_ctx_backoff(&ww);
> + if (!err)
> + goto retry;
> + }
> + i915_gem_ww_ctx_fini(&ww);
> if (err)
> break;
> }
>
> if (igt_flush_test(engine->i915))
> err = -EIO;
> -out_batch:
> +
> i915_vma_unpin_and_release(&batch, 0);
> out_scratch:
> i915_vma_unpin_and_release(&scratch, 0);
> @@ -820,7 +855,7 @@ static int scrub_whitelisted_registers(struct intel_context *ce)
> if (IS_ERR(batch))
> return PTR_ERR(batch);
>
> - cs = i915_gem_object_pin_map(batch->obj, I915_MAP_WC);
> + cs = i915_gem_object_pin_map_unlocked(batch->obj, I915_MAP_WC);
> if (IS_ERR(cs)) {
> err = PTR_ERR(cs);
> goto err_batch;
> @@ -955,11 +990,11 @@ check_whitelisted_registers(struct intel_engine_cs *engine,
> u32 *a, *b;
> int i, err;
>
> - a = i915_gem_object_pin_map(A->obj, I915_MAP_WB);
> + a = i915_gem_object_pin_map_unlocked(A->obj, I915_MAP_WB);
> if (IS_ERR(a))
> return PTR_ERR(a);
>
> - b = i915_gem_object_pin_map(B->obj, I915_MAP_WB);
> + b = i915_gem_object_pin_map_unlocked(B->obj, I915_MAP_WB);
> if (IS_ERR(b)) {
> err = PTR_ERR(b);
> goto err_a;
> --
> 2.31.0
>
> _______________________________________________
> Intel-gfx mailing list
> Intel-gfx at lists.freedesktop.org
> https://lists.freedesktop.org/mailman/listinfo/intel-gfx
--
Daniel Vetter
Software Engineer, Intel Corporation
http://blog.ffwll.ch
More information about the dri-devel
mailing list