2018-04-04 17:33:29 +08:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: MIT
|
|
|
|
*
|
|
|
|
* Copyright © 2018 Intel Corporation
|
|
|
|
*/
|
|
|
|
|
2019-02-05 20:38:35 +08:00
|
|
|
#include <linux/prime_numbers.h>
|
|
|
|
|
2019-05-28 17:29:49 +08:00
|
|
|
#include "gem/i915_gem_pm.h"
|
2019-10-23 21:31:06 +08:00
|
|
|
#include "gt/intel_engine_heartbeat.h"
|
2019-04-25 01:48:39 +08:00
|
|
|
#include "gt/intel_reset.h"
|
2020-06-17 21:09:15 +08:00
|
|
|
#include "gt/selftest_engine_heartbeat.h"
|
2019-05-28 17:29:49 +08:00
|
|
|
|
2019-04-25 01:48:39 +08:00
|
|
|
#include "i915_selftest.h"
|
|
|
|
#include "selftests/i915_random.h"
|
|
|
|
#include "selftests/igt_flush_test.h"
|
|
|
|
#include "selftests/igt_live_test.h"
|
|
|
|
#include "selftests/igt_spinner.h"
|
2019-05-22 05:11:33 +08:00
|
|
|
#include "selftests/lib_sw_fence.h"
|
2019-05-28 17:29:49 +08:00
|
|
|
|
|
|
|
#include "gem/selftests/igt_gem_utils.h"
|
|
|
|
#include "gem/selftests/mock_context.h"
|
2018-04-04 17:33:29 +08:00
|
|
|
|
2019-10-14 17:07:49 +08:00
|
|
|
#define CS_GPR(engine, n) ((engine)->mmio_base + 0x600 + (n) * 4)
|
2020-04-22 18:09:03 +08:00
|
|
|
#define NUM_GPR 16
|
|
|
|
#define NUM_GPR_DW (NUM_GPR * 2) /* each GPR is 2 dwords */
|
2019-10-14 17:07:49 +08:00
|
|
|
|
2019-10-10 19:02:52 +08:00
|
|
|
static struct i915_vma *create_scratch(struct intel_gt *gt)
|
|
|
|
{
|
|
|
|
struct drm_i915_gem_object *obj;
|
|
|
|
struct i915_vma *vma;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
obj = i915_gem_object_create_internal(gt->i915, PAGE_SIZE);
|
|
|
|
if (IS_ERR(obj))
|
|
|
|
return ERR_CAST(obj);
|
|
|
|
|
|
|
|
i915_gem_object_set_cache_coherency(obj, I915_CACHING_CACHED);
|
|
|
|
|
|
|
|
vma = i915_vma_instance(obj, >->ggtt->vm, NULL);
|
|
|
|
if (IS_ERR(vma)) {
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
return vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = i915_vma_pin(vma, 0, 0, PIN_GLOBAL);
|
|
|
|
if (err) {
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return vma;
|
|
|
|
}
|
|
|
|
|
2020-03-30 20:16:44 +08:00
|
|
|
static bool is_active(struct i915_request *rq)
|
|
|
|
{
|
|
|
|
if (i915_request_is_active(rq))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (i915_request_on_hold(rq))
|
|
|
|
return true;
|
|
|
|
|
2020-05-19 14:31:15 +08:00
|
|
|
if (i915_request_has_initial_breadcrumb(rq) && i915_request_started(rq))
|
2020-03-30 20:16:44 +08:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-01-29 04:43:15 +08:00
|
|
|
static int wait_for_submit(struct intel_engine_cs *engine,
|
|
|
|
struct i915_request *rq,
|
|
|
|
unsigned long timeout)
|
|
|
|
{
|
|
|
|
timeout += jiffies;
|
|
|
|
do {
|
2020-03-30 20:16:44 +08:00
|
|
|
bool done = time_after(jiffies, timeout);
|
2020-02-19 05:12:15 +08:00
|
|
|
|
2020-03-30 20:16:44 +08:00
|
|
|
if (i915_request_completed(rq)) /* that was quick! */
|
2020-01-29 04:43:15 +08:00
|
|
|
return 0;
|
2020-02-14 20:06:59 +08:00
|
|
|
|
2020-03-30 20:16:44 +08:00
|
|
|
/* Wait until the HW has acknowleged the submission (or err) */
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
if (!READ_ONCE(engine->execlists.pending[0]) && is_active(rq))
|
2020-02-14 20:06:59 +08:00
|
|
|
return 0;
|
2020-01-29 04:43:15 +08:00
|
|
|
|
2020-03-30 20:16:44 +08:00
|
|
|
if (done)
|
|
|
|
return -ETIME;
|
|
|
|
|
|
|
|
cond_resched();
|
|
|
|
} while (1);
|
2020-01-29 04:43:15 +08:00
|
|
|
}
|
|
|
|
|
2020-02-27 16:57:18 +08:00
|
|
|
static int wait_for_reset(struct intel_engine_cs *engine,
|
|
|
|
struct i915_request *rq,
|
|
|
|
unsigned long timeout)
|
|
|
|
{
|
|
|
|
timeout += jiffies;
|
|
|
|
|
|
|
|
do {
|
|
|
|
cond_resched();
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
|
|
|
|
if (READ_ONCE(engine->execlists.pending[0]))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (i915_request_completed(rq))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (READ_ONCE(rq->fence.error))
|
|
|
|
break;
|
|
|
|
} while (time_before(jiffies, timeout));
|
|
|
|
|
|
|
|
flush_scheduled_work();
|
|
|
|
|
|
|
|
if (rq->fence.error != -EIO) {
|
|
|
|
pr_err("%s: hanging request %llx:%lld not reset\n",
|
|
|
|
engine->name,
|
|
|
|
rq->fence.context,
|
|
|
|
rq->fence.seqno);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Give the request a jiffie to complete after flushing the worker */
|
|
|
|
if (i915_request_wait(rq, 0,
|
|
|
|
max(0l, (long)(timeout - jiffies)) + 1) < 0) {
|
|
|
|
pr_err("%s: hanging request %llx:%lld did not complete\n",
|
|
|
|
engine->name,
|
|
|
|
rq->fence.context,
|
|
|
|
rq->fence.seqno);
|
|
|
|
return -ETIME;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-04 17:33:29 +08:00
|
|
|
static int live_sanitycheck(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2019-12-22 00:03:24 +08:00
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
2018-11-30 16:02:53 +08:00
|
|
|
struct igt_spinner spin;
|
2019-12-22 00:03:24 +08:00
|
|
|
int err = 0;
|
2018-04-04 17:33:29 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (!HAS_LOGICAL_RING_CONTEXTS(gt->i915))
|
2018-04-04 17:33:29 +08:00
|
|
|
return 0;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_spinner_init(&spin, gt))
|
2019-10-04 21:40:14 +08:00
|
|
|
return -ENOMEM;
|
2018-04-04 17:33:29 +08:00
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
struct intel_context *ce;
|
2018-04-04 17:33:29 +08:00
|
|
|
struct i915_request *rq;
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq = igt_spinner_create_request(&spin, ce, MI_NOOP);
|
2018-04-04 17:33:29 +08:00
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
2019-12-22 00:03:24 +08:00
|
|
|
goto out_ctx;
|
2018-04-04 17:33:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
2018-11-30 16:02:53 +08:00
|
|
|
if (!igt_wait_for_spinner(&spin, rq)) {
|
2018-04-04 17:33:29 +08:00
|
|
|
GEM_TRACE("spinner failed to start\n");
|
|
|
|
GEM_TRACE_DUMP();
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
2018-04-04 17:33:29 +08:00
|
|
|
err = -EIO;
|
2019-12-22 00:03:24 +08:00
|
|
|
goto out_ctx;
|
2018-04-04 17:33:29 +08:00
|
|
|
}
|
|
|
|
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_end(&spin);
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_flush_test(gt->i915)) {
|
2018-04-04 17:33:29 +08:00
|
|
|
err = -EIO;
|
2019-12-22 00:03:24 +08:00
|
|
|
goto out_ctx;
|
2018-04-04 17:33:29 +08:00
|
|
|
}
|
2019-12-22 00:03:24 +08:00
|
|
|
|
|
|
|
out_ctx:
|
|
|
|
intel_context_put(ce);
|
|
|
|
if (err)
|
|
|
|
break;
|
2018-04-04 17:33:29 +08:00
|
|
|
}
|
|
|
|
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_fini(&spin);
|
2018-04-04 17:33:29 +08:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
static int live_unlite_restore(struct intel_gt *gt, int prio)
|
2019-10-03 02:34:59 +08:00
|
|
|
{
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
struct igt_spinner spin;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that we can correctly context switch between 2 instances
|
|
|
|
* on the same engine from the same parent context.
|
|
|
|
*/
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_spinner_init(&spin, gt))
|
2019-10-15 16:59:11 +08:00
|
|
|
return err;
|
2019-10-03 02:34:59 +08:00
|
|
|
|
|
|
|
err = 0;
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2019-10-03 02:34:59 +08:00
|
|
|
struct intel_context *ce[2] = {};
|
|
|
|
struct i915_request *rq[2];
|
|
|
|
struct igt_live_test t;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (prio && !intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!intel_engine_can_store_dword(engine))
|
|
|
|
continue;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_live_test_begin(&t, gt->i915, __func__, engine->name)) {
|
2019-10-03 02:34:59 +08:00
|
|
|
err = -EIO;
|
|
|
|
break;
|
|
|
|
}
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2019-10-03 02:34:59 +08:00
|
|
|
|
|
|
|
for (n = 0; n < ARRAY_SIZE(ce); n++) {
|
|
|
|
struct intel_context *tmp;
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
tmp = intel_context_create(engine);
|
2019-10-03 02:34:59 +08:00
|
|
|
if (IS_ERR(tmp)) {
|
|
|
|
err = PTR_ERR(tmp);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = intel_context_pin(tmp);
|
|
|
|
if (err) {
|
|
|
|
intel_context_put(tmp);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Setup the pair of contexts such that if we
|
|
|
|
* lite-restore using the RING_TAIL from ce[1] it
|
|
|
|
* will execute garbage from ce[0]->ring.
|
|
|
|
*/
|
|
|
|
memset(tmp->ring->vaddr,
|
|
|
|
POISON_INUSE, /* IPEHR: 0x5a5a5a5a [hung!] */
|
|
|
|
tmp->ring->vma->size);
|
|
|
|
|
|
|
|
ce[n] = tmp;
|
|
|
|
}
|
|
|
|
GEM_BUG_ON(!ce[1]->ring->size);
|
|
|
|
intel_ring_reset(ce[1]->ring, ce[1]->ring->size / 2);
|
2020-02-11 20:01:31 +08:00
|
|
|
__execlists_update_reg_state(ce[1], engine, ce[1]->ring->head);
|
2019-10-03 02:34:59 +08:00
|
|
|
|
|
|
|
rq[0] = igt_spinner_create_request(&spin, ce[0], MI_ARB_CHECK);
|
|
|
|
if (IS_ERR(rq[0])) {
|
|
|
|
err = PTR_ERR(rq[0]);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq[0]);
|
|
|
|
i915_request_add(rq[0]);
|
|
|
|
GEM_BUG_ON(rq[0]->postfix > ce[1]->ring->emit);
|
|
|
|
|
|
|
|
if (!igt_wait_for_spinner(&spin, rq[0])) {
|
|
|
|
i915_request_put(rq[0]);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq[1] = i915_request_create(ce[1]);
|
|
|
|
if (IS_ERR(rq[1])) {
|
|
|
|
err = PTR_ERR(rq[1]);
|
|
|
|
i915_request_put(rq[0]);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!prio) {
|
|
|
|
/*
|
|
|
|
* Ensure we do the switch to ce[1] on completion.
|
|
|
|
*
|
|
|
|
* rq[0] is already submitted, so this should reduce
|
|
|
|
* to a no-op (a wait on a request on the same engine
|
|
|
|
* uses the submit fence, not the completion fence),
|
|
|
|
* but it will install a dependency on rq[1] for rq[0]
|
|
|
|
* that will prevent the pair being reordered by
|
|
|
|
* timeslicing.
|
|
|
|
*/
|
|
|
|
i915_request_await_dma_fence(rq[1], &rq[0]->fence);
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq[1]);
|
|
|
|
i915_request_add(rq[1]);
|
|
|
|
GEM_BUG_ON(rq[1]->postfix <= rq[0]->postfix);
|
|
|
|
i915_request_put(rq[0]);
|
|
|
|
|
|
|
|
if (prio) {
|
|
|
|
struct i915_sched_attr attr = {
|
|
|
|
.priority = prio,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Alternatively preempt the spinner with ce[1] */
|
|
|
|
engine->schedule(rq[1], &attr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* And switch back to ce[0] for good measure */
|
|
|
|
rq[0] = i915_request_create(ce[0]);
|
|
|
|
if (IS_ERR(rq[0])) {
|
|
|
|
err = PTR_ERR(rq[0]);
|
|
|
|
i915_request_put(rq[1]);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_await_dma_fence(rq[0], &rq[1]->fence);
|
|
|
|
i915_request_get(rq[0]);
|
|
|
|
i915_request_add(rq[0]);
|
|
|
|
GEM_BUG_ON(rq[0]->postfix > rq[1]->postfix);
|
|
|
|
i915_request_put(rq[1]);
|
|
|
|
i915_request_put(rq[0]);
|
|
|
|
|
|
|
|
err_ce:
|
2020-05-21 20:43:04 +08:00
|
|
|
intel_engine_flush_submission(engine);
|
2019-10-03 02:34:59 +08:00
|
|
|
igt_spinner_end(&spin);
|
|
|
|
for (n = 0; n < ARRAY_SIZE(ce); n++) {
|
|
|
|
if (IS_ERR_OR_NULL(ce[n]))
|
|
|
|
break;
|
|
|
|
|
|
|
|
intel_context_unpin(ce[n]);
|
|
|
|
intel_context_put(ce[n]);
|
|
|
|
}
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2019-10-03 02:34:59 +08:00
|
|
|
if (igt_live_test_end(&t))
|
|
|
|
err = -EIO;
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
igt_spinner_fini(&spin);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_unlite_switch(void *arg)
|
|
|
|
{
|
|
|
|
return live_unlite_restore(arg, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_unlite_preempt(void *arg)
|
|
|
|
{
|
|
|
|
return live_unlite_restore(arg, I915_USER_PRIORITY(I915_PRIORITY_MAX));
|
|
|
|
}
|
|
|
|
|
2020-06-17 02:55:17 +08:00
|
|
|
static int live_unlite_ring(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct igt_spinner spin;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Setup a preemption event that will cause almost the entire ring
|
|
|
|
* to be unwound, potentially fooling our intel_ring_direction()
|
|
|
|
* into emitting a forward lite-restore instead of the rollback.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (igt_spinner_init(&spin, gt))
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
struct intel_context *ce[2] = {};
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct igt_live_test t;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!intel_engine_can_store_dword(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (igt_live_test_begin(&t, gt->i915, __func__, engine->name)) {
|
|
|
|
err = -EIO;
|
|
|
|
break;
|
|
|
|
}
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2020-06-17 02:55:17 +08:00
|
|
|
|
|
|
|
for (n = 0; n < ARRAY_SIZE(ce); n++) {
|
|
|
|
struct intel_context *tmp;
|
|
|
|
|
|
|
|
tmp = intel_context_create(engine);
|
|
|
|
if (IS_ERR(tmp)) {
|
|
|
|
err = PTR_ERR(tmp);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = intel_context_pin(tmp);
|
|
|
|
if (err) {
|
|
|
|
intel_context_put(tmp);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset32(tmp->ring->vaddr,
|
|
|
|
0xdeadbeef, /* trigger a hang if executed */
|
|
|
|
tmp->ring->vma->size / sizeof(u32));
|
|
|
|
|
|
|
|
ce[n] = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Create max prio spinner, followed by N low prio nops */
|
|
|
|
rq = igt_spinner_create_request(&spin, ce[0], MI_ARB_CHECK);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_BARRIER;
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
if (!igt_wait_for_spinner(&spin, rq)) {
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
i915_request_put(rq);
|
|
|
|
err = -ETIME;
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fill the ring, until we will cause a wrap */
|
|
|
|
n = 0;
|
|
|
|
while (intel_ring_direction(ce[0]->ring,
|
|
|
|
rq->wa_tail,
|
|
|
|
ce[0]->ring->tail) <= 0) {
|
|
|
|
struct i915_request *tmp;
|
|
|
|
|
|
|
|
tmp = intel_context_create_request(ce[0]);
|
|
|
|
if (IS_ERR(tmp)) {
|
|
|
|
err = PTR_ERR(tmp);
|
|
|
|
i915_request_put(rq);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(tmp);
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
pr_debug("%s: Filled ring with %d nop tails {size:%x, tail:%x, emit:%x, rq.tail:%x}\n",
|
|
|
|
engine->name, n,
|
|
|
|
ce[0]->ring->size,
|
|
|
|
ce[0]->ring->tail,
|
|
|
|
ce[0]->ring->emit,
|
|
|
|
rq->tail);
|
|
|
|
GEM_BUG_ON(intel_ring_direction(ce[0]->ring,
|
|
|
|
rq->tail,
|
|
|
|
ce[0]->ring->tail) <= 0);
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
/* Create a second ring to preempt the first ring after rq[0] */
|
|
|
|
rq = intel_context_create_request(ce[1]);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_BARRIER;
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
err = wait_for_submit(engine, rq, HZ / 2);
|
|
|
|
i915_request_put(rq);
|
|
|
|
if (err) {
|
2020-06-17 16:52:07 +08:00
|
|
|
pr_err("%s: preemption request was not submitted\n",
|
2020-06-17 02:55:17 +08:00
|
|
|
engine->name);
|
|
|
|
err = -ETIME;
|
|
|
|
}
|
|
|
|
|
|
|
|
pr_debug("%s: ring[0]:{ tail:%x, emit:%x }, ring[1]:{ tail:%x, emit:%x }\n",
|
|
|
|
engine->name,
|
|
|
|
ce[0]->ring->tail, ce[0]->ring->emit,
|
|
|
|
ce[1]->ring->tail, ce[1]->ring->emit);
|
|
|
|
|
|
|
|
err_ce:
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
igt_spinner_end(&spin);
|
|
|
|
for (n = 0; n < ARRAY_SIZE(ce); n++) {
|
|
|
|
if (IS_ERR_OR_NULL(ce[n]))
|
|
|
|
break;
|
|
|
|
|
|
|
|
intel_context_unpin(ce[n]);
|
|
|
|
intel_context_put(ce[n]);
|
|
|
|
}
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2020-06-17 02:55:17 +08:00
|
|
|
if (igt_live_test_end(&t))
|
|
|
|
err = -EIO;
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
igt_spinner_fini(&spin);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-02-12 04:56:15 +08:00
|
|
|
static int live_pin_rewind(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We have to be careful not to trust intel_ring too much, for example
|
|
|
|
* ring->head is updated upon retire which is out of sync with pinning
|
|
|
|
* the context. Thus we cannot use ring->head to set CTX_RING_HEAD,
|
|
|
|
* or else we risk writing an older, stale value.
|
|
|
|
*
|
|
|
|
* To simulate this, let's apply a bit of deliberate sabotague.
|
|
|
|
*/
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct intel_ring *ring;
|
|
|
|
struct igt_live_test t;
|
|
|
|
|
|
|
|
if (igt_live_test_begin(&t, gt->i915, __func__, engine->name)) {
|
|
|
|
err = -EIO;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = intel_context_pin(ce);
|
|
|
|
if (err) {
|
|
|
|
intel_context_put(ce);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Keep the context awake while we play games */
|
|
|
|
err = i915_active_acquire(&ce->active);
|
|
|
|
if (err) {
|
|
|
|
intel_context_unpin(ce);
|
|
|
|
intel_context_put(ce);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ring = ce->ring;
|
|
|
|
|
|
|
|
/* Poison the ring, and offset the next request from HEAD */
|
|
|
|
memset32(ring->vaddr, STACK_MAGIC, ring->size / sizeof(u32));
|
|
|
|
ring->emit = ring->size / 2;
|
|
|
|
ring->tail = ring->emit;
|
|
|
|
GEM_BUG_ON(ring->head);
|
|
|
|
|
|
|
|
intel_context_unpin(ce);
|
|
|
|
|
|
|
|
/* Submit a simple nop request */
|
|
|
|
GEM_BUG_ON(intel_context_is_pinned(ce));
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
i915_active_release(&ce->active); /* e.g. async retire */
|
|
|
|
intel_context_put(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
GEM_BUG_ON(!rq->head);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
/* Expect not to hang! */
|
|
|
|
if (igt_live_test_end(&t)) {
|
|
|
|
err = -EIO;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-01-17 02:47:53 +08:00
|
|
|
static int live_hold_reset(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
struct igt_spinner spin;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* In order to support offline error capture for fast preempt reset,
|
|
|
|
* we need to decouple the guilty request and ensure that it and its
|
|
|
|
* descendents are not executed while the capture is in progress.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!intel_has_reset_engine(gt))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (igt_spinner_init(&spin, gt))
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2020-01-17 02:47:53 +08:00
|
|
|
|
|
|
|
rq = igt_spinner_create_request(&spin, ce, MI_ARB_CHECK);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
if (!igt_wait_for_spinner(&spin, rq)) {
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -ETIME;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We have our request executing, now remove it and reset */
|
|
|
|
|
|
|
|
if (test_and_set_bit(I915_RESET_ENGINE + id,
|
|
|
|
>->reset.flags)) {
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -EBUSY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
tasklet_disable(&engine->execlists.tasklet);
|
|
|
|
|
|
|
|
engine->execlists.tasklet.func(engine->execlists.tasklet.data);
|
|
|
|
GEM_BUG_ON(execlists_active(&engine->execlists) != rq);
|
|
|
|
|
2020-01-22 22:02:41 +08:00
|
|
|
i915_request_get(rq);
|
2020-01-17 02:47:53 +08:00
|
|
|
execlists_hold(engine, rq);
|
|
|
|
GEM_BUG_ON(!i915_request_on_hold(rq));
|
|
|
|
|
|
|
|
intel_engine_reset(engine, NULL);
|
|
|
|
GEM_BUG_ON(rq->fence.error != -EIO);
|
|
|
|
|
|
|
|
tasklet_enable(&engine->execlists.tasklet);
|
|
|
|
clear_and_wake_up_bit(I915_RESET_ENGINE + id,
|
|
|
|
>->reset.flags);
|
|
|
|
|
|
|
|
/* Check that we do not resubmit the held request */
|
|
|
|
if (!i915_request_wait(rq, 0, HZ / 5)) {
|
|
|
|
pr_err("%s: on hold request completed!\n",
|
|
|
|
engine->name);
|
|
|
|
i915_request_put(rq);
|
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
GEM_BUG_ON(!i915_request_on_hold(rq));
|
|
|
|
|
|
|
|
/* But is resubmitted on release */
|
|
|
|
execlists_unhold(engine, rq);
|
|
|
|
if (i915_request_wait(rq, 0, HZ / 5) < 0) {
|
|
|
|
pr_err("%s: held request did not complete!\n",
|
|
|
|
engine->name);
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -ETIME;
|
|
|
|
}
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
out:
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2020-01-17 02:47:53 +08:00
|
|
|
intel_context_put(ce);
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
igt_spinner_fini(&spin);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-01-29 04:43:15 +08:00
|
|
|
static const char *error_repr(int err)
|
|
|
|
{
|
|
|
|
return err ? "bad" : "good";
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_error_interrupt(void *arg)
|
|
|
|
{
|
|
|
|
static const struct error_phase {
|
|
|
|
enum { GOOD = 0, BAD = -EIO } error[2];
|
|
|
|
} phases[] = {
|
|
|
|
{ { BAD, GOOD } },
|
|
|
|
{ { BAD, BAD } },
|
|
|
|
{ { BAD, GOOD } },
|
|
|
|
{ { GOOD, GOOD } }, /* sentinel */
|
|
|
|
};
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We hook up the CS_MASTER_ERROR_INTERRUPT to have forewarning
|
|
|
|
* of invalid commands in user batches that will cause a GPU hang.
|
|
|
|
* This is a faster mechanism than using hangcheck/heartbeats, but
|
|
|
|
* only detects problems the HW knows about -- it will not warn when
|
|
|
|
* we kill the HW!
|
|
|
|
*
|
|
|
|
* To verify our detection and reset, we throw some invalid commands
|
|
|
|
* at the HW and wait for the interrupt.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!intel_has_reset_engine(gt))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
const struct error_phase *p;
|
|
|
|
int err = 0;
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2020-01-29 04:43:15 +08:00
|
|
|
|
|
|
|
for (p = phases; p->error[0] != GOOD; p++) {
|
|
|
|
struct i915_request *client[ARRAY_SIZE(phases->error)];
|
|
|
|
u32 *cs;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
memset(client, 0, sizeof(*client));
|
|
|
|
for (i = 0; i < ARRAY_SIZE(client); i++) {
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
intel_context_put(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rq->engine->emit_init_breadcrumb) {
|
|
|
|
err = rq->engine->emit_init_breadcrumb(rq);
|
|
|
|
if (err) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 2);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (p->error[i]) {
|
|
|
|
*cs++ = 0xdeadbeef;
|
|
|
|
*cs++ = 0xdeadbeef;
|
|
|
|
} else {
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
}
|
|
|
|
|
|
|
|
client[i] = i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
}
|
|
|
|
|
|
|
|
err = wait_for_submit(engine, client[0], HZ / 2);
|
|
|
|
if (err) {
|
|
|
|
pr_err("%s: first request did not start within time!\n",
|
|
|
|
engine->name);
|
|
|
|
err = -ETIME;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(client); i++) {
|
2020-02-11 04:57:18 +08:00
|
|
|
if (i915_request_wait(client[i], 0, HZ / 5) < 0)
|
|
|
|
pr_debug("%s: %s request incomplete!\n",
|
|
|
|
engine->name,
|
|
|
|
error_repr(p->error[i]));
|
|
|
|
|
|
|
|
if (!i915_request_started(client[i])) {
|
2020-03-31 17:14:56 +08:00
|
|
|
pr_err("%s: %s request not started!\n",
|
|
|
|
engine->name,
|
|
|
|
error_repr(p->error[i]));
|
2020-01-29 04:43:15 +08:00
|
|
|
err = -ETIME;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2020-02-11 04:57:18 +08:00
|
|
|
/* Kick the tasklet to process the error */
|
|
|
|
intel_engine_flush_submission(engine);
|
2020-01-29 04:43:15 +08:00
|
|
|
if (client[i]->fence.error != p->error[i]) {
|
2020-03-31 17:14:56 +08:00
|
|
|
pr_err("%s: %s request (%s) with wrong error code: %d\n",
|
2020-01-29 04:43:15 +08:00
|
|
|
engine->name,
|
|
|
|
error_repr(p->error[i]),
|
2020-03-31 17:14:56 +08:00
|
|
|
i915_request_completed(client[i]) ? "completed" : "running",
|
2020-01-29 04:43:15 +08:00
|
|
|
client[i]->fence.error);
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
for (i = 0; i < ARRAY_SIZE(client); i++)
|
|
|
|
if (client[i])
|
|
|
|
i915_request_put(client[i]);
|
|
|
|
if (err) {
|
|
|
|
pr_err("%s: failed at phase[%zd] { %d, %d }\n",
|
|
|
|
engine->name, p - phases,
|
|
|
|
p->error[0], p->error[1]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2020-01-29 04:43:15 +08:00
|
|
|
if (err) {
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-06-20 22:20:52 +08:00
|
|
|
static int
|
|
|
|
emit_semaphore_chain(struct i915_request *rq, struct i915_vma *vma, int idx)
|
|
|
|
{
|
|
|
|
u32 *cs;
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 10);
|
|
|
|
if (IS_ERR(cs))
|
|
|
|
return PTR_ERR(cs);
|
|
|
|
|
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_ENABLE;
|
|
|
|
|
|
|
|
*cs++ = MI_SEMAPHORE_WAIT |
|
|
|
|
MI_SEMAPHORE_GLOBAL_GTT |
|
|
|
|
MI_SEMAPHORE_POLL |
|
|
|
|
MI_SEMAPHORE_SAD_NEQ_SDD;
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = i915_ggtt_offset(vma) + 4 * idx;
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
if (idx > 0) {
|
|
|
|
*cs++ = MI_STORE_DWORD_IMM_GEN4 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_ggtt_offset(vma) + 4 * (idx - 1);
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = 1;
|
|
|
|
} else {
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_DISABLE;
|
|
|
|
|
|
|
|
intel_ring_advance(rq, cs);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct i915_request *
|
|
|
|
semaphore_queue(struct intel_engine_cs *engine, struct i915_vma *vma, int idx)
|
|
|
|
{
|
2019-12-22 00:03:24 +08:00
|
|
|
struct intel_context *ce;
|
2019-06-20 22:20:52 +08:00
|
|
|
struct i915_request *rq;
|
|
|
|
int err;
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce))
|
|
|
|
return ERR_CAST(ce);
|
2019-06-20 22:20:52 +08:00
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
rq = intel_context_create_request(ce);
|
2019-06-20 22:20:52 +08:00
|
|
|
if (IS_ERR(rq))
|
2019-12-22 00:03:24 +08:00
|
|
|
goto out_ce;
|
2019-06-20 22:20:52 +08:00
|
|
|
|
2019-10-18 15:20:27 +08:00
|
|
|
err = 0;
|
|
|
|
if (rq->engine->emit_init_breadcrumb)
|
|
|
|
err = rq->engine->emit_init_breadcrumb(rq);
|
|
|
|
if (err == 0)
|
|
|
|
err = emit_semaphore_chain(rq, vma, idx);
|
|
|
|
if (err == 0)
|
|
|
|
i915_request_get(rq);
|
2019-06-20 22:20:52 +08:00
|
|
|
i915_request_add(rq);
|
|
|
|
if (err)
|
|
|
|
rq = ERR_PTR(err);
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
out_ce:
|
|
|
|
intel_context_put(ce);
|
2019-06-20 22:20:52 +08:00
|
|
|
return rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
release_queue(struct intel_engine_cs *engine,
|
|
|
|
struct i915_vma *vma,
|
2019-10-18 15:20:27 +08:00
|
|
|
int idx, int prio)
|
2019-06-20 22:20:52 +08:00
|
|
|
{
|
|
|
|
struct i915_sched_attr attr = {
|
2019-10-18 15:20:27 +08:00
|
|
|
.priority = prio,
|
2019-06-20 22:20:52 +08:00
|
|
|
};
|
|
|
|
struct i915_request *rq;
|
|
|
|
u32 *cs;
|
|
|
|
|
2019-11-25 18:58:56 +08:00
|
|
|
rq = intel_engine_create_kernel_request(engine);
|
2019-06-20 22:20:52 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
return PTR_ERR(rq);
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 4);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
return PTR_ERR(cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_DWORD_IMM_GEN4 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_ggtt_offset(vma) + 4 * (idx - 1);
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = 1;
|
|
|
|
|
|
|
|
intel_ring_advance(rq, cs);
|
2019-10-18 15:20:27 +08:00
|
|
|
|
|
|
|
i915_request_get(rq);
|
2019-06-20 22:20:52 +08:00
|
|
|
i915_request_add(rq);
|
|
|
|
|
2019-10-18 15:20:27 +08:00
|
|
|
local_bh_disable();
|
2019-06-20 22:20:52 +08:00
|
|
|
engine->schedule(rq, &attr);
|
2019-10-18 15:20:27 +08:00
|
|
|
local_bh_enable(); /* kick tasklet */
|
|
|
|
|
|
|
|
i915_request_put(rq);
|
2019-06-20 22:20:52 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
slice_semaphore_queue(struct intel_engine_cs *outer,
|
|
|
|
struct i915_vma *vma,
|
|
|
|
int count)
|
|
|
|
{
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct i915_request *head;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err, i, n = 0;
|
|
|
|
|
|
|
|
head = semaphore_queue(outer, vma, n++);
|
|
|
|
if (IS_ERR(head))
|
|
|
|
return PTR_ERR(head);
|
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, outer->gt, id) {
|
2019-06-20 22:20:52 +08:00
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
struct i915_request *rq;
|
|
|
|
|
|
|
|
rq = semaphore_queue(engine, vma, n++);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
2019-10-18 15:20:27 +08:00
|
|
|
|
|
|
|
i915_request_put(rq);
|
2019-06-20 22:20:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-28 00:24:18 +08:00
|
|
|
err = release_queue(outer, vma, n, I915_PRIORITY_BARRIER);
|
2019-06-20 22:20:52 +08:00
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
2019-10-04 21:40:02 +08:00
|
|
|
if (i915_request_wait(head, 0,
|
2020-07-08 08:39:47 +08:00
|
|
|
2 * outer->gt->info.num_engines * (count + 2) * (count + 3)) < 0) {
|
2019-06-20 22:20:52 +08:00
|
|
|
pr_err("Failed to slice along semaphore chain of length (%d, %d)!\n",
|
|
|
|
count, n);
|
|
|
|
GEM_TRACE_DUMP();
|
2019-07-13 03:29:53 +08:00
|
|
|
intel_gt_set_wedged(outer->gt);
|
2019-06-20 22:20:52 +08:00
|
|
|
err = -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
i915_request_put(head);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_timeslice_preempt(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2019-06-20 22:20:52 +08:00
|
|
|
struct drm_i915_gem_object *obj;
|
2020-06-08 06:20:45 +08:00
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
2019-06-20 22:20:52 +08:00
|
|
|
struct i915_vma *vma;
|
|
|
|
void *vaddr;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If a request takes too long, we would like to give other users
|
|
|
|
* a fair go on the GPU. In particular, users may create batches
|
|
|
|
* that wait upon external input, where that input may even be
|
|
|
|
* supplied by another GPU job. To avoid blocking forever, we
|
|
|
|
* need to preempt the current task and replace it with another
|
|
|
|
* ready task.
|
|
|
|
*/
|
2019-10-29 17:16:32 +08:00
|
|
|
if (!IS_ACTIVE(CONFIG_DRM_I915_TIMESLICE_DURATION))
|
|
|
|
return 0;
|
2019-06-20 22:20:52 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
obj = i915_gem_object_create_internal(gt->i915, PAGE_SIZE);
|
2019-10-04 21:40:14 +08:00
|
|
|
if (IS_ERR(obj))
|
|
|
|
return PTR_ERR(obj);
|
2019-06-20 22:20:52 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
vma = i915_vma_instance(obj, >->ggtt->vm, NULL);
|
2019-06-20 22:20:52 +08:00
|
|
|
if (IS_ERR(vma)) {
|
|
|
|
err = PTR_ERR(vma);
|
|
|
|
goto err_obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
vaddr = i915_gem_object_pin_map(obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(vaddr)) {
|
|
|
|
err = PTR_ERR(vaddr);
|
|
|
|
goto err_obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = i915_vma_pin(vma, 0, 0, PIN_GLOBAL);
|
|
|
|
if (err)
|
|
|
|
goto err_map;
|
|
|
|
|
2020-01-31 22:26:10 +08:00
|
|
|
err = i915_vma_sync(vma);
|
|
|
|
if (err)
|
|
|
|
goto err_pin;
|
|
|
|
|
2020-06-08 06:20:45 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
2019-07-09 17:12:33 +08:00
|
|
|
|
2020-06-08 06:20:45 +08:00
|
|
|
memset(vaddr, 0, PAGE_SIZE);
|
2019-06-20 22:20:52 +08:00
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2020-06-08 06:20:45 +08:00
|
|
|
err = slice_semaphore_queue(engine, vma, 5);
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2020-06-08 06:20:45 +08:00
|
|
|
if (err)
|
|
|
|
goto err_pin;
|
2019-06-20 22:20:52 +08:00
|
|
|
|
2020-06-08 06:20:45 +08:00
|
|
|
if (igt_flush_test(gt->i915)) {
|
|
|
|
err = -EIO;
|
|
|
|
goto err_pin;
|
2019-06-20 22:20:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err_pin:
|
|
|
|
i915_vma_unpin(vma);
|
|
|
|
err_map:
|
|
|
|
i915_gem_object_unpin_map(obj);
|
|
|
|
err_obj:
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-02-13 22:01:50 +08:00
|
|
|
static struct i915_request *
|
|
|
|
create_rewinder(struct intel_context *ce,
|
|
|
|
struct i915_request *wait,
|
|
|
|
void *slot, int idx)
|
|
|
|
{
|
|
|
|
const u32 offset =
|
|
|
|
i915_ggtt_offset(ce->engine->status_page.vma) +
|
|
|
|
offset_in_page(slot);
|
|
|
|
struct i915_request *rq;
|
|
|
|
u32 *cs;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
return rq;
|
|
|
|
|
|
|
|
if (wait) {
|
|
|
|
err = i915_request_await_dma_fence(rq, &wait->fence);
|
|
|
|
if (err)
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
drm/i915/gt: Yield the timeslice if caught waiting on a user semaphore
If we find ourselves waiting on a MI_SEMAPHORE_WAIT, either within the
user batch or in our own preamble, the engine raises a
GT_WAIT_ON_SEMAPHORE interrupt. We can unmask that interrupt and so
respond to a semaphore wait by yielding the timeslice, if we have
another context to yield to!
The only real complication is that the interrupt is only generated for
the start of the semaphore wait, and is asynchronous to our
process_csb() -- that is, we may not have registered the timeslice before
we see the interrupt. To ensure we don't miss a potential semaphore
blocking forward progress (e.g. selftests/live_timeslice_preempt) we mark
the interrupt and apply it to the next timeslice regardless of whether it
was active at the time.
v2: We use semaphores in preempt-to-busy, within the timeslicing
implementation itself! Ergo, when we do insert a preemption due to an
expired timeslice, the new context may start with the missed semaphore
flagged by the retired context and be yielded, ad infinitum. To avoid
this, read the context id at the time of the semaphore interrupt and
only yield if that context is still active.
Fixes: 8ee36e048c98 ("drm/i915/execlists: Minimalistic timeslicing")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Cc: Kenneth Graunke <kenneth@whitecape.org>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200407130811.17321-1-chris@chris-wilson.co.uk
2020-04-07 21:08:11 +08:00
|
|
|
cs = intel_ring_begin(rq, 14);
|
2020-02-13 22:01:50 +08:00
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_ENABLE;
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
|
|
|
|
*cs++ = MI_SEMAPHORE_WAIT |
|
|
|
|
MI_SEMAPHORE_GLOBAL_GTT |
|
|
|
|
MI_SEMAPHORE_POLL |
|
drm/i915/gt: Yield the timeslice if caught waiting on a user semaphore
If we find ourselves waiting on a MI_SEMAPHORE_WAIT, either within the
user batch or in our own preamble, the engine raises a
GT_WAIT_ON_SEMAPHORE interrupt. We can unmask that interrupt and so
respond to a semaphore wait by yielding the timeslice, if we have
another context to yield to!
The only real complication is that the interrupt is only generated for
the start of the semaphore wait, and is asynchronous to our
process_csb() -- that is, we may not have registered the timeslice before
we see the interrupt. To ensure we don't miss a potential semaphore
blocking forward progress (e.g. selftests/live_timeslice_preempt) we mark
the interrupt and apply it to the next timeslice regardless of whether it
was active at the time.
v2: We use semaphores in preempt-to-busy, within the timeslicing
implementation itself! Ergo, when we do insert a preemption due to an
expired timeslice, the new context may start with the missed semaphore
flagged by the retired context and be yielded, ad infinitum. To avoid
this, read the context id at the time of the semaphore interrupt and
only yield if that context is still active.
Fixes: 8ee36e048c98 ("drm/i915/execlists: Minimalistic timeslicing")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Cc: Kenneth Graunke <kenneth@whitecape.org>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200407130811.17321-1-chris@chris-wilson.co.uk
2020-04-07 21:08:11 +08:00
|
|
|
MI_SEMAPHORE_SAD_GTE_SDD;
|
|
|
|
*cs++ = idx;
|
2020-02-13 22:01:50 +08:00
|
|
|
*cs++ = offset;
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_REGISTER_MEM_GEN8 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_mmio_reg_offset(RING_TIMESTAMP(rq->engine->mmio_base));
|
|
|
|
*cs++ = offset + idx * sizeof(u32);
|
|
|
|
*cs++ = 0;
|
|
|
|
|
drm/i915/gt: Yield the timeslice if caught waiting on a user semaphore
If we find ourselves waiting on a MI_SEMAPHORE_WAIT, either within the
user batch or in our own preamble, the engine raises a
GT_WAIT_ON_SEMAPHORE interrupt. We can unmask that interrupt and so
respond to a semaphore wait by yielding the timeslice, if we have
another context to yield to!
The only real complication is that the interrupt is only generated for
the start of the semaphore wait, and is asynchronous to our
process_csb() -- that is, we may not have registered the timeslice before
we see the interrupt. To ensure we don't miss a potential semaphore
blocking forward progress (e.g. selftests/live_timeslice_preempt) we mark
the interrupt and apply it to the next timeslice regardless of whether it
was active at the time.
v2: We use semaphores in preempt-to-busy, within the timeslicing
implementation itself! Ergo, when we do insert a preemption due to an
expired timeslice, the new context may start with the missed semaphore
flagged by the retired context and be yielded, ad infinitum. To avoid
this, read the context id at the time of the semaphore interrupt and
only yield if that context is still active.
Fixes: 8ee36e048c98 ("drm/i915/execlists: Minimalistic timeslicing")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Cc: Kenneth Graunke <kenneth@whitecape.org>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200407130811.17321-1-chris@chris-wilson.co.uk
2020-04-07 21:08:11 +08:00
|
|
|
*cs++ = MI_STORE_DWORD_IMM_GEN4 | MI_USE_GGTT;
|
|
|
|
*cs++ = offset;
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = idx + 1;
|
|
|
|
|
2020-02-13 22:01:50 +08:00
|
|
|
intel_ring_advance(rq, cs);
|
|
|
|
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_MASK;
|
|
|
|
err = 0;
|
|
|
|
err:
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (err) {
|
|
|
|
i915_request_put(rq);
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_timeslice_rewind(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The usual presumption on timeslice expiration is that we replace
|
|
|
|
* the active context with another. However, given a chain of
|
|
|
|
* dependencies we may end up with replacing the context with itself,
|
|
|
|
* but only a few of those requests, forcing us to rewind the
|
|
|
|
* RING_TAIL of the original request.
|
|
|
|
*/
|
|
|
|
if (!IS_ACTIVE(CONFIG_DRM_I915_TIMESLICE_DURATION))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
enum { A1, A2, B1 };
|
drm/i915/gt: Yield the timeslice if caught waiting on a user semaphore
If we find ourselves waiting on a MI_SEMAPHORE_WAIT, either within the
user batch or in our own preamble, the engine raises a
GT_WAIT_ON_SEMAPHORE interrupt. We can unmask that interrupt and so
respond to a semaphore wait by yielding the timeslice, if we have
another context to yield to!
The only real complication is that the interrupt is only generated for
the start of the semaphore wait, and is asynchronous to our
process_csb() -- that is, we may not have registered the timeslice before
we see the interrupt. To ensure we don't miss a potential semaphore
blocking forward progress (e.g. selftests/live_timeslice_preempt) we mark
the interrupt and apply it to the next timeslice regardless of whether it
was active at the time.
v2: We use semaphores in preempt-to-busy, within the timeslicing
implementation itself! Ergo, when we do insert a preemption due to an
expired timeslice, the new context may start with the missed semaphore
flagged by the retired context and be yielded, ad infinitum. To avoid
this, read the context id at the time of the semaphore interrupt and
only yield if that context is still active.
Fixes: 8ee36e048c98 ("drm/i915/execlists: Minimalistic timeslicing")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Cc: Kenneth Graunke <kenneth@whitecape.org>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200407130811.17321-1-chris@chris-wilson.co.uk
2020-04-07 21:08:11 +08:00
|
|
|
enum { X = 1, Z, Y };
|
2020-02-13 22:01:50 +08:00
|
|
|
struct i915_request *rq[3] = {};
|
|
|
|
struct intel_context *ce;
|
|
|
|
unsigned long timeslice;
|
|
|
|
int i, err = 0;
|
|
|
|
u32 *slot;
|
|
|
|
|
|
|
|
if (!intel_engine_has_timeslices(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A:rq1 -- semaphore wait, timestamp X
|
|
|
|
* A:rq2 -- write timestamp Y
|
|
|
|
*
|
|
|
|
* B:rq1 [await A:rq1] -- write timestamp Z
|
|
|
|
*
|
|
|
|
* Force timeslice, release semaphore.
|
|
|
|
*
|
|
|
|
* Expect execution/evaluation order XZY
|
|
|
|
*/
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2020-02-13 22:01:50 +08:00
|
|
|
timeslice = xchg(&engine->props.timeslice_duration_ms, 1);
|
|
|
|
|
|
|
|
slot = memset32(engine->status_page.addr + 1000, 0, 4);
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2020-06-17 02:55:18 +08:00
|
|
|
rq[A1] = create_rewinder(ce, NULL, slot, X);
|
|
|
|
if (IS_ERR(rq[A1])) {
|
2020-02-13 22:01:50 +08:00
|
|
|
intel_context_put(ce);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2020-06-17 02:55:18 +08:00
|
|
|
rq[A2] = create_rewinder(ce, NULL, slot, Y);
|
2020-02-13 22:01:50 +08:00
|
|
|
intel_context_put(ce);
|
2020-06-17 02:55:18 +08:00
|
|
|
if (IS_ERR(rq[A2]))
|
2020-02-13 22:01:50 +08:00
|
|
|
goto err;
|
|
|
|
|
2020-06-17 02:55:18 +08:00
|
|
|
err = wait_for_submit(engine, rq[A2], HZ / 2);
|
2020-02-13 22:01:50 +08:00
|
|
|
if (err) {
|
|
|
|
pr_err("%s: failed to submit first context\n",
|
|
|
|
engine->name);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2020-06-17 02:55:18 +08:00
|
|
|
rq[B1] = create_rewinder(ce, rq[A1], slot, Z);
|
2020-02-13 22:01:50 +08:00
|
|
|
intel_context_put(ce);
|
|
|
|
if (IS_ERR(rq[2]))
|
|
|
|
goto err;
|
|
|
|
|
2020-06-17 02:55:18 +08:00
|
|
|
err = wait_for_submit(engine, rq[B1], HZ / 2);
|
2020-02-13 22:01:50 +08:00
|
|
|
if (err) {
|
|
|
|
pr_err("%s: failed to submit second context\n",
|
|
|
|
engine->name);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ELSP[] = { { A:rq1, A:rq2 }, { B:rq1 } } */
|
2020-06-17 02:55:18 +08:00
|
|
|
ENGINE_TRACE(engine, "forcing tasklet for rewind\n");
|
drm/i915/gt: Yield the timeslice if caught waiting on a user semaphore
If we find ourselves waiting on a MI_SEMAPHORE_WAIT, either within the
user batch or in our own preamble, the engine raises a
GT_WAIT_ON_SEMAPHORE interrupt. We can unmask that interrupt and so
respond to a semaphore wait by yielding the timeslice, if we have
another context to yield to!
The only real complication is that the interrupt is only generated for
the start of the semaphore wait, and is asynchronous to our
process_csb() -- that is, we may not have registered the timeslice before
we see the interrupt. To ensure we don't miss a potential semaphore
blocking forward progress (e.g. selftests/live_timeslice_preempt) we mark
the interrupt and apply it to the next timeslice regardless of whether it
was active at the time.
v2: We use semaphores in preempt-to-busy, within the timeslicing
implementation itself! Ergo, when we do insert a preemption due to an
expired timeslice, the new context may start with the missed semaphore
flagged by the retired context and be yielded, ad infinitum. To avoid
this, read the context id at the time of the semaphore interrupt and
only yield if that context is still active.
Fixes: 8ee36e048c98 ("drm/i915/execlists: Minimalistic timeslicing")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Cc: Kenneth Graunke <kenneth@whitecape.org>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200407130811.17321-1-chris@chris-wilson.co.uk
2020-04-07 21:08:11 +08:00
|
|
|
if (i915_request_is_active(rq[A2])) { /* semaphore yielded! */
|
|
|
|
/* Wait for the timeslice to kick in */
|
|
|
|
del_timer(&engine->execlists.timer);
|
|
|
|
tasklet_hi_schedule(&engine->execlists.tasklet);
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
}
|
2020-02-13 22:01:50 +08:00
|
|
|
/* -> ELSP[] = { { A:rq1 }, { B:rq1 } } */
|
|
|
|
GEM_BUG_ON(!i915_request_is_active(rq[A1]));
|
|
|
|
GEM_BUG_ON(!i915_request_is_active(rq[B1]));
|
|
|
|
GEM_BUG_ON(i915_request_is_active(rq[A2]));
|
|
|
|
|
|
|
|
/* Release the hounds! */
|
|
|
|
slot[0] = 1;
|
|
|
|
wmb(); /* "pairs" with GPU; paranoid kick of internal CPU$ */
|
|
|
|
|
|
|
|
for (i = 1; i <= 3; i++) {
|
|
|
|
unsigned long timeout = jiffies + HZ / 2;
|
|
|
|
|
|
|
|
while (!READ_ONCE(slot[i]) &&
|
|
|
|
time_before(jiffies, timeout))
|
|
|
|
;
|
|
|
|
|
|
|
|
if (!time_before(jiffies, timeout)) {
|
|
|
|
pr_err("%s: rq[%d] timed out\n",
|
|
|
|
engine->name, i - 1);
|
|
|
|
err = -ETIME;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
pr_debug("%s: slot[%d]:%x\n", engine->name, i, slot[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* XZY: XZ < XY */
|
|
|
|
if (slot[Z] - slot[X] >= slot[Y] - slot[X]) {
|
|
|
|
pr_err("%s: timeslicing did not run context B [%u] before A [%u]!\n",
|
|
|
|
engine->name,
|
|
|
|
slot[Z] - slot[X],
|
|
|
|
slot[Y] - slot[X]);
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
err:
|
|
|
|
memset32(&slot[0], -1, 4);
|
|
|
|
wmb();
|
|
|
|
|
|
|
|
engine->props.timeslice_duration_ms = timeslice;
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2020-02-13 22:01:50 +08:00
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
i915_request_put(rq[i]);
|
|
|
|
if (igt_flush_test(gt->i915))
|
|
|
|
err = -EIO;
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-10-18 15:20:27 +08:00
|
|
|
static struct i915_request *nop_request(struct intel_engine_cs *engine)
|
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
|
2019-11-25 18:58:56 +08:00
|
|
|
rq = intel_engine_create_kernel_request(engine);
|
2019-10-18 15:20:27 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
return rq;
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
return rq;
|
|
|
|
}
|
|
|
|
|
2020-06-04 21:59:38 +08:00
|
|
|
static long slice_timeout(struct intel_engine_cs *engine)
|
2019-10-29 17:16:32 +08:00
|
|
|
{
|
2020-06-04 21:59:38 +08:00
|
|
|
long timeout;
|
|
|
|
|
|
|
|
/* Enough time for a timeslice to kick in, and kick out */
|
|
|
|
timeout = 2 * msecs_to_jiffies_timeout(timeslice(engine));
|
|
|
|
|
|
|
|
/* Enough time for the nop request to complete */
|
|
|
|
timeout += HZ / 5;
|
|
|
|
|
|
|
|
return timeout + 1;
|
2019-10-29 17:16:32 +08:00
|
|
|
}
|
|
|
|
|
2019-10-18 15:20:27 +08:00
|
|
|
static int live_timeslice_queue(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct drm_i915_gem_object *obj;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
struct i915_vma *vma;
|
|
|
|
void *vaddr;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Make sure that even if ELSP[0] and ELSP[1] are filled with
|
|
|
|
* timeslicing between them disabled, we *do* enable timeslicing
|
|
|
|
* if the queue demands it. (Normally, we do not submit if
|
|
|
|
* ELSP[1] is already occupied, so must rely on timeslicing to
|
|
|
|
* eject ELSP[0] in favour of the queue.)
|
|
|
|
*/
|
2019-10-29 17:16:32 +08:00
|
|
|
if (!IS_ACTIVE(CONFIG_DRM_I915_TIMESLICE_DURATION))
|
|
|
|
return 0;
|
2019-10-18 15:20:27 +08:00
|
|
|
|
|
|
|
obj = i915_gem_object_create_internal(gt->i915, PAGE_SIZE);
|
|
|
|
if (IS_ERR(obj))
|
|
|
|
return PTR_ERR(obj);
|
|
|
|
|
|
|
|
vma = i915_vma_instance(obj, >->ggtt->vm, NULL);
|
|
|
|
if (IS_ERR(vma)) {
|
|
|
|
err = PTR_ERR(vma);
|
|
|
|
goto err_obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
vaddr = i915_gem_object_pin_map(obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(vaddr)) {
|
|
|
|
err = PTR_ERR(vaddr);
|
|
|
|
goto err_obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = i915_vma_pin(vma, 0, 0, PIN_GLOBAL);
|
|
|
|
if (err)
|
|
|
|
goto err_map;
|
|
|
|
|
2020-01-31 22:26:10 +08:00
|
|
|
err = i915_vma_sync(vma);
|
|
|
|
if (err)
|
|
|
|
goto err_pin;
|
|
|
|
|
2019-10-18 15:20:27 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
struct i915_sched_attr attr = {
|
|
|
|
.priority = I915_USER_PRIORITY(I915_PRIORITY_MAX),
|
|
|
|
};
|
|
|
|
struct i915_request *rq, *nop;
|
|
|
|
|
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2019-10-18 15:20:27 +08:00
|
|
|
memset(vaddr, 0, PAGE_SIZE);
|
|
|
|
|
|
|
|
/* ELSP[0]: semaphore wait */
|
|
|
|
rq = semaphore_queue(engine, vma, 0);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
2019-12-22 00:03:24 +08:00
|
|
|
goto err_heartbeat;
|
2019-10-18 15:20:27 +08:00
|
|
|
}
|
|
|
|
engine->schedule(rq, &attr);
|
2020-01-06 19:42:30 +08:00
|
|
|
err = wait_for_submit(engine, rq, HZ / 2);
|
|
|
|
if (err) {
|
|
|
|
pr_err("%s: Timed out trying to submit semaphores\n",
|
|
|
|
engine->name);
|
|
|
|
goto err_rq;
|
|
|
|
}
|
2019-10-18 15:20:27 +08:00
|
|
|
|
|
|
|
/* ELSP[1]: nop request */
|
|
|
|
nop = nop_request(engine);
|
|
|
|
if (IS_ERR(nop)) {
|
|
|
|
err = PTR_ERR(nop);
|
2019-12-22 00:03:24 +08:00
|
|
|
goto err_rq;
|
2019-10-18 15:20:27 +08:00
|
|
|
}
|
2020-01-06 19:42:30 +08:00
|
|
|
err = wait_for_submit(engine, nop, HZ / 2);
|
2019-10-18 15:20:27 +08:00
|
|
|
i915_request_put(nop);
|
2020-01-06 19:42:30 +08:00
|
|
|
if (err) {
|
|
|
|
pr_err("%s: Timed out trying to submit nop\n",
|
|
|
|
engine->name);
|
|
|
|
goto err_rq;
|
|
|
|
}
|
2019-10-18 15:20:27 +08:00
|
|
|
|
|
|
|
GEM_BUG_ON(i915_request_completed(rq));
|
|
|
|
GEM_BUG_ON(execlists_active(&engine->execlists) != rq);
|
|
|
|
|
|
|
|
/* Queue: semaphore signal, matching priority as semaphore */
|
|
|
|
err = release_queue(engine, vma, 1, effective_prio(rq));
|
2019-12-22 00:03:24 +08:00
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
2019-10-18 15:20:27 +08:00
|
|
|
|
2020-04-04 03:02:09 +08:00
|
|
|
/* Wait until we ack the release_queue and start timeslicing */
|
|
|
|
do {
|
|
|
|
cond_resched();
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
} while (READ_ONCE(engine->execlists.pending[0]));
|
|
|
|
|
2019-10-29 17:16:32 +08:00
|
|
|
/* Timeslice every jiffy, so within 2 we should signal */
|
2020-06-04 21:59:38 +08:00
|
|
|
if (i915_request_wait(rq, 0, slice_timeout(engine)) < 0) {
|
2019-10-18 15:20:27 +08:00
|
|
|
struct drm_printer p =
|
|
|
|
drm_info_printer(gt->i915->drm.dev);
|
|
|
|
|
|
|
|
pr_err("%s: Failed to timeslice into queue\n",
|
|
|
|
engine->name);
|
|
|
|
intel_engine_dump(engine, &p,
|
|
|
|
"%s\n", engine->name);
|
|
|
|
|
|
|
|
memset(vaddr, 0xff, PAGE_SIZE);
|
|
|
|
err = -EIO;
|
|
|
|
}
|
2019-12-22 00:03:24 +08:00
|
|
|
err_rq:
|
2019-10-18 15:20:27 +08:00
|
|
|
i915_request_put(rq);
|
2019-12-22 00:03:24 +08:00
|
|
|
err_heartbeat:
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2019-10-18 15:20:27 +08:00
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-01-31 22:26:10 +08:00
|
|
|
err_pin:
|
2019-10-18 15:20:27 +08:00
|
|
|
i915_vma_unpin(vma);
|
|
|
|
err_map:
|
|
|
|
i915_gem_object_unpin_map(obj);
|
|
|
|
err_obj:
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-05-28 00:24:18 +08:00
|
|
|
static int live_timeslice_nopreempt(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
struct igt_spinner spin;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We should not timeslice into a request that is marked with
|
|
|
|
* I915_REQUEST_NOPREEMPT.
|
|
|
|
*/
|
|
|
|
if (!IS_ACTIVE(CONFIG_DRM_I915_TIMESLICE_DURATION))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (igt_spinner_init(&spin, gt))
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
unsigned long timeslice;
|
|
|
|
|
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2020-05-28 00:24:18 +08:00
|
|
|
timeslice = xchg(&engine->props.timeslice_duration_ms, 1);
|
|
|
|
|
|
|
|
/* Create an unpreemptible spinner */
|
|
|
|
|
|
|
|
rq = igt_spinner_create_request(&spin, ce, MI_ARB_CHECK);
|
|
|
|
intel_context_put(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out_heartbeat;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
if (!igt_wait_for_spinner(&spin, rq)) {
|
|
|
|
i915_request_put(rq);
|
|
|
|
err = -ETIME;
|
|
|
|
goto out_spin;
|
|
|
|
}
|
|
|
|
|
|
|
|
set_bit(I915_FENCE_FLAG_NOPREEMPT, &rq->fence.flags);
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
/* Followed by a maximum priority barrier (heartbeat) */
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce)) {
|
2020-06-16 22:54:52 +08:00
|
|
|
err = PTR_ERR(ce);
|
2020-05-28 00:24:18 +08:00
|
|
|
goto out_spin;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
intel_context_put(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out_spin;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_BARRIER;
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wait until the barrier is in ELSP, and we know timeslicing
|
|
|
|
* will have been activated.
|
|
|
|
*/
|
|
|
|
if (wait_for_submit(engine, rq, HZ / 2)) {
|
|
|
|
i915_request_put(rq);
|
|
|
|
err = -ETIME;
|
|
|
|
goto out_spin;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Since the ELSP[0] request is unpreemptible, it should not
|
|
|
|
* allow the maximum priority barrier through. Wait long
|
|
|
|
* enough to see if it is timesliced in by mistake.
|
|
|
|
*/
|
2020-06-04 21:59:38 +08:00
|
|
|
if (i915_request_wait(rq, 0, slice_timeout(engine)) >= 0) {
|
2020-05-28 00:24:18 +08:00
|
|
|
pr_err("%s: I915_PRIORITY_BARRIER request completed, bypassing no-preempt request\n",
|
|
|
|
engine->name);
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
out_spin:
|
|
|
|
igt_spinner_end(&spin);
|
|
|
|
out_heartbeat:
|
|
|
|
xchg(&engine->props.timeslice_duration_ms, timeslice);
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2020-05-28 00:24:18 +08:00
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (igt_flush_test(gt->i915)) {
|
|
|
|
err = -EIO;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
igt_spinner_fini(&spin);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-03-29 21:40:24 +08:00
|
|
|
static int live_busywait_preempt(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2019-03-29 21:40:24 +08:00
|
|
|
struct i915_gem_context *ctx_hi, *ctx_lo;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct drm_i915_gem_object *obj;
|
|
|
|
struct i915_vma *vma;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
u32 *map;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Verify that even without HAS_LOGICAL_RING_PREEMPTION, we can
|
|
|
|
* preempt the busywaits used to synchronise between rings.
|
|
|
|
*/
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
ctx_hi = kernel_context(gt->i915);
|
2019-03-29 21:40:24 +08:00
|
|
|
if (!ctx_hi)
|
2019-10-04 21:40:14 +08:00
|
|
|
return -ENOMEM;
|
2019-05-15 21:00:50 +08:00
|
|
|
ctx_hi->sched.priority =
|
|
|
|
I915_USER_PRIORITY(I915_CONTEXT_MAX_USER_PRIORITY);
|
2019-03-29 21:40:24 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
ctx_lo = kernel_context(gt->i915);
|
2019-03-29 21:40:24 +08:00
|
|
|
if (!ctx_lo)
|
|
|
|
goto err_ctx_hi;
|
2019-05-15 21:00:50 +08:00
|
|
|
ctx_lo->sched.priority =
|
|
|
|
I915_USER_PRIORITY(I915_CONTEXT_MIN_USER_PRIORITY);
|
2019-03-29 21:40:24 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
obj = i915_gem_object_create_internal(gt->i915, PAGE_SIZE);
|
2019-03-29 21:40:24 +08:00
|
|
|
if (IS_ERR(obj)) {
|
|
|
|
err = PTR_ERR(obj);
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
map = i915_gem_object_pin_map(obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(map)) {
|
|
|
|
err = PTR_ERR(map);
|
|
|
|
goto err_obj;
|
|
|
|
}
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
vma = i915_vma_instance(obj, >->ggtt->vm, NULL);
|
2019-03-29 21:40:24 +08:00
|
|
|
if (IS_ERR(vma)) {
|
|
|
|
err = PTR_ERR(vma);
|
|
|
|
goto err_map;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = i915_vma_pin(vma, 0, 0, PIN_GLOBAL);
|
|
|
|
if (err)
|
|
|
|
goto err_map;
|
|
|
|
|
2020-01-31 22:26:10 +08:00
|
|
|
err = i915_vma_sync(vma);
|
|
|
|
if (err)
|
|
|
|
goto err_vma;
|
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2019-03-29 21:40:24 +08:00
|
|
|
struct i915_request *lo, *hi;
|
|
|
|
struct igt_live_test t;
|
|
|
|
u32 *cs;
|
|
|
|
|
2019-07-09 17:12:33 +08:00
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
2019-03-29 21:40:24 +08:00
|
|
|
if (!intel_engine_can_store_dword(engine))
|
|
|
|
continue;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_live_test_begin(&t, gt->i915, __func__, engine->name)) {
|
2019-03-29 21:40:24 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We create two requests. The low priority request
|
|
|
|
* busywaits on a semaphore (inside the ringbuffer where
|
|
|
|
* is should be preemptible) and the high priority requests
|
|
|
|
* uses a MI_STORE_DWORD_IMM to update the semaphore value
|
|
|
|
* allowing the first request to complete. If preemption
|
|
|
|
* fails, we hang instead.
|
|
|
|
*/
|
|
|
|
|
2019-04-27 00:33:36 +08:00
|
|
|
lo = igt_request_alloc(ctx_lo, engine);
|
2019-03-29 21:40:24 +08:00
|
|
|
if (IS_ERR(lo)) {
|
|
|
|
err = PTR_ERR(lo);
|
|
|
|
goto err_vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
cs = intel_ring_begin(lo, 8);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
i915_request_add(lo);
|
|
|
|
goto err_vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_DWORD_IMM_GEN4 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_ggtt_offset(vma);
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = 1;
|
|
|
|
|
|
|
|
/* XXX Do we need a flush + invalidate here? */
|
|
|
|
|
|
|
|
*cs++ = MI_SEMAPHORE_WAIT |
|
|
|
|
MI_SEMAPHORE_GLOBAL_GTT |
|
|
|
|
MI_SEMAPHORE_POLL |
|
|
|
|
MI_SEMAPHORE_SAD_EQ_SDD;
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = i915_ggtt_offset(vma);
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
intel_ring_advance(lo, cs);
|
2019-11-21 15:10:43 +08:00
|
|
|
|
|
|
|
i915_request_get(lo);
|
2019-03-29 21:40:24 +08:00
|
|
|
i915_request_add(lo);
|
|
|
|
|
|
|
|
if (wait_for(READ_ONCE(*map), 10)) {
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(lo);
|
2019-03-29 21:40:24 +08:00
|
|
|
err = -ETIMEDOUT;
|
|
|
|
goto err_vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Low priority request should be busywaiting now */
|
2019-06-18 15:41:30 +08:00
|
|
|
if (i915_request_wait(lo, 0, 1) != -ETIME) {
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(lo);
|
2019-03-29 21:40:24 +08:00
|
|
|
pr_err("%s: Busywaiting request did not!\n",
|
|
|
|
engine->name);
|
|
|
|
err = -EIO;
|
|
|
|
goto err_vma;
|
|
|
|
}
|
|
|
|
|
2019-04-27 00:33:36 +08:00
|
|
|
hi = igt_request_alloc(ctx_hi, engine);
|
2019-03-29 21:40:24 +08:00
|
|
|
if (IS_ERR(hi)) {
|
|
|
|
err = PTR_ERR(hi);
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(lo);
|
2019-03-29 21:40:24 +08:00
|
|
|
goto err_vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
cs = intel_ring_begin(hi, 4);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
i915_request_add(hi);
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(lo);
|
2019-03-29 21:40:24 +08:00
|
|
|
goto err_vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_DWORD_IMM_GEN4 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_ggtt_offset(vma);
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
intel_ring_advance(hi, cs);
|
|
|
|
i915_request_add(hi);
|
|
|
|
|
2019-06-18 15:41:30 +08:00
|
|
|
if (i915_request_wait(lo, 0, HZ / 5) < 0) {
|
2019-10-16 20:02:49 +08:00
|
|
|
struct drm_printer p = drm_info_printer(gt->i915->drm.dev);
|
2019-03-29 21:40:24 +08:00
|
|
|
|
|
|
|
pr_err("%s: Failed to preempt semaphore busywait!\n",
|
|
|
|
engine->name);
|
|
|
|
|
|
|
|
intel_engine_dump(engine, &p, "%s\n", engine->name);
|
|
|
|
GEM_TRACE_DUMP();
|
|
|
|
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(lo);
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
2019-03-29 21:40:24 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_vma;
|
|
|
|
}
|
|
|
|
GEM_BUG_ON(READ_ONCE(*map));
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(lo);
|
2019-03-29 21:40:24 +08:00
|
|
|
|
|
|
|
if (igt_live_test_end(&t)) {
|
|
|
|
err = -EIO;
|
|
|
|
goto err_vma;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
err_vma:
|
|
|
|
i915_vma_unpin(vma);
|
|
|
|
err_map:
|
|
|
|
i915_gem_object_unpin_map(obj);
|
|
|
|
err_obj:
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
err_ctx_lo:
|
|
|
|
kernel_context_close(ctx_lo);
|
|
|
|
err_ctx_hi:
|
|
|
|
kernel_context_close(ctx_hi);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
static struct i915_request *
|
|
|
|
spinner_create_request(struct igt_spinner *spin,
|
|
|
|
struct i915_gem_context *ctx,
|
|
|
|
struct intel_engine_cs *engine,
|
|
|
|
u32 arb)
|
|
|
|
{
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
|
2019-08-08 19:06:12 +08:00
|
|
|
ce = i915_gem_context_get_engine(ctx, engine->legacy_idx);
|
2019-07-31 16:11:26 +08:00
|
|
|
if (IS_ERR(ce))
|
|
|
|
return ERR_CAST(ce);
|
|
|
|
|
|
|
|
rq = igt_spinner_create_request(spin, ce, arb);
|
|
|
|
intel_context_put(ce);
|
|
|
|
return rq;
|
|
|
|
}
|
|
|
|
|
2018-04-04 17:33:29 +08:00
|
|
|
static int live_preempt(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2018-04-04 17:33:29 +08:00
|
|
|
struct i915_gem_context *ctx_hi, *ctx_lo;
|
2018-11-30 16:02:53 +08:00
|
|
|
struct igt_spinner spin_hi, spin_lo;
|
2018-04-04 17:33:29 +08:00
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(gt->i915))
|
2018-04-04 17:33:29 +08:00
|
|
|
return 0;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (!(gt->i915->caps.scheduler & I915_SCHEDULER_CAP_PREEMPTION))
|
2019-03-06 22:25:01 +08:00
|
|
|
pr_err("Logical preemption supported, but not exposed\n");
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_spinner_init(&spin_hi, gt))
|
2019-10-04 21:40:14 +08:00
|
|
|
return -ENOMEM;
|
2018-04-04 17:33:29 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_spinner_init(&spin_lo, gt))
|
2018-04-04 17:33:29 +08:00
|
|
|
goto err_spin_hi;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
ctx_hi = kernel_context(gt->i915);
|
2018-04-04 17:33:29 +08:00
|
|
|
if (!ctx_hi)
|
|
|
|
goto err_spin_lo;
|
2018-10-01 20:32:03 +08:00
|
|
|
ctx_hi->sched.priority =
|
|
|
|
I915_USER_PRIORITY(I915_CONTEXT_MAX_USER_PRIORITY);
|
2018-04-04 17:33:29 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
ctx_lo = kernel_context(gt->i915);
|
2018-04-04 17:33:29 +08:00
|
|
|
if (!ctx_lo)
|
|
|
|
goto err_ctx_hi;
|
2018-10-01 20:32:03 +08:00
|
|
|
ctx_lo->sched.priority =
|
|
|
|
I915_USER_PRIORITY(I915_CONTEXT_MIN_USER_PRIORITY);
|
2018-04-04 17:33:29 +08:00
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2019-03-22 03:40:31 +08:00
|
|
|
struct igt_live_test t;
|
2018-04-04 17:33:29 +08:00
|
|
|
struct i915_request *rq;
|
|
|
|
|
2019-03-06 22:25:01 +08:00
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_live_test_begin(&t, gt->i915, __func__, engine->name)) {
|
2019-03-22 03:40:31 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq = spinner_create_request(&spin_lo, ctx_lo, engine,
|
|
|
|
MI_ARB_CHECK);
|
2018-04-04 17:33:29 +08:00
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
2018-11-30 16:02:53 +08:00
|
|
|
if (!igt_wait_for_spinner(&spin_lo, rq)) {
|
2018-04-04 17:33:29 +08:00
|
|
|
GEM_TRACE("lo spinner failed to start\n");
|
|
|
|
GEM_TRACE_DUMP();
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
2018-04-04 17:33:29 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq = spinner_create_request(&spin_hi, ctx_hi, engine,
|
|
|
|
MI_ARB_CHECK);
|
2018-04-04 17:33:29 +08:00
|
|
|
if (IS_ERR(rq)) {
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_end(&spin_lo);
|
2018-04-04 17:33:29 +08:00
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
2018-11-30 16:02:53 +08:00
|
|
|
if (!igt_wait_for_spinner(&spin_hi, rq)) {
|
2018-04-04 17:33:29 +08:00
|
|
|
GEM_TRACE("hi spinner failed to start\n");
|
|
|
|
GEM_TRACE_DUMP();
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
2018-04-04 17:33:29 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_end(&spin_hi);
|
|
|
|
igt_spinner_end(&spin_lo);
|
2019-03-22 03:40:31 +08:00
|
|
|
|
|
|
|
if (igt_live_test_end(&t)) {
|
2018-04-04 17:33:29 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
err_ctx_lo:
|
|
|
|
kernel_context_close(ctx_lo);
|
|
|
|
err_ctx_hi:
|
|
|
|
kernel_context_close(ctx_hi);
|
|
|
|
err_spin_lo:
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_fini(&spin_lo);
|
2018-04-04 17:33:29 +08:00
|
|
|
err_spin_hi:
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_fini(&spin_hi);
|
2018-04-04 17:33:29 +08:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_late_preempt(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2018-04-04 17:33:29 +08:00
|
|
|
struct i915_gem_context *ctx_hi, *ctx_lo;
|
2018-11-30 16:02:53 +08:00
|
|
|
struct igt_spinner spin_hi, spin_lo;
|
2018-04-04 17:33:29 +08:00
|
|
|
struct intel_engine_cs *engine;
|
2018-04-19 02:40:52 +08:00
|
|
|
struct i915_sched_attr attr = {};
|
2018-04-04 17:33:29 +08:00
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(gt->i915))
|
2018-04-04 17:33:29 +08:00
|
|
|
return 0;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_spinner_init(&spin_hi, gt))
|
2019-10-04 21:40:14 +08:00
|
|
|
return -ENOMEM;
|
2018-04-04 17:33:29 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_spinner_init(&spin_lo, gt))
|
2018-04-04 17:33:29 +08:00
|
|
|
goto err_spin_hi;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
ctx_hi = kernel_context(gt->i915);
|
2018-04-04 17:33:29 +08:00
|
|
|
if (!ctx_hi)
|
|
|
|
goto err_spin_lo;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
ctx_lo = kernel_context(gt->i915);
|
2018-04-04 17:33:29 +08:00
|
|
|
if (!ctx_lo)
|
|
|
|
goto err_ctx_hi;
|
|
|
|
|
2019-06-20 22:20:52 +08:00
|
|
|
/* Make sure ctx_lo stays before ctx_hi until we trigger preemption. */
|
|
|
|
ctx_lo->sched.priority = I915_USER_PRIORITY(1);
|
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2019-03-22 03:40:31 +08:00
|
|
|
struct igt_live_test t;
|
2018-04-04 17:33:29 +08:00
|
|
|
struct i915_request *rq;
|
|
|
|
|
2019-03-06 22:25:01 +08:00
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_live_test_begin(&t, gt->i915, __func__, engine->name)) {
|
2019-03-22 03:40:31 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq = spinner_create_request(&spin_lo, ctx_lo, engine,
|
|
|
|
MI_ARB_CHECK);
|
2018-04-04 17:33:29 +08:00
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
2018-11-30 16:02:53 +08:00
|
|
|
if (!igt_wait_for_spinner(&spin_lo, rq)) {
|
2018-04-04 17:33:29 +08:00
|
|
|
pr_err("First context failed to start\n");
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq = spinner_create_request(&spin_hi, ctx_hi, engine,
|
|
|
|
MI_NOOP);
|
2018-04-04 17:33:29 +08:00
|
|
|
if (IS_ERR(rq)) {
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_end(&spin_lo);
|
2018-04-04 17:33:29 +08:00
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
2018-11-30 16:02:53 +08:00
|
|
|
if (igt_wait_for_spinner(&spin_hi, rq)) {
|
2018-04-04 17:33:29 +08:00
|
|
|
pr_err("Second context overtook first?\n");
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
2018-10-01 20:32:03 +08:00
|
|
|
attr.priority = I915_USER_PRIORITY(I915_PRIORITY_MAX);
|
2018-04-19 02:40:52 +08:00
|
|
|
engine->schedule(rq, &attr);
|
2018-04-04 17:33:29 +08:00
|
|
|
|
2018-11-30 16:02:53 +08:00
|
|
|
if (!igt_wait_for_spinner(&spin_hi, rq)) {
|
2018-04-04 17:33:29 +08:00
|
|
|
pr_err("High priority context failed to preempt the low priority context\n");
|
|
|
|
GEM_TRACE_DUMP();
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_end(&spin_hi);
|
|
|
|
igt_spinner_end(&spin_lo);
|
2019-03-22 03:40:31 +08:00
|
|
|
|
|
|
|
if (igt_live_test_end(&t)) {
|
2018-04-04 17:33:29 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
err_ctx_lo:
|
|
|
|
kernel_context_close(ctx_lo);
|
|
|
|
err_ctx_hi:
|
|
|
|
kernel_context_close(ctx_hi);
|
|
|
|
err_spin_lo:
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_fini(&spin_lo);
|
2018-04-04 17:33:29 +08:00
|
|
|
err_spin_hi:
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_fini(&spin_hi);
|
2018-04-04 17:33:29 +08:00
|
|
|
return err;
|
|
|
|
|
|
|
|
err_wedged:
|
2018-11-30 16:02:53 +08:00
|
|
|
igt_spinner_end(&spin_hi);
|
|
|
|
igt_spinner_end(&spin_lo);
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
2018-04-04 17:33:29 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
struct preempt_client {
|
|
|
|
struct igt_spinner spin;
|
|
|
|
struct i915_gem_context *ctx;
|
|
|
|
};
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
static int preempt_client_init(struct intel_gt *gt, struct preempt_client *c)
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
c->ctx = kernel_context(gt->i915);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
if (!c->ctx)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_spinner_init(&c->spin, gt))
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
goto err_ctx;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_ctx:
|
|
|
|
kernel_context_close(c->ctx);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void preempt_client_fini(struct preempt_client *c)
|
|
|
|
{
|
|
|
|
igt_spinner_fini(&c->spin);
|
|
|
|
kernel_context_close(c->ctx);
|
|
|
|
}
|
|
|
|
|
2019-07-10 00:42:27 +08:00
|
|
|
static int live_nopreempt(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2019-07-10 00:42:27 +08:00
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct preempt_client a, b;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Verify that we can disable preemption for an individual request
|
|
|
|
* that may be being observed and not want to be interrupted.
|
|
|
|
*/
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(gt->i915))
|
2019-07-10 00:42:27 +08:00
|
|
|
return 0;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (preempt_client_init(gt, &a))
|
2019-10-04 21:40:14 +08:00
|
|
|
return -ENOMEM;
|
2019-10-16 20:02:49 +08:00
|
|
|
if (preempt_client_init(gt, &b))
|
2019-07-10 00:42:27 +08:00
|
|
|
goto err_client_a;
|
|
|
|
b.ctx->sched.priority = I915_USER_PRIORITY(I915_PRIORITY_MAX);
|
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2019-07-10 00:42:27 +08:00
|
|
|
struct i915_request *rq_a, *rq_b;
|
|
|
|
|
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
engine->execlists.preempt_hang.count = 0;
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq_a = spinner_create_request(&a.spin,
|
|
|
|
a.ctx, engine,
|
|
|
|
MI_ARB_CHECK);
|
2019-07-10 00:42:27 +08:00
|
|
|
if (IS_ERR(rq_a)) {
|
|
|
|
err = PTR_ERR(rq_a);
|
|
|
|
goto err_client_b;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Low priority client, but unpreemptable! */
|
2020-01-06 19:42:31 +08:00
|
|
|
__set_bit(I915_FENCE_FLAG_NOPREEMPT, &rq_a->fence.flags);
|
2019-07-10 00:42:27 +08:00
|
|
|
|
|
|
|
i915_request_add(rq_a);
|
|
|
|
if (!igt_wait_for_spinner(&a.spin, rq_a)) {
|
|
|
|
pr_err("First client failed to start\n");
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq_b = spinner_create_request(&b.spin,
|
|
|
|
b.ctx, engine,
|
|
|
|
MI_ARB_CHECK);
|
2019-07-10 00:42:27 +08:00
|
|
|
if (IS_ERR(rq_b)) {
|
|
|
|
err = PTR_ERR(rq_b);
|
|
|
|
goto err_client_b;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq_b);
|
|
|
|
|
|
|
|
/* B is much more important than A! (But A is unpreemptable.) */
|
|
|
|
GEM_BUG_ON(rq_prio(rq_b) <= rq_prio(rq_a));
|
|
|
|
|
|
|
|
/* Wait long enough for preemption and timeslicing */
|
|
|
|
if (igt_wait_for_spinner(&b.spin, rq_b)) {
|
|
|
|
pr_err("Second client started too early!\n");
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
|
|
|
igt_spinner_end(&a.spin);
|
|
|
|
|
|
|
|
if (!igt_wait_for_spinner(&b.spin, rq_b)) {
|
|
|
|
pr_err("Second client failed to start\n");
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
|
|
|
igt_spinner_end(&b.spin);
|
|
|
|
|
|
|
|
if (engine->execlists.preempt_hang.count) {
|
|
|
|
pr_err("Preemption recorded x%d; should have been suppressed!\n",
|
|
|
|
engine->execlists.preempt_hang.count);
|
|
|
|
err = -EINVAL;
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_flush_test(gt->i915))
|
2019-07-10 00:42:27 +08:00
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
err_client_b:
|
|
|
|
preempt_client_fini(&b);
|
|
|
|
err_client_a:
|
|
|
|
preempt_client_fini(&a);
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err_wedged:
|
|
|
|
igt_spinner_end(&b.spin);
|
|
|
|
igt_spinner_end(&a.spin);
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
2019-07-10 00:42:27 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_client_b;
|
|
|
|
}
|
|
|
|
|
2019-10-23 21:31:06 +08:00
|
|
|
struct live_preempt_cancel {
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct preempt_client a, b;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __cancel_active0(struct live_preempt_cancel *arg)
|
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct igt_live_test t;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
/* Preempt cancel of ELSP0 */
|
|
|
|
GEM_TRACE("%s(%s)\n", __func__, arg->engine->name);
|
|
|
|
if (igt_live_test_begin(&t, arg->engine->i915,
|
|
|
|
__func__, arg->engine->name))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
rq = spinner_create_request(&arg->a.spin,
|
|
|
|
arg->a.ctx, arg->engine,
|
|
|
|
MI_ARB_CHECK);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
return PTR_ERR(rq);
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
clear_bit(CONTEXT_BANNED, &rq->context->flags);
|
2019-10-23 21:31:06 +08:00
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (!igt_wait_for_spinner(&arg->a.spin, rq)) {
|
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
intel_context_set_banned(rq->context);
|
2019-10-23 21:31:06 +08:00
|
|
|
err = intel_engine_pulse(arg->engine);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
2020-02-27 16:57:18 +08:00
|
|
|
err = wait_for_reset(arg->engine, rq, HZ / 2);
|
|
|
|
if (err) {
|
|
|
|
pr_err("Cancelled inflight0 request did not reset\n");
|
2019-10-23 21:31:06 +08:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
i915_request_put(rq);
|
|
|
|
if (igt_live_test_end(&t))
|
|
|
|
err = -EIO;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __cancel_active1(struct live_preempt_cancel *arg)
|
|
|
|
{
|
|
|
|
struct i915_request *rq[2] = {};
|
|
|
|
struct igt_live_test t;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
/* Preempt cancel of ELSP1 */
|
|
|
|
GEM_TRACE("%s(%s)\n", __func__, arg->engine->name);
|
|
|
|
if (igt_live_test_begin(&t, arg->engine->i915,
|
|
|
|
__func__, arg->engine->name))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
rq[0] = spinner_create_request(&arg->a.spin,
|
|
|
|
arg->a.ctx, arg->engine,
|
|
|
|
MI_NOOP); /* no preemption */
|
|
|
|
if (IS_ERR(rq[0]))
|
|
|
|
return PTR_ERR(rq[0]);
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
clear_bit(CONTEXT_BANNED, &rq[0]->context->flags);
|
2019-10-23 21:31:06 +08:00
|
|
|
i915_request_get(rq[0]);
|
|
|
|
i915_request_add(rq[0]);
|
|
|
|
if (!igt_wait_for_spinner(&arg->a.spin, rq[0])) {
|
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq[1] = spinner_create_request(&arg->b.spin,
|
|
|
|
arg->b.ctx, arg->engine,
|
|
|
|
MI_ARB_CHECK);
|
|
|
|
if (IS_ERR(rq[1])) {
|
|
|
|
err = PTR_ERR(rq[1]);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
clear_bit(CONTEXT_BANNED, &rq[1]->context->flags);
|
2019-10-23 21:31:06 +08:00
|
|
|
i915_request_get(rq[1]);
|
|
|
|
err = i915_request_await_dma_fence(rq[1], &rq[0]->fence);
|
|
|
|
i915_request_add(rq[1]);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
intel_context_set_banned(rq[1]->context);
|
2019-10-23 21:31:06 +08:00
|
|
|
err = intel_engine_pulse(arg->engine);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
igt_spinner_end(&arg->a.spin);
|
2020-02-27 16:57:18 +08:00
|
|
|
err = wait_for_reset(arg->engine, rq[1], HZ / 2);
|
|
|
|
if (err)
|
2019-10-23 21:31:06 +08:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (rq[0]->fence.error != 0) {
|
|
|
|
pr_err("Normal inflight0 request did not complete\n");
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rq[1]->fence.error != -EIO) {
|
|
|
|
pr_err("Cancelled inflight1 request did not report -EIO\n");
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
i915_request_put(rq[1]);
|
|
|
|
i915_request_put(rq[0]);
|
|
|
|
if (igt_live_test_end(&t))
|
|
|
|
err = -EIO;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __cancel_queued(struct live_preempt_cancel *arg)
|
|
|
|
{
|
|
|
|
struct i915_request *rq[3] = {};
|
|
|
|
struct igt_live_test t;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
/* Full ELSP and one in the wings */
|
|
|
|
GEM_TRACE("%s(%s)\n", __func__, arg->engine->name);
|
|
|
|
if (igt_live_test_begin(&t, arg->engine->i915,
|
|
|
|
__func__, arg->engine->name))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
rq[0] = spinner_create_request(&arg->a.spin,
|
|
|
|
arg->a.ctx, arg->engine,
|
|
|
|
MI_ARB_CHECK);
|
|
|
|
if (IS_ERR(rq[0]))
|
|
|
|
return PTR_ERR(rq[0]);
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
clear_bit(CONTEXT_BANNED, &rq[0]->context->flags);
|
2019-10-23 21:31:06 +08:00
|
|
|
i915_request_get(rq[0]);
|
|
|
|
i915_request_add(rq[0]);
|
|
|
|
if (!igt_wait_for_spinner(&arg->a.spin, rq[0])) {
|
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq[1] = igt_request_alloc(arg->b.ctx, arg->engine);
|
|
|
|
if (IS_ERR(rq[1])) {
|
|
|
|
err = PTR_ERR(rq[1]);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
clear_bit(CONTEXT_BANNED, &rq[1]->context->flags);
|
2019-10-23 21:31:06 +08:00
|
|
|
i915_request_get(rq[1]);
|
|
|
|
err = i915_request_await_dma_fence(rq[1], &rq[0]->fence);
|
|
|
|
i915_request_add(rq[1]);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
rq[2] = spinner_create_request(&arg->b.spin,
|
|
|
|
arg->a.ctx, arg->engine,
|
|
|
|
MI_ARB_CHECK);
|
|
|
|
if (IS_ERR(rq[2])) {
|
|
|
|
err = PTR_ERR(rq[2]);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq[2]);
|
|
|
|
err = i915_request_await_dma_fence(rq[2], &rq[1]->fence);
|
|
|
|
i915_request_add(rq[2]);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
intel_context_set_banned(rq[2]->context);
|
2019-10-23 21:31:06 +08:00
|
|
|
err = intel_engine_pulse(arg->engine);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
2020-02-27 16:57:18 +08:00
|
|
|
err = wait_for_reset(arg->engine, rq[2], HZ / 2);
|
|
|
|
if (err)
|
2019-10-23 21:31:06 +08:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (rq[0]->fence.error != -EIO) {
|
|
|
|
pr_err("Cancelled inflight0 request did not report -EIO\n");
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rq[1]->fence.error != 0) {
|
|
|
|
pr_err("Normal inflight1 request did not complete\n");
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rq[2]->fence.error != -EIO) {
|
|
|
|
pr_err("Cancelled queued request did not report -EIO\n");
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
i915_request_put(rq[2]);
|
|
|
|
i915_request_put(rq[1]);
|
|
|
|
i915_request_put(rq[0]);
|
|
|
|
if (igt_live_test_end(&t))
|
|
|
|
err = -EIO;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __cancel_hostile(struct live_preempt_cancel *arg)
|
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
/* Preempt cancel non-preemptible spinner in ELSP0 */
|
2019-10-29 17:16:32 +08:00
|
|
|
if (!IS_ACTIVE(CONFIG_DRM_I915_PREEMPT_TIMEOUT))
|
2019-10-23 21:31:06 +08:00
|
|
|
return 0;
|
|
|
|
|
2020-04-03 04:58:39 +08:00
|
|
|
if (!intel_has_reset_engine(arg->engine->gt))
|
|
|
|
return 0;
|
|
|
|
|
2019-10-23 21:31:06 +08:00
|
|
|
GEM_TRACE("%s(%s)\n", __func__, arg->engine->name);
|
|
|
|
rq = spinner_create_request(&arg->a.spin,
|
|
|
|
arg->a.ctx, arg->engine,
|
|
|
|
MI_NOOP); /* preemption disabled */
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
return PTR_ERR(rq);
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
clear_bit(CONTEXT_BANNED, &rq->context->flags);
|
2019-10-23 21:31:06 +08:00
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (!igt_wait_for_spinner(&arg->a.spin, rq)) {
|
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-12-20 18:12:29 +08:00
|
|
|
intel_context_set_banned(rq->context);
|
2019-10-23 21:31:06 +08:00
|
|
|
err = intel_engine_pulse(arg->engine); /* force reset */
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
2020-02-27 16:57:18 +08:00
|
|
|
err = wait_for_reset(arg->engine, rq, HZ / 2);
|
|
|
|
if (err) {
|
|
|
|
pr_err("Cancelled inflight0 request did not reset\n");
|
2019-10-23 21:31:06 +08:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
i915_request_put(rq);
|
|
|
|
if (igt_flush_test(arg->engine->i915))
|
|
|
|
err = -EIO;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_preempt_cancel(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct live_preempt_cancel data;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* To cancel an inflight context, we need to first remove it from the
|
|
|
|
* GPU. That sounds like preemption! Plus a little bit of bookkeeping.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(gt->i915))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (preempt_client_init(gt, &data.a))
|
|
|
|
return -ENOMEM;
|
|
|
|
if (preempt_client_init(gt, &data.b))
|
|
|
|
goto err_client_a;
|
|
|
|
|
|
|
|
for_each_engine(data.engine, gt, id) {
|
|
|
|
if (!intel_engine_has_preemption(data.engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
err = __cancel_active0(&data);
|
|
|
|
if (err)
|
|
|
|
goto err_wedged;
|
|
|
|
|
|
|
|
err = __cancel_active1(&data);
|
|
|
|
if (err)
|
|
|
|
goto err_wedged;
|
|
|
|
|
|
|
|
err = __cancel_queued(&data);
|
|
|
|
if (err)
|
|
|
|
goto err_wedged;
|
|
|
|
|
|
|
|
err = __cancel_hostile(&data);
|
|
|
|
if (err)
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
err_client_b:
|
|
|
|
preempt_client_fini(&data.b);
|
|
|
|
err_client_a:
|
|
|
|
preempt_client_fini(&data.a);
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err_wedged:
|
|
|
|
GEM_TRACE_DUMP();
|
|
|
|
igt_spinner_end(&data.b.spin);
|
|
|
|
igt_spinner_end(&data.a.spin);
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
goto err_client_b;
|
|
|
|
}
|
|
|
|
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
static int live_suppress_self_preempt(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct i915_sched_attr attr = {
|
|
|
|
.priority = I915_USER_PRIORITY(I915_PRIORITY_MAX)
|
|
|
|
};
|
|
|
|
struct preempt_client a, b;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Verify that if a preemption request does not cause a change in
|
|
|
|
* the current execution order, the preempt-to-idle injection is
|
|
|
|
* skipped and that we do not accidentally apply it after the CS
|
|
|
|
* completion event.
|
|
|
|
*/
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(gt->i915))
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
return 0;
|
|
|
|
|
2020-02-19 06:33:20 +08:00
|
|
|
if (intel_uc_uses_guc_submission(>->uc))
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
return 0; /* presume black blox */
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (intel_vgpu_active(gt->i915))
|
2019-07-12 16:25:49 +08:00
|
|
|
return 0; /* GVT forces single port & request submission */
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (preempt_client_init(gt, &a))
|
2019-10-04 21:40:14 +08:00
|
|
|
return -ENOMEM;
|
2019-10-16 20:02:49 +08:00
|
|
|
if (preempt_client_init(gt, &b))
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
goto err_client_a;
|
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
struct i915_request *rq_a, *rq_b;
|
|
|
|
int depth;
|
|
|
|
|
2019-03-06 22:25:01 +08:00
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_flush_test(gt->i915))
|
2019-09-12 20:26:39 +08:00
|
|
|
goto err_wedged;
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
engine->execlists.preempt_hang.count = 0;
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq_a = spinner_create_request(&a.spin,
|
|
|
|
a.ctx, engine,
|
|
|
|
MI_NOOP);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
if (IS_ERR(rq_a)) {
|
|
|
|
err = PTR_ERR(rq_a);
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
goto err_client_b;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq_a);
|
|
|
|
if (!igt_wait_for_spinner(&a.spin, rq_a)) {
|
|
|
|
pr_err("First client failed to start\n");
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
2019-08-12 17:10:39 +08:00
|
|
|
/* Keep postponing the timer to avoid premature slicing */
|
|
|
|
mod_timer(&engine->execlists.timer, jiffies + HZ);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
for (depth = 0; depth < 8; depth++) {
|
2019-07-31 16:11:26 +08:00
|
|
|
rq_b = spinner_create_request(&b.spin,
|
|
|
|
b.ctx, engine,
|
|
|
|
MI_NOOP);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
if (IS_ERR(rq_b)) {
|
|
|
|
err = PTR_ERR(rq_b);
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
goto err_client_b;
|
|
|
|
}
|
|
|
|
i915_request_add(rq_b);
|
|
|
|
|
|
|
|
GEM_BUG_ON(i915_request_completed(rq_a));
|
|
|
|
engine->schedule(rq_a, &attr);
|
|
|
|
igt_spinner_end(&a.spin);
|
|
|
|
|
|
|
|
if (!igt_wait_for_spinner(&b.spin, rq_b)) {
|
|
|
|
pr_err("Second client failed to start\n");
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
|
|
|
swap(a, b);
|
|
|
|
rq_a = rq_b;
|
|
|
|
}
|
|
|
|
igt_spinner_end(&a.spin);
|
|
|
|
|
|
|
|
if (engine->execlists.preempt_hang.count) {
|
2019-08-12 17:10:39 +08:00
|
|
|
pr_err("Preemption on %s recorded x%d, depth %d; should have been suppressed!\n",
|
|
|
|
engine->name,
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
engine->execlists.preempt_hang.count,
|
|
|
|
depth);
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
err = -EINVAL;
|
|
|
|
goto err_client_b;
|
|
|
|
}
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_flush_test(gt->i915))
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
err_client_b:
|
|
|
|
preempt_client_fini(&b);
|
|
|
|
err_client_a:
|
|
|
|
preempt_client_fini(&a);
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err_wedged:
|
|
|
|
igt_spinner_end(&b.spin);
|
|
|
|
igt_spinner_end(&a.spin);
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_client_b;
|
|
|
|
}
|
|
|
|
|
2019-02-05 20:38:35 +08:00
|
|
|
static int live_chain_preempt(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2019-02-05 20:38:35 +08:00
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct preempt_client hi, lo;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Build a chain AB...BA between two contexts (A, B) and request
|
|
|
|
* preemption of the last request. It should then complete before
|
|
|
|
* the previously submitted spinner in B.
|
|
|
|
*/
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(gt->i915))
|
2019-02-05 20:38:35 +08:00
|
|
|
return 0;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (preempt_client_init(gt, &hi))
|
2019-10-04 21:40:14 +08:00
|
|
|
return -ENOMEM;
|
2019-02-05 20:38:35 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (preempt_client_init(gt, &lo))
|
2019-02-05 20:38:35 +08:00
|
|
|
goto err_client_hi;
|
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2019-02-05 20:38:35 +08:00
|
|
|
struct i915_sched_attr attr = {
|
|
|
|
.priority = I915_USER_PRIORITY(I915_PRIORITY_MAX),
|
|
|
|
};
|
2019-03-22 03:40:31 +08:00
|
|
|
struct igt_live_test t;
|
2019-03-22 03:40:30 +08:00
|
|
|
struct i915_request *rq;
|
|
|
|
int ring_size, count, i;
|
2019-02-05 20:38:35 +08:00
|
|
|
|
2019-03-06 22:25:01 +08:00
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq = spinner_create_request(&lo.spin,
|
|
|
|
lo.ctx, engine,
|
|
|
|
MI_ARB_CHECK);
|
2019-03-22 03:40:30 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
goto err_wedged;
|
2019-11-21 15:10:43 +08:00
|
|
|
|
|
|
|
i915_request_get(rq);
|
2019-03-22 03:40:30 +08:00
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
ring_size = rq->wa_tail - rq->head;
|
|
|
|
if (ring_size < 0)
|
|
|
|
ring_size += rq->ring->size;
|
|
|
|
ring_size = rq->ring->size / ring_size;
|
|
|
|
pr_debug("%s(%s): Using maximum of %d requests\n",
|
|
|
|
__func__, engine->name, ring_size);
|
2019-02-05 20:38:35 +08:00
|
|
|
|
2019-03-22 03:40:30 +08:00
|
|
|
igt_spinner_end(&lo.spin);
|
2019-06-18 15:41:30 +08:00
|
|
|
if (i915_request_wait(rq, 0, HZ / 2) < 0) {
|
2019-03-22 03:40:30 +08:00
|
|
|
pr_err("Timed out waiting to flush %s\n", engine->name);
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(rq);
|
2019-03-22 03:40:30 +08:00
|
|
|
goto err_wedged;
|
|
|
|
}
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(rq);
|
2019-03-22 03:40:30 +08:00
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_live_test_begin(&t, gt->i915, __func__, engine->name)) {
|
2019-03-22 03:40:31 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
|
2019-03-22 03:40:30 +08:00
|
|
|
for_each_prime_number_from(count, 1, ring_size) {
|
2019-07-31 16:11:26 +08:00
|
|
|
rq = spinner_create_request(&hi.spin,
|
|
|
|
hi.ctx, engine,
|
|
|
|
MI_ARB_CHECK);
|
2019-02-05 20:38:35 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
goto err_wedged;
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (!igt_wait_for_spinner(&hi.spin, rq))
|
|
|
|
goto err_wedged;
|
|
|
|
|
2019-07-31 16:11:26 +08:00
|
|
|
rq = spinner_create_request(&lo.spin,
|
|
|
|
lo.ctx, engine,
|
|
|
|
MI_ARB_CHECK);
|
2019-02-05 20:38:35 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
goto err_wedged;
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
for (i = 0; i < count; i++) {
|
2019-04-27 00:33:36 +08:00
|
|
|
rq = igt_request_alloc(lo.ctx, engine);
|
2019-02-05 20:38:35 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
goto err_wedged;
|
|
|
|
i915_request_add(rq);
|
|
|
|
}
|
|
|
|
|
2019-04-27 00:33:36 +08:00
|
|
|
rq = igt_request_alloc(hi.ctx, engine);
|
2019-02-05 20:38:35 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
goto err_wedged;
|
2019-11-21 15:10:43 +08:00
|
|
|
|
|
|
|
i915_request_get(rq);
|
2019-02-05 20:38:35 +08:00
|
|
|
i915_request_add(rq);
|
|
|
|
engine->schedule(rq, &attr);
|
|
|
|
|
|
|
|
igt_spinner_end(&hi.spin);
|
2019-06-18 15:41:30 +08:00
|
|
|
if (i915_request_wait(rq, 0, HZ / 5) < 0) {
|
2019-02-05 20:38:35 +08:00
|
|
|
struct drm_printer p =
|
2019-10-16 20:02:49 +08:00
|
|
|
drm_info_printer(gt->i915->drm.dev);
|
2019-02-05 20:38:35 +08:00
|
|
|
|
|
|
|
pr_err("Failed to preempt over chain of %d\n",
|
|
|
|
count);
|
|
|
|
intel_engine_dump(engine, &p,
|
|
|
|
"%s\n", engine->name);
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(rq);
|
2019-02-05 20:38:35 +08:00
|
|
|
goto err_wedged;
|
|
|
|
}
|
|
|
|
igt_spinner_end(&lo.spin);
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(rq);
|
2019-03-22 03:40:30 +08:00
|
|
|
|
2019-04-27 00:33:36 +08:00
|
|
|
rq = igt_request_alloc(lo.ctx, engine);
|
2019-03-22 03:40:30 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
goto err_wedged;
|
2019-11-21 15:10:43 +08:00
|
|
|
|
|
|
|
i915_request_get(rq);
|
2019-03-22 03:40:30 +08:00
|
|
|
i915_request_add(rq);
|
2019-11-21 15:10:43 +08:00
|
|
|
|
2019-06-18 15:41:30 +08:00
|
|
|
if (i915_request_wait(rq, 0, HZ / 5) < 0) {
|
2019-03-22 03:40:30 +08:00
|
|
|
struct drm_printer p =
|
2019-10-16 20:02:49 +08:00
|
|
|
drm_info_printer(gt->i915->drm.dev);
|
2019-03-22 03:40:30 +08:00
|
|
|
|
|
|
|
pr_err("Failed to flush low priority chain of %d requests\n",
|
|
|
|
count);
|
|
|
|
intel_engine_dump(engine, &p,
|
|
|
|
"%s\n", engine->name);
|
2019-11-21 15:10:43 +08:00
|
|
|
|
|
|
|
i915_request_put(rq);
|
2019-03-22 03:40:30 +08:00
|
|
|
goto err_wedged;
|
|
|
|
}
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(rq);
|
2019-02-05 20:38:35 +08:00
|
|
|
}
|
2019-03-22 03:40:31 +08:00
|
|
|
|
|
|
|
if (igt_live_test_end(&t)) {
|
|
|
|
err = -EIO;
|
|
|
|
goto err_wedged;
|
|
|
|
}
|
2019-02-05 20:38:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
err_client_lo:
|
|
|
|
preempt_client_fini(&lo);
|
|
|
|
err_client_hi:
|
|
|
|
preempt_client_fini(&hi);
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err_wedged:
|
|
|
|
igt_spinner_end(&hi.spin);
|
|
|
|
igt_spinner_end(&lo.spin);
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
2019-02-05 20:38:35 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_client_lo;
|
|
|
|
}
|
|
|
|
|
2019-11-15 06:57:35 +08:00
|
|
|
static int create_gang(struct intel_engine_cs *engine,
|
|
|
|
struct i915_request **prev)
|
|
|
|
{
|
|
|
|
struct drm_i915_gem_object *obj;
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct i915_vma *vma;
|
|
|
|
u32 *cs;
|
|
|
|
int err;
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
ce = intel_context_create(engine);
|
2019-11-15 06:57:35 +08:00
|
|
|
if (IS_ERR(ce))
|
|
|
|
return PTR_ERR(ce);
|
|
|
|
|
|
|
|
obj = i915_gem_object_create_internal(engine->i915, 4096);
|
|
|
|
if (IS_ERR(obj)) {
|
|
|
|
err = PTR_ERR(obj);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
vma = i915_vma_instance(obj, ce->vm, NULL);
|
|
|
|
if (IS_ERR(vma)) {
|
|
|
|
err = PTR_ERR(vma);
|
|
|
|
goto err_obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = i915_vma_pin(vma, 0, 0, PIN_USER);
|
|
|
|
if (err)
|
|
|
|
goto err_obj;
|
|
|
|
|
|
|
|
cs = i915_gem_object_pin_map(obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(cs))
|
|
|
|
goto err_obj;
|
|
|
|
|
|
|
|
/* Semaphore target: spin until zero */
|
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_ENABLE;
|
|
|
|
|
|
|
|
*cs++ = MI_SEMAPHORE_WAIT |
|
|
|
|
MI_SEMAPHORE_POLL |
|
|
|
|
MI_SEMAPHORE_SAD_EQ_SDD;
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = lower_32_bits(vma->node.start);
|
|
|
|
*cs++ = upper_32_bits(vma->node.start);
|
|
|
|
|
|
|
|
if (*prev) {
|
|
|
|
u64 offset = (*prev)->batch->node.start;
|
|
|
|
|
|
|
|
/* Terminate the spinner in the next lower priority batch. */
|
|
|
|
*cs++ = MI_STORE_DWORD_IMM_GEN4;
|
|
|
|
*cs++ = lower_32_bits(offset);
|
|
|
|
*cs++ = upper_32_bits(offset);
|
|
|
|
*cs++ = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_BATCH_BUFFER_END;
|
|
|
|
i915_gem_object_flush_map(obj);
|
|
|
|
i915_gem_object_unpin_map(obj);
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
goto err_obj;
|
|
|
|
|
2020-04-08 17:17:23 +08:00
|
|
|
rq->batch = i915_vma_get(vma);
|
2019-11-15 06:57:35 +08:00
|
|
|
i915_request_get(rq);
|
|
|
|
|
|
|
|
i915_vma_lock(vma);
|
|
|
|
err = i915_request_await_object(rq, vma->obj, false);
|
|
|
|
if (!err)
|
|
|
|
err = i915_vma_move_to_active(vma, rq, 0);
|
|
|
|
if (!err)
|
|
|
|
err = rq->engine->emit_bb_start(rq,
|
|
|
|
vma->node.start,
|
|
|
|
PAGE_SIZE, 0);
|
|
|
|
i915_vma_unlock(vma);
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
|
|
|
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
intel_context_put(ce);
|
|
|
|
|
2020-07-28 23:20:10 +08:00
|
|
|
rq->mock.link.next = &(*prev)->mock.link;
|
2019-11-15 06:57:35 +08:00
|
|
|
*prev = rq;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_rq:
|
2020-04-08 17:17:23 +08:00
|
|
|
i915_vma_put(rq->batch);
|
2019-11-15 06:57:35 +08:00
|
|
|
i915_request_put(rq);
|
|
|
|
err_obj:
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
err_ce:
|
|
|
|
intel_context_put(ce);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-17 07:37:33 +08:00
|
|
|
static int __live_preempt_ring(struct intel_engine_cs *engine,
|
|
|
|
struct igt_spinner *spin,
|
|
|
|
int queue_sz, int ring_sz)
|
|
|
|
{
|
|
|
|
struct intel_context *ce[2] = {};
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct igt_live_test t;
|
|
|
|
int err = 0;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (igt_live_test_begin(&t, engine->i915, __func__, engine->name))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
for (n = 0; n < ARRAY_SIZE(ce); n++) {
|
|
|
|
struct intel_context *tmp;
|
|
|
|
|
|
|
|
tmp = intel_context_create(engine);
|
|
|
|
if (IS_ERR(tmp)) {
|
|
|
|
err = PTR_ERR(tmp);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp->ring = __intel_context_ring_size(ring_sz);
|
|
|
|
|
|
|
|
err = intel_context_pin(tmp);
|
|
|
|
if (err) {
|
|
|
|
intel_context_put(tmp);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset32(tmp->ring->vaddr,
|
|
|
|
0xdeadbeef, /* trigger a hang if executed */
|
|
|
|
tmp->ring->vma->size / sizeof(u32));
|
|
|
|
|
|
|
|
ce[n] = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = igt_spinner_create_request(spin, ce[0], MI_ARB_CHECK);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_BARRIER;
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
if (!igt_wait_for_spinner(spin, rq)) {
|
|
|
|
intel_gt_set_wedged(engine->gt);
|
|
|
|
i915_request_put(rq);
|
|
|
|
err = -ETIME;
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fill the ring, until we will cause a wrap */
|
|
|
|
n = 0;
|
|
|
|
while (ce[0]->ring->tail - rq->wa_tail <= queue_sz) {
|
|
|
|
struct i915_request *tmp;
|
|
|
|
|
|
|
|
tmp = intel_context_create_request(ce[0]);
|
|
|
|
if (IS_ERR(tmp)) {
|
|
|
|
err = PTR_ERR(tmp);
|
|
|
|
i915_request_put(rq);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(tmp);
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
pr_debug("%s: Filled %d with %d nop tails {size:%x, tail:%x, emit:%x, rq.tail:%x}\n",
|
|
|
|
engine->name, queue_sz, n,
|
|
|
|
ce[0]->ring->size,
|
|
|
|
ce[0]->ring->tail,
|
|
|
|
ce[0]->ring->emit,
|
|
|
|
rq->tail);
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
/* Create a second request to preempt the first ring */
|
|
|
|
rq = intel_context_create_request(ce[1]);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_BARRIER;
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
err = wait_for_submit(engine, rq, HZ / 2);
|
|
|
|
i915_request_put(rq);
|
|
|
|
if (err) {
|
|
|
|
pr_err("%s: preemption request was not submited\n",
|
|
|
|
engine->name);
|
|
|
|
err = -ETIME;
|
|
|
|
}
|
|
|
|
|
|
|
|
pr_debug("%s: ring[0]:{ tail:%x, emit:%x }, ring[1]:{ tail:%x, emit:%x }\n",
|
|
|
|
engine->name,
|
|
|
|
ce[0]->ring->tail, ce[0]->ring->emit,
|
|
|
|
ce[1]->ring->tail, ce[1]->ring->emit);
|
|
|
|
|
|
|
|
err_ce:
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
igt_spinner_end(spin);
|
|
|
|
for (n = 0; n < ARRAY_SIZE(ce); n++) {
|
|
|
|
if (IS_ERR_OR_NULL(ce[n]))
|
|
|
|
break;
|
|
|
|
|
|
|
|
intel_context_unpin(ce[n]);
|
|
|
|
intel_context_put(ce[n]);
|
|
|
|
}
|
|
|
|
if (igt_live_test_end(&t))
|
|
|
|
err = -EIO;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_preempt_ring(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct igt_spinner spin;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that we rollback large chunks of a ring in order to do a
|
|
|
|
* preemption event. Similar to live_unlite_ring, but looking at
|
|
|
|
* ring size rather than the impact of intel_ring_direction().
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (igt_spinner_init(&spin, gt))
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!intel_engine_can_store_dword(engine))
|
|
|
|
continue;
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2020-06-17 07:37:33 +08:00
|
|
|
|
|
|
|
for (n = 0; n <= 3; n++) {
|
|
|
|
err = __live_preempt_ring(engine, &spin,
|
|
|
|
n * SZ_4K / 4, SZ_4K);
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2020-06-17 07:37:33 +08:00
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
igt_spinner_fini(&spin);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-11-15 06:57:35 +08:00
|
|
|
static int live_preempt_gang(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
|
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(gt->i915))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Build as long a chain of preempters as we can, with each
|
|
|
|
* request higher priority than the last. Once we are ready, we release
|
|
|
|
* the last batch which then precolates down the chain, each releasing
|
|
|
|
* the next oldest in turn. The intent is to simply push as hard as we
|
|
|
|
* can with the number of preemptions, trying to exceed narrow HW
|
|
|
|
* limits. At a minimum, we insist that we can sort all the user
|
|
|
|
* high priority levels into execution order.
|
|
|
|
*/
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
struct i915_request *rq = NULL;
|
|
|
|
struct igt_live_test t;
|
|
|
|
IGT_TIMEOUT(end_time);
|
|
|
|
int prio = 0;
|
|
|
|
int err = 0;
|
|
|
|
u32 *cs;
|
|
|
|
|
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (igt_live_test_begin(&t, gt->i915, __func__, engine->name))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
do {
|
|
|
|
struct i915_sched_attr attr = {
|
|
|
|
.priority = I915_USER_PRIORITY(prio++),
|
|
|
|
};
|
|
|
|
|
|
|
|
err = create_gang(engine, &rq);
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Submit each spinner at increasing priority */
|
|
|
|
engine->schedule(rq, &attr);
|
2020-06-08 06:20:45 +08:00
|
|
|
} while (prio <= I915_PRIORITY_MAX &&
|
|
|
|
!__igt_timeout(end_time, NULL));
|
2019-11-15 06:57:35 +08:00
|
|
|
pr_debug("%s: Preempt chain of %d requests\n",
|
|
|
|
engine->name, prio);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Such that the last spinner is the highest priority and
|
|
|
|
* should execute first. When that spinner completes,
|
|
|
|
* it will terminate the next lowest spinner until there
|
|
|
|
* are no more spinners and the gang is complete.
|
|
|
|
*/
|
|
|
|
cs = i915_gem_object_pin_map(rq->batch->obj, I915_MAP_WC);
|
|
|
|
if (!IS_ERR(cs)) {
|
|
|
|
*cs = 0;
|
|
|
|
i915_gem_object_unpin_map(rq->batch->obj);
|
|
|
|
} else {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (rq) { /* wait for each rq from highest to lowest prio */
|
2020-07-28 23:20:10 +08:00
|
|
|
struct i915_request *n = list_next_entry(rq, mock.link);
|
2019-11-15 06:57:35 +08:00
|
|
|
|
|
|
|
if (err == 0 && i915_request_wait(rq, 0, HZ / 5) < 0) {
|
|
|
|
struct drm_printer p =
|
|
|
|
drm_info_printer(engine->i915->drm.dev);
|
|
|
|
|
|
|
|
pr_err("Failed to flush chain of %d requests, at %d\n",
|
|
|
|
prio, rq_prio(rq) >> I915_USER_PRIORITY_SHIFT);
|
|
|
|
intel_engine_dump(engine, &p,
|
|
|
|
"%s\n", engine->name);
|
|
|
|
|
|
|
|
err = -ETIME;
|
|
|
|
}
|
|
|
|
|
2020-04-08 17:17:23 +08:00
|
|
|
i915_vma_put(rq->batch);
|
2019-11-15 06:57:35 +08:00
|
|
|
i915_request_put(rq);
|
|
|
|
rq = n;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (igt_live_test_end(&t))
|
|
|
|
err = -EIO;
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-04-22 18:09:03 +08:00
|
|
|
static struct i915_vma *
|
|
|
|
create_gpr_user(struct intel_engine_cs *engine,
|
|
|
|
struct i915_vma *result,
|
|
|
|
unsigned int offset)
|
|
|
|
{
|
|
|
|
struct drm_i915_gem_object *obj;
|
|
|
|
struct i915_vma *vma;
|
|
|
|
u32 *cs;
|
|
|
|
int err;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
obj = i915_gem_object_create_internal(engine->i915, 4096);
|
|
|
|
if (IS_ERR(obj))
|
|
|
|
return ERR_CAST(obj);
|
|
|
|
|
|
|
|
vma = i915_vma_instance(obj, result->vm, NULL);
|
|
|
|
if (IS_ERR(vma)) {
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
return vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = i915_vma_pin(vma, 0, 0, PIN_USER);
|
|
|
|
if (err) {
|
|
|
|
i915_vma_put(vma);
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
cs = i915_gem_object_pin_map(obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_vma_put(vma);
|
|
|
|
return ERR_CAST(cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* All GPR are clear for new contexts. We use GPR(0) as a constant */
|
|
|
|
*cs++ = MI_LOAD_REGISTER_IMM(1);
|
|
|
|
*cs++ = CS_GPR(engine, 0);
|
|
|
|
*cs++ = 1;
|
|
|
|
|
|
|
|
for (i = 1; i < NUM_GPR; i++) {
|
|
|
|
u64 addr;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Perform: GPR[i]++
|
|
|
|
*
|
|
|
|
* As we read and write into the context saved GPR[i], if
|
|
|
|
* we restart this batch buffer from an earlier point, we
|
|
|
|
* will repeat the increment and store a value > 1.
|
|
|
|
*/
|
|
|
|
*cs++ = MI_MATH(4);
|
|
|
|
*cs++ = MI_MATH_LOAD(MI_MATH_REG_SRCA, MI_MATH_REG(i));
|
|
|
|
*cs++ = MI_MATH_LOAD(MI_MATH_REG_SRCB, MI_MATH_REG(0));
|
|
|
|
*cs++ = MI_MATH_ADD;
|
|
|
|
*cs++ = MI_MATH_STORE(MI_MATH_REG(i), MI_MATH_REG_ACCU);
|
|
|
|
|
|
|
|
addr = result->node.start + offset + i * sizeof(*cs);
|
|
|
|
*cs++ = MI_STORE_REGISTER_MEM_GEN8;
|
|
|
|
*cs++ = CS_GPR(engine, 2 * i);
|
|
|
|
*cs++ = lower_32_bits(addr);
|
|
|
|
*cs++ = upper_32_bits(addr);
|
|
|
|
|
|
|
|
*cs++ = MI_SEMAPHORE_WAIT |
|
|
|
|
MI_SEMAPHORE_POLL |
|
|
|
|
MI_SEMAPHORE_SAD_GTE_SDD;
|
|
|
|
*cs++ = i;
|
|
|
|
*cs++ = lower_32_bits(result->node.start);
|
|
|
|
*cs++ = upper_32_bits(result->node.start);
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_BATCH_BUFFER_END;
|
|
|
|
i915_gem_object_flush_map(obj);
|
|
|
|
i915_gem_object_unpin_map(obj);
|
|
|
|
|
|
|
|
return vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct i915_vma *create_global(struct intel_gt *gt, size_t sz)
|
|
|
|
{
|
|
|
|
struct drm_i915_gem_object *obj;
|
|
|
|
struct i915_vma *vma;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
obj = i915_gem_object_create_internal(gt->i915, sz);
|
|
|
|
if (IS_ERR(obj))
|
|
|
|
return ERR_CAST(obj);
|
|
|
|
|
|
|
|
vma = i915_vma_instance(obj, >->ggtt->vm, NULL);
|
|
|
|
if (IS_ERR(vma)) {
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
return vma;
|
|
|
|
}
|
|
|
|
|
2020-08-19 22:08:54 +08:00
|
|
|
err = i915_ggtt_pin(vma, NULL, 0, 0);
|
2020-04-22 18:09:03 +08:00
|
|
|
if (err) {
|
|
|
|
i915_vma_put(vma);
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct i915_request *
|
|
|
|
create_gpr_client(struct intel_engine_cs *engine,
|
|
|
|
struct i915_vma *global,
|
|
|
|
unsigned int offset)
|
|
|
|
{
|
|
|
|
struct i915_vma *batch, *vma;
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce))
|
|
|
|
return ERR_CAST(ce);
|
|
|
|
|
|
|
|
vma = i915_vma_instance(global->obj, ce->vm, NULL);
|
|
|
|
if (IS_ERR(vma)) {
|
|
|
|
err = PTR_ERR(vma);
|
|
|
|
goto out_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = i915_vma_pin(vma, 0, 0, PIN_USER);
|
|
|
|
if (err)
|
|
|
|
goto out_ce;
|
|
|
|
|
|
|
|
batch = create_gpr_user(engine, vma, offset);
|
|
|
|
if (IS_ERR(batch)) {
|
|
|
|
err = PTR_ERR(batch);
|
|
|
|
goto out_vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out_batch;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_vma_lock(vma);
|
|
|
|
err = i915_request_await_object(rq, vma->obj, false);
|
|
|
|
if (!err)
|
|
|
|
err = i915_vma_move_to_active(vma, rq, 0);
|
|
|
|
i915_vma_unlock(vma);
|
|
|
|
|
|
|
|
i915_vma_lock(batch);
|
|
|
|
if (!err)
|
|
|
|
err = i915_request_await_object(rq, batch->obj, false);
|
|
|
|
if (!err)
|
|
|
|
err = i915_vma_move_to_active(batch, rq, 0);
|
|
|
|
if (!err)
|
|
|
|
err = rq->engine->emit_bb_start(rq,
|
|
|
|
batch->node.start,
|
|
|
|
PAGE_SIZE, 0);
|
|
|
|
i915_vma_unlock(batch);
|
|
|
|
i915_vma_unpin(batch);
|
|
|
|
|
|
|
|
if (!err)
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
out_batch:
|
|
|
|
i915_vma_put(batch);
|
|
|
|
out_vma:
|
|
|
|
i915_vma_unpin(vma);
|
|
|
|
out_ce:
|
|
|
|
intel_context_put(ce);
|
|
|
|
return err ? ERR_PTR(err) : rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int preempt_user(struct intel_engine_cs *engine,
|
|
|
|
struct i915_vma *global,
|
|
|
|
int id)
|
|
|
|
{
|
|
|
|
struct i915_sched_attr attr = {
|
|
|
|
.priority = I915_PRIORITY_MAX
|
|
|
|
};
|
|
|
|
struct i915_request *rq;
|
|
|
|
int err = 0;
|
|
|
|
u32 *cs;
|
|
|
|
|
|
|
|
rq = intel_engine_create_kernel_request(engine);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
return PTR_ERR(rq);
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 4);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
return PTR_ERR(cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_DWORD_IMM_GEN4 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_ggtt_offset(global);
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = id;
|
|
|
|
|
|
|
|
intel_ring_advance(rq, cs);
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
engine->schedule(rq, &attr);
|
|
|
|
|
|
|
|
if (i915_request_wait(rq, 0, HZ / 2) < 0)
|
|
|
|
err = -ETIME;
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_preempt_user(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct i915_vma *global;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
u32 *result;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(gt->i915))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* In our other tests, we look at preemption in carefully
|
|
|
|
* controlled conditions in the ringbuffer. Since most of the
|
|
|
|
* time is spent in user batches, most of our preemptions naturally
|
|
|
|
* occur there. We want to verify that when we preempt inside a batch
|
|
|
|
* we continue on from the current instruction and do not roll back
|
|
|
|
* to the start, or another earlier arbitration point.
|
|
|
|
*
|
|
|
|
* To verify this, we create a batch which is a mixture of
|
|
|
|
* MI_MATH (gpr++) MI_SRM (gpr) and preemption points. Then with
|
|
|
|
* a few preempting contexts thrown into the mix, we look for any
|
|
|
|
* repeated instructions (which show up as incorrect values).
|
|
|
|
*/
|
|
|
|
|
|
|
|
global = create_global(gt, 4096);
|
|
|
|
if (IS_ERR(global))
|
|
|
|
return PTR_ERR(global);
|
|
|
|
|
|
|
|
result = i915_gem_object_pin_map(global->obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(result)) {
|
|
|
|
i915_vma_unpin_and_release(&global, 0);
|
|
|
|
return PTR_ERR(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
struct i915_request *client[3] = {};
|
|
|
|
struct igt_live_test t;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (IS_GEN(gt->i915, 8) && engine->class != RENDER_CLASS)
|
|
|
|
continue; /* we need per-context GPR */
|
|
|
|
|
|
|
|
if (igt_live_test_begin(&t, gt->i915, __func__, engine->name)) {
|
|
|
|
err = -EIO;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(result, 0, 4096);
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(client); i++) {
|
|
|
|
struct i915_request *rq;
|
|
|
|
|
|
|
|
rq = create_gpr_client(engine, global,
|
|
|
|
NUM_GPR * i * sizeof(u32));
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
goto end_test;
|
|
|
|
|
|
|
|
client[i] = rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Continuously preempt the set of 3 running contexts */
|
|
|
|
for (i = 1; i <= NUM_GPR; i++) {
|
|
|
|
err = preempt_user(engine, global, i);
|
|
|
|
if (err)
|
|
|
|
goto end_test;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (READ_ONCE(result[0]) != NUM_GPR) {
|
|
|
|
pr_err("%s: Failed to release semaphore\n",
|
|
|
|
engine->name);
|
|
|
|
err = -EIO;
|
|
|
|
goto end_test;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(client); i++) {
|
|
|
|
int gpr;
|
|
|
|
|
|
|
|
if (i915_request_wait(client[i], 0, HZ / 2) < 0) {
|
|
|
|
err = -ETIME;
|
|
|
|
goto end_test;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (gpr = 1; gpr < NUM_GPR; gpr++) {
|
|
|
|
if (result[NUM_GPR * i + gpr] != 1) {
|
|
|
|
pr_err("%s: Invalid result, client %d, gpr %d, result: %d\n",
|
|
|
|
engine->name,
|
|
|
|
i, gpr, result[NUM_GPR * i + gpr]);
|
|
|
|
err = -EINVAL;
|
|
|
|
goto end_test;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
end_test:
|
|
|
|
for (i = 0; i < ARRAY_SIZE(client); i++) {
|
|
|
|
if (!client[i])
|
|
|
|
break;
|
|
|
|
|
|
|
|
i915_request_put(client[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Flush the semaphores on error */
|
|
|
|
smp_store_mb(result[0], -1);
|
|
|
|
if (igt_live_test_end(&t))
|
|
|
|
err = -EIO;
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_vma_unpin_and_release(&global, I915_VMA_RELEASE_MAP);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-10-23 21:31:05 +08:00
|
|
|
static int live_preempt_timeout(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct i915_gem_context *ctx_hi, *ctx_lo;
|
|
|
|
struct igt_spinner spin_lo;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that we force preemption to occur by cancelling the previous
|
|
|
|
* context if it refuses to yield the GPU.
|
|
|
|
*/
|
2019-10-29 17:16:32 +08:00
|
|
|
if (!IS_ACTIVE(CONFIG_DRM_I915_PREEMPT_TIMEOUT))
|
2019-10-23 21:31:05 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(gt->i915))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!intel_has_reset_engine(gt))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (igt_spinner_init(&spin_lo, gt))
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
ctx_hi = kernel_context(gt->i915);
|
|
|
|
if (!ctx_hi)
|
|
|
|
goto err_spin_lo;
|
|
|
|
ctx_hi->sched.priority =
|
|
|
|
I915_USER_PRIORITY(I915_CONTEXT_MAX_USER_PRIORITY);
|
|
|
|
|
|
|
|
ctx_lo = kernel_context(gt->i915);
|
|
|
|
if (!ctx_lo)
|
|
|
|
goto err_ctx_hi;
|
|
|
|
ctx_lo->sched.priority =
|
|
|
|
I915_USER_PRIORITY(I915_CONTEXT_MIN_USER_PRIORITY);
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
unsigned long saved_timeout;
|
|
|
|
struct i915_request *rq;
|
|
|
|
|
|
|
|
if (!intel_engine_has_preemption(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
rq = spinner_create_request(&spin_lo, ctx_lo, engine,
|
|
|
|
MI_NOOP); /* preemption disabled */
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (!igt_wait_for_spinner(&spin_lo, rq)) {
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -EIO;
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = igt_request_alloc(ctx_hi, engine);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
igt_spinner_end(&spin_lo);
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Flush the previous CS ack before changing timeouts */
|
|
|
|
while (READ_ONCE(engine->execlists.pending[0]))
|
|
|
|
cpu_relax();
|
|
|
|
|
|
|
|
saved_timeout = engine->props.preempt_timeout_ms;
|
|
|
|
engine->props.preempt_timeout_ms = 1; /* in ms, -> 1 jiffie */
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
engine->props.preempt_timeout_ms = saved_timeout;
|
|
|
|
|
|
|
|
if (i915_request_wait(rq, 0, HZ / 10) < 0) {
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
i915_request_put(rq);
|
|
|
|
err = -ETIME;
|
|
|
|
goto err_ctx_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
igt_spinner_end(&spin_lo);
|
|
|
|
i915_request_put(rq);
|
|
|
|
}
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
err_ctx_lo:
|
|
|
|
kernel_context_close(ctx_lo);
|
|
|
|
err_ctx_hi:
|
|
|
|
kernel_context_close(ctx_hi);
|
|
|
|
err_spin_lo:
|
|
|
|
igt_spinner_fini(&spin_lo);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2018-09-25 16:31:58 +08:00
|
|
|
static int random_range(struct rnd_state *rnd, int min, int max)
|
|
|
|
{
|
|
|
|
return i915_prandom_u32_max_state(max - min, rnd) + min;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int random_priority(struct rnd_state *rnd)
|
|
|
|
{
|
|
|
|
return random_range(rnd, I915_PRIORITY_MIN, I915_PRIORITY_MAX);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct preempt_smoke {
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt;
|
2018-09-25 16:31:58 +08:00
|
|
|
struct i915_gem_context **contexts;
|
2018-10-01 20:32:01 +08:00
|
|
|
struct intel_engine_cs *engine;
|
2018-10-01 20:32:02 +08:00
|
|
|
struct drm_i915_gem_object *batch;
|
2018-09-25 16:31:58 +08:00
|
|
|
unsigned int ncontext;
|
|
|
|
struct rnd_state prng;
|
2018-10-01 20:32:01 +08:00
|
|
|
unsigned long count;
|
2018-09-25 16:31:58 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct i915_gem_context *smoke_context(struct preempt_smoke *smoke)
|
|
|
|
{
|
|
|
|
return smoke->contexts[i915_prandom_u32_max_state(smoke->ncontext,
|
|
|
|
&smoke->prng)];
|
|
|
|
}
|
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
static int smoke_submit(struct preempt_smoke *smoke,
|
|
|
|
struct i915_gem_context *ctx, int prio,
|
|
|
|
struct drm_i915_gem_object *batch)
|
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct i915_vma *vma = NULL;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
if (batch) {
|
2019-10-04 21:40:09 +08:00
|
|
|
struct i915_address_space *vm;
|
|
|
|
|
|
|
|
vm = i915_gem_context_get_vm_rcu(ctx);
|
|
|
|
vma = i915_vma_instance(batch, vm, NULL);
|
|
|
|
i915_vm_put(vm);
|
2018-10-01 20:32:02 +08:00
|
|
|
if (IS_ERR(vma))
|
|
|
|
return PTR_ERR(vma);
|
|
|
|
|
|
|
|
err = i915_vma_pin(vma, 0, 0, PIN_USER);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx->sched.priority = prio;
|
|
|
|
|
2019-04-27 00:33:36 +08:00
|
|
|
rq = igt_request_alloc(ctx, smoke->engine);
|
2018-10-01 20:32:02 +08:00
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto unpin;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vma) {
|
2019-05-28 17:29:51 +08:00
|
|
|
i915_vma_lock(vma);
|
2019-08-19 19:20:33 +08:00
|
|
|
err = i915_request_await_object(rq, vma->obj, false);
|
2018-10-01 20:32:02 +08:00
|
|
|
if (!err)
|
|
|
|
err = i915_vma_move_to_active(vma, rq, 0);
|
2019-08-19 19:20:33 +08:00
|
|
|
if (!err)
|
|
|
|
err = rq->engine->emit_bb_start(rq,
|
|
|
|
vma->node.start,
|
|
|
|
PAGE_SIZE, 0);
|
2019-05-28 17:29:51 +08:00
|
|
|
i915_vma_unlock(vma);
|
2018-10-01 20:32:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
unpin:
|
|
|
|
if (vma)
|
|
|
|
i915_vma_unpin(vma);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2018-10-01 20:32:01 +08:00
|
|
|
static int smoke_crescendo_thread(void *arg)
|
|
|
|
{
|
|
|
|
struct preempt_smoke *smoke = arg;
|
|
|
|
IGT_TIMEOUT(end_time);
|
|
|
|
unsigned long count;
|
|
|
|
|
|
|
|
count = 0;
|
|
|
|
do {
|
|
|
|
struct i915_gem_context *ctx = smoke_context(smoke);
|
2018-10-01 20:32:02 +08:00
|
|
|
int err;
|
2018-10-01 20:32:01 +08:00
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
err = smoke_submit(smoke,
|
|
|
|
ctx, count % I915_PRIORITY_MAX,
|
|
|
|
smoke->batch);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2018-10-01 20:32:01 +08:00
|
|
|
|
|
|
|
count++;
|
2020-06-08 06:20:45 +08:00
|
|
|
} while (count < smoke->ncontext && !__igt_timeout(end_time, NULL));
|
2018-10-01 20:32:01 +08:00
|
|
|
|
|
|
|
smoke->count = count;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
static int smoke_crescendo(struct preempt_smoke *smoke, unsigned int flags)
|
|
|
|
#define BATCH BIT(0)
|
2018-09-25 16:31:58 +08:00
|
|
|
{
|
2018-10-01 20:32:01 +08:00
|
|
|
struct task_struct *tsk[I915_NUM_ENGINES] = {};
|
|
|
|
struct preempt_smoke arg[I915_NUM_ENGINES];
|
2018-09-25 16:31:58 +08:00
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
unsigned long count;
|
2018-10-01 20:32:01 +08:00
|
|
|
int err = 0;
|
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, smoke->gt, id) {
|
2018-10-01 20:32:01 +08:00
|
|
|
arg[id] = *smoke;
|
|
|
|
arg[id].engine = engine;
|
2018-10-01 20:32:02 +08:00
|
|
|
if (!(flags & BATCH))
|
|
|
|
arg[id].batch = NULL;
|
2018-10-01 20:32:01 +08:00
|
|
|
arg[id].count = 0;
|
|
|
|
|
|
|
|
tsk[id] = kthread_run(smoke_crescendo_thread, &arg,
|
|
|
|
"igt/smoke:%d", id);
|
|
|
|
if (IS_ERR(tsk[id])) {
|
|
|
|
err = PTR_ERR(tsk[id]);
|
|
|
|
break;
|
|
|
|
}
|
2018-10-02 21:29:27 +08:00
|
|
|
get_task_struct(tsk[id]);
|
2018-10-01 20:32:01 +08:00
|
|
|
}
|
2018-09-25 16:31:58 +08:00
|
|
|
|
2019-11-01 16:49:40 +08:00
|
|
|
yield(); /* start all threads before we kthread_stop() */
|
|
|
|
|
2018-10-01 20:32:01 +08:00
|
|
|
count = 0;
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, smoke->gt, id) {
|
2018-10-01 20:32:01 +08:00
|
|
|
int status;
|
2018-09-25 16:31:58 +08:00
|
|
|
|
2018-10-01 20:32:01 +08:00
|
|
|
if (IS_ERR_OR_NULL(tsk[id]))
|
|
|
|
continue;
|
2018-09-25 16:31:58 +08:00
|
|
|
|
2018-10-01 20:32:01 +08:00
|
|
|
status = kthread_stop(tsk[id]);
|
|
|
|
if (status && !err)
|
|
|
|
err = status;
|
2018-09-25 16:31:58 +08:00
|
|
|
|
2018-10-01 20:32:01 +08:00
|
|
|
count += arg[id].count;
|
2018-10-02 21:29:27 +08:00
|
|
|
|
|
|
|
put_task_struct(tsk[id]);
|
2018-09-25 16:31:58 +08:00
|
|
|
}
|
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
pr_info("Submitted %lu crescendo:%x requests across %d engines and %d contexts\n",
|
2020-07-08 08:39:47 +08:00
|
|
|
count, flags, smoke->gt->info.num_engines, smoke->ncontext);
|
2018-09-25 16:31:58 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
static int smoke_random(struct preempt_smoke *smoke, unsigned int flags)
|
2018-09-25 16:31:58 +08:00
|
|
|
{
|
|
|
|
enum intel_engine_id id;
|
|
|
|
IGT_TIMEOUT(end_time);
|
|
|
|
unsigned long count;
|
|
|
|
|
|
|
|
count = 0;
|
|
|
|
do {
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(smoke->engine, smoke->gt, id) {
|
2018-09-25 16:31:58 +08:00
|
|
|
struct i915_gem_context *ctx = smoke_context(smoke);
|
2018-10-01 20:32:02 +08:00
|
|
|
int err;
|
2018-09-25 16:31:58 +08:00
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
err = smoke_submit(smoke,
|
|
|
|
ctx, random_priority(&smoke->prng),
|
|
|
|
flags & BATCH ? smoke->batch : NULL);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2018-09-25 16:31:58 +08:00
|
|
|
|
|
|
|
count++;
|
|
|
|
}
|
2020-06-08 06:20:45 +08:00
|
|
|
} while (count < smoke->ncontext && !__igt_timeout(end_time, NULL));
|
2018-09-25 16:31:58 +08:00
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
pr_info("Submitted %lu random:%x requests across %d engines and %d contexts\n",
|
2020-07-08 08:39:47 +08:00
|
|
|
count, flags, smoke->gt->info.num_engines, smoke->ncontext);
|
2018-09-25 16:31:58 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_preempt_smoke(void *arg)
|
|
|
|
{
|
|
|
|
struct preempt_smoke smoke = {
|
2019-10-16 20:02:49 +08:00
|
|
|
.gt = arg,
|
2018-09-25 16:31:58 +08:00
|
|
|
.prng = I915_RND_STATE_INITIALIZER(i915_selftest.random_seed),
|
2020-06-08 06:20:45 +08:00
|
|
|
.ncontext = 256,
|
2018-09-25 16:31:58 +08:00
|
|
|
};
|
2018-10-01 20:32:02 +08:00
|
|
|
const unsigned int phase[] = { 0, BATCH };
|
2019-03-22 03:40:31 +08:00
|
|
|
struct igt_live_test t;
|
2018-09-25 16:31:58 +08:00
|
|
|
int err = -ENOMEM;
|
2018-10-01 20:32:02 +08:00
|
|
|
u32 *cs;
|
2018-09-25 16:31:58 +08:00
|
|
|
int n;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (!HAS_LOGICAL_RING_PREEMPTION(smoke.gt->i915))
|
2018-09-25 16:31:58 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
smoke.contexts = kmalloc_array(smoke.ncontext,
|
|
|
|
sizeof(*smoke.contexts),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!smoke.contexts)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
smoke.batch =
|
|
|
|
i915_gem_object_create_internal(smoke.gt->i915, PAGE_SIZE);
|
2018-10-01 20:32:02 +08:00
|
|
|
if (IS_ERR(smoke.batch)) {
|
|
|
|
err = PTR_ERR(smoke.batch);
|
2019-10-04 21:40:14 +08:00
|
|
|
goto err_free;
|
2018-10-01 20:32:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
cs = i915_gem_object_pin_map(smoke.batch, I915_MAP_WB);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto err_batch;
|
|
|
|
}
|
|
|
|
for (n = 0; n < PAGE_SIZE / sizeof(*cs) - 1; n++)
|
|
|
|
cs[n] = MI_ARB_CHECK;
|
|
|
|
cs[n] = MI_BATCH_BUFFER_END;
|
drm/i915: Flush pages on acquisition
When we return pages to the system, we ensure that they are marked as
being in the CPU domain since any external access is uncontrolled and we
must assume the worst. This means that we need to always flush the pages
on acquisition if we need to use them on the GPU, and from the beginning
have used set-domain. Set-domain is overkill for the purpose as it is a
general synchronisation barrier, but our intent is to only flush the
pages being swapped in. If we move that flush into the pages acquisition
phase, we know then that when we have obj->mm.pages, they are coherent
with the GPU and need only maintain that status without resorting to
heavy handed use of set-domain.
The principle knock-on effect for userspace is through mmap-gtt
pagefaulting. Our uAPI has always implied that the GTT mmap was async
(especially as when any pagefault occurs is unpredicatable to userspace)
and so userspace had to apply explicit domain control itself
(set-domain). However, swapping is transparent to the kernel, and so on
first fault we need to acquire the pages and make them coherent for
access through the GTT. Our use of set-domain here leaks into the uABI
that the first pagefault was synchronous. This is unintentional and
baring a few igt should be unoticed, nevertheless we bump the uABI
version for mmap-gtt to reflect the change in behaviour.
Another implication of the change is that gem_create() is presumed to
create an object that is coherent with the CPU and is in the CPU write
domain, so a set-domain(CPU) following a gem_create() would be a minor
operation that merely checked whether we could allocate all pages for
the object. On applying this change, a set-domain(CPU) causes a clflush
as we acquire the pages. This will have a small impact on mesa as we move
the clflush here on !llc from execbuf time to create, but that should
have minimal performance impact as the same clflush exists but is now
done early and because of the clflush issue, userspace recycles bo and
so should resist allocating fresh objects.
Internally, the presumption that objects are created in the CPU
write-domain and remain so through writes to obj->mm.mapping is more
prevalent than I expected; but easy enough to catch and apply a manual
flush.
For the future, we should push the page flush from the central
set_pages() into the callers so that we can more finely control when it
is applied, but for now doing it one location is easier to validate, at
the cost of sometimes flushing when there is no need.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Matthew Auld <matthew.william.auld@gmail.com>
Cc: Daniele Ceraolo Spurio <daniele.ceraolospurio@intel.com>
Cc: Antonio Argenziano <antonio.argenziano@intel.com>
Cc: Joonas Lahtinen <joonas.lahtinen@linux.intel.com>
Reviewed-by: Matthew Auld <matthew.william.auld@gmail.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190321161908.8007-1-chris@chris-wilson.co.uk
2019-03-22 00:19:07 +08:00
|
|
|
i915_gem_object_flush_map(smoke.batch);
|
2018-10-01 20:32:02 +08:00
|
|
|
i915_gem_object_unpin_map(smoke.batch);
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_live_test_begin(&t, smoke.gt->i915, __func__, "all")) {
|
2019-03-22 03:40:31 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto err_batch;
|
|
|
|
}
|
|
|
|
|
2018-09-25 16:31:58 +08:00
|
|
|
for (n = 0; n < smoke.ncontext; n++) {
|
2019-10-16 20:02:49 +08:00
|
|
|
smoke.contexts[n] = kernel_context(smoke.gt->i915);
|
2018-09-25 16:31:58 +08:00
|
|
|
if (!smoke.contexts[n])
|
|
|
|
goto err_ctx;
|
|
|
|
}
|
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
for (n = 0; n < ARRAY_SIZE(phase); n++) {
|
|
|
|
err = smoke_crescendo(&smoke, phase[n]);
|
|
|
|
if (err)
|
|
|
|
goto err_ctx;
|
2018-09-25 16:31:58 +08:00
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
err = smoke_random(&smoke, phase[n]);
|
|
|
|
if (err)
|
|
|
|
goto err_ctx;
|
|
|
|
}
|
2018-09-25 16:31:58 +08:00
|
|
|
|
|
|
|
err_ctx:
|
2019-03-22 03:40:31 +08:00
|
|
|
if (igt_live_test_end(&t))
|
2018-09-25 16:31:58 +08:00
|
|
|
err = -EIO;
|
|
|
|
|
|
|
|
for (n = 0; n < smoke.ncontext; n++) {
|
|
|
|
if (!smoke.contexts[n])
|
|
|
|
break;
|
|
|
|
kernel_context_close(smoke.contexts[n]);
|
|
|
|
}
|
|
|
|
|
2018-10-01 20:32:02 +08:00
|
|
|
err_batch:
|
|
|
|
i915_gem_object_put(smoke.batch);
|
2019-10-04 21:40:14 +08:00
|
|
|
err_free:
|
2018-09-25 16:31:58 +08:00
|
|
|
kfree(smoke.contexts);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
static int nop_virtual_engine(struct intel_gt *gt,
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
struct intel_engine_cs **siblings,
|
|
|
|
unsigned int nsibling,
|
|
|
|
unsigned int nctx,
|
|
|
|
unsigned int flags)
|
|
|
|
#define CHAIN BIT(0)
|
|
|
|
{
|
|
|
|
IGT_TIMEOUT(end_time);
|
2019-11-21 15:10:43 +08:00
|
|
|
struct i915_request *request[16] = {};
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
struct intel_context *ve[16];
|
|
|
|
unsigned long n, prime, nc;
|
|
|
|
struct igt_live_test t;
|
|
|
|
ktime_t times[2] = {};
|
|
|
|
int err;
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
GEM_BUG_ON(!nctx || nctx > ARRAY_SIZE(ve));
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
|
|
|
|
for (n = 0; n < nctx; n++) {
|
2019-12-22 00:03:24 +08:00
|
|
|
ve[n] = intel_execlists_create_virtual(siblings, nsibling);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
if (IS_ERR(ve[n])) {
|
|
|
|
err = PTR_ERR(ve[n]);
|
|
|
|
nctx = n;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = intel_context_pin(ve[n]);
|
|
|
|
if (err) {
|
|
|
|
intel_context_put(ve[n]);
|
|
|
|
nctx = n;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
err = igt_live_test_begin(&t, gt->i915, __func__, ve[0]->engine->name);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
for_each_prime_number_from(prime, 1, 8192) {
|
|
|
|
times[1] = ktime_get_raw();
|
|
|
|
|
|
|
|
if (flags & CHAIN) {
|
|
|
|
for (nc = 0; nc < nctx; nc++) {
|
|
|
|
for (n = 0; n < prime; n++) {
|
2019-11-21 15:10:43 +08:00
|
|
|
struct i915_request *rq;
|
|
|
|
|
|
|
|
rq = i915_request_create(ve[nc]);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-11-21 15:10:43 +08:00
|
|
|
if (request[nc])
|
|
|
|
i915_request_put(request[nc]);
|
|
|
|
request[nc] = i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (n = 0; n < prime; n++) {
|
|
|
|
for (nc = 0; nc < nctx; nc++) {
|
2019-11-21 15:10:43 +08:00
|
|
|
struct i915_request *rq;
|
|
|
|
|
|
|
|
rq = i915_request_create(ve[nc]);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-11-21 15:10:43 +08:00
|
|
|
if (request[nc])
|
|
|
|
i915_request_put(request[nc]);
|
|
|
|
request[nc] = i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (nc = 0; nc < nctx; nc++) {
|
2019-06-18 15:41:30 +08:00
|
|
|
if (i915_request_wait(request[nc], 0, HZ / 10) < 0) {
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
pr_err("%s(%s): wait for %llx:%lld timed out\n",
|
|
|
|
__func__, ve[0]->engine->name,
|
|
|
|
request[nc]->fence.context,
|
|
|
|
request[nc]->fence.seqno);
|
|
|
|
|
|
|
|
GEM_TRACE("%s(%s) failed at request %llx:%lld\n",
|
|
|
|
__func__, ve[0]->engine->name,
|
|
|
|
request[nc]->fence.context,
|
|
|
|
request[nc]->fence.seqno);
|
|
|
|
GEM_TRACE_DUMP();
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
times[1] = ktime_sub(ktime_get_raw(), times[1]);
|
|
|
|
if (prime == 1)
|
|
|
|
times[0] = times[1];
|
|
|
|
|
2019-11-21 15:10:43 +08:00
|
|
|
for (nc = 0; nc < nctx; nc++) {
|
|
|
|
i915_request_put(request[nc]);
|
|
|
|
request[nc] = NULL;
|
|
|
|
}
|
|
|
|
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
if (__igt_timeout(end_time, NULL))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = igt_live_test_end(&t);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
pr_info("Requestx%d latencies on %s: 1 = %lluns, %lu = %lluns\n",
|
|
|
|
nctx, ve[0]->engine->name, ktime_to_ns(times[0]),
|
|
|
|
prime, div64_u64(ktime_to_ns(times[1]), prime));
|
|
|
|
|
|
|
|
out:
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_flush_test(gt->i915))
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
err = -EIO;
|
|
|
|
|
|
|
|
for (nc = 0; nc < nctx; nc++) {
|
2019-11-21 15:10:43 +08:00
|
|
|
i915_request_put(request[nc]);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
intel_context_unpin(ve[nc]);
|
|
|
|
intel_context_put(ve[nc]);
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-05-19 21:20:44 +08:00
|
|
|
static unsigned int
|
|
|
|
__select_siblings(struct intel_gt *gt,
|
|
|
|
unsigned int class,
|
|
|
|
struct intel_engine_cs **siblings,
|
|
|
|
bool (*filter)(const struct intel_engine_cs *))
|
2020-05-18 18:29:11 +08:00
|
|
|
{
|
|
|
|
unsigned int n = 0;
|
|
|
|
unsigned int inst;
|
|
|
|
|
|
|
|
for (inst = 0; inst <= MAX_ENGINE_INSTANCE; inst++) {
|
|
|
|
if (!gt->engine_class[class][inst])
|
|
|
|
continue;
|
|
|
|
|
2020-05-19 21:20:44 +08:00
|
|
|
if (filter && !filter(gt->engine_class[class][inst]))
|
|
|
|
continue;
|
|
|
|
|
2020-05-18 18:29:11 +08:00
|
|
|
siblings[n++] = gt->engine_class[class][inst];
|
|
|
|
}
|
|
|
|
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2020-05-19 21:20:44 +08:00
|
|
|
static unsigned int
|
|
|
|
select_siblings(struct intel_gt *gt,
|
|
|
|
unsigned int class,
|
|
|
|
struct intel_engine_cs **siblings)
|
|
|
|
{
|
|
|
|
return __select_siblings(gt, class, siblings, NULL);
|
|
|
|
}
|
|
|
|
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
static int live_virtual_engine(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
struct intel_engine_cs *siblings[MAX_ENGINE_INSTANCE + 1];
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
2020-05-18 18:29:11 +08:00
|
|
|
unsigned int class;
|
2019-10-04 21:40:14 +08:00
|
|
|
int err;
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
|
2020-02-19 06:33:20 +08:00
|
|
|
if (intel_uc_uses_guc_submission(>->uc))
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
return 0;
|
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2019-10-16 20:02:49 +08:00
|
|
|
err = nop_virtual_engine(gt, &engine, 1, 1, 0);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
if (err) {
|
|
|
|
pr_err("Failed to wrap engine %s: err=%d\n",
|
|
|
|
engine->name, err);
|
2019-10-04 21:40:14 +08:00
|
|
|
return err;
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (class = 0; class <= MAX_ENGINE_CLASS; class++) {
|
|
|
|
int nsibling, n;
|
|
|
|
|
2020-05-18 18:29:11 +08:00
|
|
|
nsibling = select_siblings(gt, class, siblings);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
if (nsibling < 2)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (n = 1; n <= nsibling + 1; n++) {
|
2019-10-16 20:02:49 +08:00
|
|
|
err = nop_virtual_engine(gt, siblings, nsibling,
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
n, 0);
|
|
|
|
if (err)
|
2019-10-04 21:40:14 +08:00
|
|
|
return err;
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
}
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
err = nop_virtual_engine(gt, siblings, nsibling, n, CHAIN);
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
if (err)
|
2019-10-04 21:40:14 +08:00
|
|
|
return err;
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
}
|
|
|
|
|
2019-10-04 21:40:14 +08:00
|
|
|
return 0;
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
}
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
static int mask_virtual_engine(struct intel_gt *gt,
|
2019-05-22 05:11:31 +08:00
|
|
|
struct intel_engine_cs **siblings,
|
|
|
|
unsigned int nsibling)
|
|
|
|
{
|
|
|
|
struct i915_request *request[MAX_ENGINE_INSTANCE + 1];
|
|
|
|
struct intel_context *ve;
|
|
|
|
struct igt_live_test t;
|
|
|
|
unsigned int n;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that by setting the execution mask on a request, we can
|
|
|
|
* restrict it to our desired engine within the virtual engine.
|
|
|
|
*/
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
ve = intel_execlists_create_virtual(siblings, nsibling);
|
2019-05-22 05:11:31 +08:00
|
|
|
if (IS_ERR(ve)) {
|
|
|
|
err = PTR_ERR(ve);
|
|
|
|
goto out_close;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = intel_context_pin(ve);
|
|
|
|
if (err)
|
|
|
|
goto out_put;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
err = igt_live_test_begin(&t, gt->i915, __func__, ve->engine->name);
|
2019-05-22 05:11:31 +08:00
|
|
|
if (err)
|
|
|
|
goto out_unpin;
|
|
|
|
|
|
|
|
for (n = 0; n < nsibling; n++) {
|
|
|
|
request[n] = i915_request_create(ve);
|
2019-05-29 19:03:55 +08:00
|
|
|
if (IS_ERR(request[n])) {
|
|
|
|
err = PTR_ERR(request[n]);
|
2019-05-22 05:11:31 +08:00
|
|
|
nsibling = n;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reverse order as it's more likely to be unnatural */
|
|
|
|
request[n]->execution_mask = siblings[nsibling - n - 1]->mask;
|
|
|
|
|
|
|
|
i915_request_get(request[n]);
|
|
|
|
i915_request_add(request[n]);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (n = 0; n < nsibling; n++) {
|
2019-06-18 15:41:30 +08:00
|
|
|
if (i915_request_wait(request[n], 0, HZ / 10) < 0) {
|
2019-05-22 05:11:31 +08:00
|
|
|
pr_err("%s(%s): wait for %llx:%lld timed out\n",
|
|
|
|
__func__, ve->engine->name,
|
|
|
|
request[n]->fence.context,
|
|
|
|
request[n]->fence.seqno);
|
|
|
|
|
|
|
|
GEM_TRACE("%s(%s) failed at request %llx:%lld\n",
|
|
|
|
__func__, ve->engine->name,
|
|
|
|
request[n]->fence.context,
|
|
|
|
request[n]->fence.seqno);
|
|
|
|
GEM_TRACE_DUMP();
|
2019-10-16 20:02:49 +08:00
|
|
|
intel_gt_set_wedged(gt);
|
2019-05-22 05:11:31 +08:00
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (request[n]->engine != siblings[nsibling - n - 1]) {
|
|
|
|
pr_err("Executed on wrong sibling '%s', expected '%s'\n",
|
|
|
|
request[n]->engine->name,
|
|
|
|
siblings[nsibling - n - 1]->name);
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = igt_live_test_end(&t);
|
|
|
|
out:
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_flush_test(gt->i915))
|
2019-05-22 05:11:31 +08:00
|
|
|
err = -EIO;
|
|
|
|
|
|
|
|
for (n = 0; n < nsibling; n++)
|
|
|
|
i915_request_put(request[n]);
|
|
|
|
|
|
|
|
out_unpin:
|
|
|
|
intel_context_unpin(ve);
|
|
|
|
out_put:
|
|
|
|
intel_context_put(ve);
|
|
|
|
out_close:
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_virtual_mask(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2019-05-22 05:11:31 +08:00
|
|
|
struct intel_engine_cs *siblings[MAX_ENGINE_INSTANCE + 1];
|
2020-05-18 18:29:11 +08:00
|
|
|
unsigned int class;
|
2019-10-04 21:40:14 +08:00
|
|
|
int err;
|
2019-05-22 05:11:31 +08:00
|
|
|
|
2020-02-19 06:33:20 +08:00
|
|
|
if (intel_uc_uses_guc_submission(>->uc))
|
2019-05-22 05:11:31 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (class = 0; class <= MAX_ENGINE_CLASS; class++) {
|
|
|
|
unsigned int nsibling;
|
|
|
|
|
2020-05-18 18:29:11 +08:00
|
|
|
nsibling = select_siblings(gt, class, siblings);
|
2019-05-22 05:11:31 +08:00
|
|
|
if (nsibling < 2)
|
|
|
|
continue;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
err = mask_virtual_engine(gt, siblings, nsibling);
|
2019-05-22 05:11:31 +08:00
|
|
|
if (err)
|
2019-10-04 21:40:14 +08:00
|
|
|
return err;
|
2019-05-22 05:11:31 +08:00
|
|
|
}
|
|
|
|
|
2019-10-04 21:40:14 +08:00
|
|
|
return 0;
|
2019-05-22 05:11:31 +08:00
|
|
|
}
|
|
|
|
|
2020-05-19 21:20:44 +08:00
|
|
|
static int slicein_virtual_engine(struct intel_gt *gt,
|
|
|
|
struct intel_engine_cs **siblings,
|
|
|
|
unsigned int nsibling)
|
|
|
|
{
|
|
|
|
const long timeout = slice_timeout(siblings[0]);
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct igt_spinner spin;
|
|
|
|
unsigned int n;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Virtual requests must take part in timeslicing on the target engines.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (igt_spinner_init(&spin, gt))
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
for (n = 0; n < nsibling; n++) {
|
|
|
|
ce = intel_context_create(siblings[n]);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = igt_spinner_create_request(&spin, ce, MI_ARB_CHECK);
|
|
|
|
intel_context_put(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
|
|
|
}
|
|
|
|
|
|
|
|
ce = intel_execlists_create_virtual(siblings, nsibling);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
intel_context_put(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (i915_request_wait(rq, 0, timeout) < 0) {
|
|
|
|
GEM_TRACE_ERR("%s(%s) failed to slice in virtual request\n",
|
|
|
|
__func__, rq->engine->name);
|
|
|
|
GEM_TRACE_DUMP();
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -EIO;
|
|
|
|
}
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
out:
|
|
|
|
igt_spinner_end(&spin);
|
|
|
|
if (igt_flush_test(gt->i915))
|
|
|
|
err = -EIO;
|
|
|
|
igt_spinner_fini(&spin);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sliceout_virtual_engine(struct intel_gt *gt,
|
|
|
|
struct intel_engine_cs **siblings,
|
|
|
|
unsigned int nsibling)
|
|
|
|
{
|
|
|
|
const long timeout = slice_timeout(siblings[0]);
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct igt_spinner spin;
|
|
|
|
unsigned int n;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Virtual requests must allow others a fair timeslice.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (igt_spinner_init(&spin, gt))
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
/* XXX We do not handle oversubscription and fairness with normal rq */
|
|
|
|
for (n = 0; n < nsibling; n++) {
|
|
|
|
ce = intel_execlists_create_virtual(siblings, nsibling);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = igt_spinner_create_request(&spin, ce, MI_ARB_CHECK);
|
|
|
|
intel_context_put(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (n = 0; !err && n < nsibling; n++) {
|
|
|
|
ce = intel_context_create(siblings[n]);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
intel_context_put(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (i915_request_wait(rq, 0, timeout) < 0) {
|
|
|
|
GEM_TRACE_ERR("%s(%s) failed to slice out virtual request\n",
|
|
|
|
__func__, siblings[n]->name);
|
|
|
|
GEM_TRACE_DUMP();
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -EIO;
|
|
|
|
}
|
|
|
|
i915_request_put(rq);
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
igt_spinner_end(&spin);
|
|
|
|
if (igt_flush_test(gt->i915))
|
|
|
|
err = -EIO;
|
|
|
|
igt_spinner_fini(&spin);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_virtual_slice(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *siblings[MAX_ENGINE_INSTANCE + 1];
|
|
|
|
unsigned int class;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
if (intel_uc_uses_guc_submission(>->uc))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (class = 0; class <= MAX_ENGINE_CLASS; class++) {
|
|
|
|
unsigned int nsibling;
|
|
|
|
|
|
|
|
nsibling = __select_siblings(gt, class, siblings,
|
|
|
|
intel_engine_has_timeslices);
|
|
|
|
if (nsibling < 2)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
err = slicein_virtual_engine(gt, siblings, nsibling);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = sliceout_virtual_engine(gt, siblings, nsibling);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
static int preserved_virtual_engine(struct intel_gt *gt,
|
2019-10-10 19:02:52 +08:00
|
|
|
struct intel_engine_cs **siblings,
|
|
|
|
unsigned int nsibling)
|
|
|
|
{
|
|
|
|
struct i915_request *last = NULL;
|
|
|
|
struct intel_context *ve;
|
|
|
|
struct i915_vma *scratch;
|
|
|
|
struct igt_live_test t;
|
|
|
|
unsigned int n;
|
|
|
|
int err = 0;
|
2019-10-14 17:07:49 +08:00
|
|
|
u32 *cs;
|
2019-10-10 19:02:52 +08:00
|
|
|
|
|
|
|
scratch = create_scratch(siblings[0]->gt);
|
2019-12-22 00:03:24 +08:00
|
|
|
if (IS_ERR(scratch))
|
|
|
|
return PTR_ERR(scratch);
|
2019-10-10 19:02:52 +08:00
|
|
|
|
2020-01-31 22:26:10 +08:00
|
|
|
err = i915_vma_sync(scratch);
|
|
|
|
if (err)
|
|
|
|
goto out_scratch;
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
ve = intel_execlists_create_virtual(siblings, nsibling);
|
2019-10-10 19:02:52 +08:00
|
|
|
if (IS_ERR(ve)) {
|
|
|
|
err = PTR_ERR(ve);
|
|
|
|
goto out_scratch;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = intel_context_pin(ve);
|
|
|
|
if (err)
|
|
|
|
goto out_put;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
err = igt_live_test_begin(&t, gt->i915, __func__, ve->engine->name);
|
2019-10-10 19:02:52 +08:00
|
|
|
if (err)
|
|
|
|
goto out_unpin;
|
|
|
|
|
2019-10-14 17:07:49 +08:00
|
|
|
for (n = 0; n < NUM_GPR_DW; n++) {
|
2019-10-10 19:02:52 +08:00
|
|
|
struct intel_engine_cs *engine = siblings[n % nsibling];
|
|
|
|
struct i915_request *rq;
|
|
|
|
|
|
|
|
rq = i915_request_create(ve);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_put(last);
|
|
|
|
last = i915_request_get(rq);
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 8);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto out_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_REGISTER_MEM_GEN8 | MI_USE_GGTT;
|
|
|
|
*cs++ = CS_GPR(engine, n);
|
|
|
|
*cs++ = i915_ggtt_offset(scratch) + n * sizeof(u32);
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
*cs++ = MI_LOAD_REGISTER_IMM(1);
|
2019-10-14 17:07:49 +08:00
|
|
|
*cs++ = CS_GPR(engine, (n + 1) % NUM_GPR_DW);
|
2019-10-10 19:02:52 +08:00
|
|
|
*cs++ = n + 1;
|
|
|
|
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
intel_ring_advance(rq, cs);
|
|
|
|
|
|
|
|
/* Restrict this request to run on a particular engine */
|
|
|
|
rq->execution_mask = engine->mask;
|
|
|
|
i915_request_add(rq);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i915_request_wait(last, 0, HZ / 5) < 0) {
|
|
|
|
err = -ETIME;
|
2019-10-14 17:07:49 +08:00
|
|
|
goto out_end;
|
|
|
|
}
|
2019-10-10 19:02:52 +08:00
|
|
|
|
2019-10-14 17:07:49 +08:00
|
|
|
cs = i915_gem_object_pin_map(scratch->obj, I915_MAP_WB);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto out_end;
|
|
|
|
}
|
2019-10-10 19:02:52 +08:00
|
|
|
|
2019-10-14 17:07:49 +08:00
|
|
|
for (n = 0; n < NUM_GPR_DW; n++) {
|
|
|
|
if (cs[n] != n) {
|
|
|
|
pr_err("Incorrect value[%d] found for GPR[%d]\n",
|
|
|
|
cs[n], n);
|
|
|
|
err = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
2019-10-10 19:02:52 +08:00
|
|
|
}
|
|
|
|
|
2019-10-14 17:07:49 +08:00
|
|
|
i915_gem_object_unpin_map(scratch->obj);
|
|
|
|
|
2019-10-10 19:02:52 +08:00
|
|
|
out_end:
|
|
|
|
if (igt_live_test_end(&t))
|
|
|
|
err = -EIO;
|
|
|
|
i915_request_put(last);
|
|
|
|
out_unpin:
|
|
|
|
intel_context_unpin(ve);
|
|
|
|
out_put:
|
|
|
|
intel_context_put(ve);
|
|
|
|
out_scratch:
|
|
|
|
i915_vma_unpin_and_release(&scratch, 0);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_virtual_preserved(void *arg)
|
|
|
|
{
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2019-10-10 19:02:52 +08:00
|
|
|
struct intel_engine_cs *siblings[MAX_ENGINE_INSTANCE + 1];
|
2020-05-18 18:29:11 +08:00
|
|
|
unsigned int class;
|
2019-10-10 19:02:52 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that the context image retains non-privileged (user) registers
|
|
|
|
* from one engine to the next. For this we check that the CS_GPR
|
|
|
|
* are preserved.
|
|
|
|
*/
|
|
|
|
|
2020-02-19 06:33:20 +08:00
|
|
|
if (intel_uc_uses_guc_submission(>->uc))
|
2019-10-10 19:02:52 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* As we use CS_GPR we cannot run before they existed on all engines. */
|
2019-10-16 20:02:49 +08:00
|
|
|
if (INTEL_GEN(gt->i915) < 9)
|
2019-10-10 19:02:52 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (class = 0; class <= MAX_ENGINE_CLASS; class++) {
|
|
|
|
int nsibling, err;
|
|
|
|
|
2020-05-18 18:29:11 +08:00
|
|
|
nsibling = select_siblings(gt, class, siblings);
|
2019-10-10 19:02:52 +08:00
|
|
|
if (nsibling < 2)
|
|
|
|
continue;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
err = preserved_virtual_engine(gt, siblings, nsibling);
|
2019-10-10 19:02:52 +08:00
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
static int bond_virtual_engine(struct intel_gt *gt,
|
2019-05-22 05:11:33 +08:00
|
|
|
unsigned int class,
|
|
|
|
struct intel_engine_cs **siblings,
|
|
|
|
unsigned int nsibling,
|
|
|
|
unsigned int flags)
|
|
|
|
#define BOND_SCHEDULE BIT(0)
|
|
|
|
{
|
|
|
|
struct intel_engine_cs *master;
|
|
|
|
struct i915_request *rq[16];
|
|
|
|
enum intel_engine_id id;
|
2019-11-22 19:21:48 +08:00
|
|
|
struct igt_spinner spin;
|
2019-05-22 05:11:33 +08:00
|
|
|
unsigned long n;
|
|
|
|
int err;
|
|
|
|
|
2019-11-22 19:21:48 +08:00
|
|
|
/*
|
|
|
|
* A set of bonded requests is intended to be run concurrently
|
|
|
|
* across a number of engines. We use one request per-engine
|
|
|
|
* and a magic fence to schedule each of the bonded requests
|
|
|
|
* at the same time. A consequence of our current scheduler is that
|
|
|
|
* we only move requests to the HW ready queue when the request
|
|
|
|
* becomes ready, that is when all of its prerequisite fences have
|
|
|
|
* been signaled. As one of those fences is the master submit fence,
|
|
|
|
* there is a delay on all secondary fences as the HW may be
|
|
|
|
* currently busy. Equally, as all the requests are independent,
|
|
|
|
* they may have other fences that delay individual request
|
|
|
|
* submission to HW. Ergo, we do not guarantee that all requests are
|
|
|
|
* immediately submitted to HW at the same time, just that if the
|
|
|
|
* rules are abided by, they are ready at the same time as the
|
|
|
|
* first is submitted. Userspace can embed semaphores in its batch
|
|
|
|
* to ensure parallel execution of its phases as it requires.
|
|
|
|
* Though naturally it gets requested that perhaps the scheduler should
|
|
|
|
* take care of parallel execution, even across preemption events on
|
|
|
|
* different HW. (The proper answer is of course "lalalala".)
|
|
|
|
*
|
|
|
|
* With the submit-fence, we have identified three possible phases
|
|
|
|
* of synchronisation depending on the master fence: queued (not
|
|
|
|
* ready), executing, and signaled. The first two are quite simple
|
|
|
|
* and checked below. However, the signaled master fence handling is
|
|
|
|
* contentious. Currently we do not distinguish between a signaled
|
|
|
|
* fence and an expired fence, as once signaled it does not convey
|
|
|
|
* any information about the previous execution. It may even be freed
|
|
|
|
* and hence checking later it may not exist at all. Ergo we currently
|
|
|
|
* do not apply the bonding constraint for an already signaled fence,
|
|
|
|
* as our expectation is that it should not constrain the secondaries
|
|
|
|
* and is outside of the scope of the bonded request API (i.e. all
|
|
|
|
* userspace requests are meant to be running in parallel). As
|
|
|
|
* it imposes no constraint, and is effectively a no-op, we do not
|
|
|
|
* check below as normal execution flows are checked extensively above.
|
|
|
|
*
|
|
|
|
* XXX Is the degenerate handling of signaled submit fences the
|
|
|
|
* expected behaviour for userpace?
|
|
|
|
*/
|
|
|
|
|
2019-05-22 05:11:33 +08:00
|
|
|
GEM_BUG_ON(nsibling >= ARRAY_SIZE(rq) - 1);
|
|
|
|
|
2019-11-22 19:21:48 +08:00
|
|
|
if (igt_spinner_init(&spin, gt))
|
2019-05-22 05:11:33 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
rq[0] = ERR_PTR(-ENOMEM);
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(master, gt, id) {
|
2019-05-22 05:11:33 +08:00
|
|
|
struct i915_sw_fence fence = {};
|
drm/i915: Disable use of hwsp_cacheline for kernel_context
Currently on execlists, we use a local hwsp for the kernel_context,
rather than the engine's HWSP, as this is the default for execlists.
However, seqno wrap requires allocating a new HWSP cacheline, and may
require pinning a new HWSP page in the GGTT. This operation requiring
pinning in the GGTT is not allowed within the kernel_context timeline,
as doing so may require re-entering the kernel_context in order to evict
from the GGTT. As we want to avoid requiring a new HWSP for the
kernel_context, we can use the permanently pinned engine's HWSP instead.
However to do so we must prevent the use of semaphores reading the
kernel_context's HWSP, as the use of semaphores do not support rollover
onto the same cacheline. Fortunately, the kernel_context is mostly
isolated, so unlikely to give benefit to semaphores.
Reported-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200210205722.794180-5-chris@chris-wilson.co.uk
2020-02-11 04:57:20 +08:00
|
|
|
struct intel_context *ce;
|
2019-05-22 05:11:33 +08:00
|
|
|
|
|
|
|
if (master->class == class)
|
|
|
|
continue;
|
|
|
|
|
drm/i915: Disable use of hwsp_cacheline for kernel_context
Currently on execlists, we use a local hwsp for the kernel_context,
rather than the engine's HWSP, as this is the default for execlists.
However, seqno wrap requires allocating a new HWSP cacheline, and may
require pinning a new HWSP page in the GGTT. This operation requiring
pinning in the GGTT is not allowed within the kernel_context timeline,
as doing so may require re-entering the kernel_context in order to evict
from the GGTT. As we want to avoid requiring a new HWSP for the
kernel_context, we can use the permanently pinned engine's HWSP instead.
However to do so we must prevent the use of semaphores reading the
kernel_context's HWSP, as the use of semaphores do not support rollover
onto the same cacheline. Fortunately, the kernel_context is mostly
isolated, so unlikely to give benefit to semaphores.
Reported-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200210205722.794180-5-chris@chris-wilson.co.uk
2020-02-11 04:57:20 +08:00
|
|
|
ce = intel_context_create(master);
|
|
|
|
if (IS_ERR(ce)) {
|
|
|
|
err = PTR_ERR(ce);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-05-22 05:11:33 +08:00
|
|
|
memset_p((void *)rq, ERR_PTR(-EINVAL), ARRAY_SIZE(rq));
|
|
|
|
|
drm/i915: Disable use of hwsp_cacheline for kernel_context
Currently on execlists, we use a local hwsp for the kernel_context,
rather than the engine's HWSP, as this is the default for execlists.
However, seqno wrap requires allocating a new HWSP cacheline, and may
require pinning a new HWSP page in the GGTT. This operation requiring
pinning in the GGTT is not allowed within the kernel_context timeline,
as doing so may require re-entering the kernel_context in order to evict
from the GGTT. As we want to avoid requiring a new HWSP for the
kernel_context, we can use the permanently pinned engine's HWSP instead.
However to do so we must prevent the use of semaphores reading the
kernel_context's HWSP, as the use of semaphores do not support rollover
onto the same cacheline. Fortunately, the kernel_context is mostly
isolated, so unlikely to give benefit to semaphores.
Reported-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200210205722.794180-5-chris@chris-wilson.co.uk
2020-02-11 04:57:20 +08:00
|
|
|
rq[0] = igt_spinner_create_request(&spin, ce, MI_NOOP);
|
|
|
|
intel_context_put(ce);
|
2019-05-22 05:11:33 +08:00
|
|
|
if (IS_ERR(rq[0])) {
|
|
|
|
err = PTR_ERR(rq[0]);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
i915_request_get(rq[0]);
|
|
|
|
|
|
|
|
if (flags & BOND_SCHEDULE) {
|
|
|
|
onstack_fence_init(&fence);
|
|
|
|
err = i915_sw_fence_await_sw_fence_gfp(&rq[0]->submit,
|
|
|
|
&fence,
|
|
|
|
GFP_KERNEL);
|
|
|
|
}
|
2019-11-22 19:21:48 +08:00
|
|
|
|
2019-05-22 05:11:33 +08:00
|
|
|
i915_request_add(rq[0]);
|
|
|
|
if (err < 0)
|
|
|
|
goto out;
|
|
|
|
|
2019-11-22 19:21:48 +08:00
|
|
|
if (!(flags & BOND_SCHEDULE) &&
|
|
|
|
!igt_wait_for_spinner(&spin, rq[0])) {
|
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-05-22 05:11:33 +08:00
|
|
|
for (n = 0; n < nsibling; n++) {
|
|
|
|
struct intel_context *ve;
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
ve = intel_execlists_create_virtual(siblings, nsibling);
|
2019-05-22 05:11:33 +08:00
|
|
|
if (IS_ERR(ve)) {
|
|
|
|
err = PTR_ERR(ve);
|
|
|
|
onstack_fence_fini(&fence);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = intel_virtual_engine_attach_bond(ve->engine,
|
|
|
|
master,
|
|
|
|
siblings[n]);
|
|
|
|
if (err) {
|
|
|
|
intel_context_put(ve);
|
|
|
|
onstack_fence_fini(&fence);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = intel_context_pin(ve);
|
|
|
|
intel_context_put(ve);
|
|
|
|
if (err) {
|
|
|
|
onstack_fence_fini(&fence);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq[n + 1] = i915_request_create(ve);
|
|
|
|
intel_context_unpin(ve);
|
|
|
|
if (IS_ERR(rq[n + 1])) {
|
|
|
|
err = PTR_ERR(rq[n + 1]);
|
|
|
|
onstack_fence_fini(&fence);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
i915_request_get(rq[n + 1]);
|
|
|
|
|
|
|
|
err = i915_request_await_execution(rq[n + 1],
|
|
|
|
&rq[0]->fence,
|
|
|
|
ve->engine->bond_execute);
|
|
|
|
i915_request_add(rq[n + 1]);
|
|
|
|
if (err < 0) {
|
|
|
|
onstack_fence_fini(&fence);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
onstack_fence_fini(&fence);
|
2019-11-22 19:21:48 +08:00
|
|
|
intel_engine_flush_submission(master);
|
|
|
|
igt_spinner_end(&spin);
|
2019-05-22 05:11:33 +08:00
|
|
|
|
2019-06-18 15:41:30 +08:00
|
|
|
if (i915_request_wait(rq[0], 0, HZ / 10) < 0) {
|
2019-05-22 05:11:33 +08:00
|
|
|
pr_err("Master request did not execute (on %s)!\n",
|
|
|
|
rq[0]->engine->name);
|
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (n = 0; n < nsibling; n++) {
|
2019-06-18 15:41:30 +08:00
|
|
|
if (i915_request_wait(rq[n + 1], 0,
|
2019-05-22 05:11:33 +08:00
|
|
|
MAX_SCHEDULE_TIMEOUT) < 0) {
|
|
|
|
err = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rq[n + 1]->engine != siblings[n]) {
|
|
|
|
pr_err("Bonded request did not execute on target engine: expected %s, used %s; master was %s\n",
|
|
|
|
siblings[n]->name,
|
|
|
|
rq[n + 1]->engine->name,
|
|
|
|
rq[0]->engine->name);
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (n = 0; !IS_ERR(rq[n]); n++)
|
|
|
|
i915_request_put(rq[n]);
|
|
|
|
rq[0] = ERR_PTR(-ENOMEM);
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
for (n = 0; !IS_ERR(rq[n]); n++)
|
|
|
|
i915_request_put(rq[n]);
|
2019-10-16 20:02:49 +08:00
|
|
|
if (igt_flush_test(gt->i915))
|
2019-05-22 05:11:33 +08:00
|
|
|
err = -EIO;
|
|
|
|
|
2019-11-22 19:21:48 +08:00
|
|
|
igt_spinner_fini(&spin);
|
2019-05-22 05:11:33 +08:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_virtual_bond(void *arg)
|
|
|
|
{
|
|
|
|
static const struct phase {
|
|
|
|
const char *name;
|
|
|
|
unsigned int flags;
|
|
|
|
} phases[] = {
|
|
|
|
{ "", 0 },
|
|
|
|
{ "schedule", BOND_SCHEDULE },
|
|
|
|
{ },
|
|
|
|
};
|
2019-10-16 20:02:49 +08:00
|
|
|
struct intel_gt *gt = arg;
|
2019-05-22 05:11:33 +08:00
|
|
|
struct intel_engine_cs *siblings[MAX_ENGINE_INSTANCE + 1];
|
2020-05-18 18:29:11 +08:00
|
|
|
unsigned int class;
|
2019-10-04 21:40:14 +08:00
|
|
|
int err;
|
2019-05-22 05:11:33 +08:00
|
|
|
|
2020-02-19 06:33:20 +08:00
|
|
|
if (intel_uc_uses_guc_submission(>->uc))
|
2019-05-22 05:11:33 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (class = 0; class <= MAX_ENGINE_CLASS; class++) {
|
|
|
|
const struct phase *p;
|
|
|
|
int nsibling;
|
|
|
|
|
2020-05-18 18:29:11 +08:00
|
|
|
nsibling = select_siblings(gt, class, siblings);
|
2019-05-22 05:11:33 +08:00
|
|
|
if (nsibling < 2)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (p = phases; p->name; p++) {
|
2019-10-16 20:02:49 +08:00
|
|
|
err = bond_virtual_engine(gt,
|
2019-05-22 05:11:33 +08:00
|
|
|
class, siblings, nsibling,
|
|
|
|
p->flags);
|
|
|
|
if (err) {
|
|
|
|
pr_err("%s(%s): failed class=%d, nsibling=%d, err=%d\n",
|
|
|
|
__func__, p->name, class, nsibling, err);
|
2019-10-04 21:40:14 +08:00
|
|
|
return err;
|
2019-05-22 05:11:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-04 21:40:14 +08:00
|
|
|
return 0;
|
2019-05-22 05:11:33 +08:00
|
|
|
}
|
|
|
|
|
2020-01-22 22:02:42 +08:00
|
|
|
static int reset_virtual_engine(struct intel_gt *gt,
|
|
|
|
struct intel_engine_cs **siblings,
|
|
|
|
unsigned int nsibling)
|
|
|
|
{
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct intel_context *ve;
|
|
|
|
struct igt_spinner spin;
|
|
|
|
struct i915_request *rq;
|
|
|
|
unsigned int n;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* In order to support offline error capture for fast preempt reset,
|
|
|
|
* we need to decouple the guilty request and ensure that it and its
|
|
|
|
* descendents are not executed while the capture is in progress.
|
|
|
|
*/
|
|
|
|
|
2020-05-19 14:31:14 +08:00
|
|
|
if (igt_spinner_init(&spin, gt))
|
2020-01-22 22:02:42 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
ve = intel_execlists_create_virtual(siblings, nsibling);
|
|
|
|
if (IS_ERR(ve)) {
|
|
|
|
err = PTR_ERR(ve);
|
|
|
|
goto out_spin;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (n = 0; n < nsibling; n++)
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(siblings[n]);
|
2020-01-22 22:02:42 +08:00
|
|
|
|
|
|
|
rq = igt_spinner_create_request(&spin, ve, MI_ARB_CHECK);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out_heartbeat;
|
|
|
|
}
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
if (!igt_wait_for_spinner(&spin, rq)) {
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -ETIME;
|
|
|
|
goto out_heartbeat;
|
|
|
|
}
|
|
|
|
|
|
|
|
engine = rq->engine;
|
|
|
|
GEM_BUG_ON(engine == ve->engine);
|
|
|
|
|
|
|
|
/* Take ownership of the reset and tasklet */
|
|
|
|
if (test_and_set_bit(I915_RESET_ENGINE + engine->id,
|
|
|
|
>->reset.flags)) {
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -EBUSY;
|
|
|
|
goto out_heartbeat;
|
|
|
|
}
|
|
|
|
tasklet_disable(&engine->execlists.tasklet);
|
|
|
|
|
|
|
|
engine->execlists.tasklet.func(engine->execlists.tasklet.data);
|
|
|
|
GEM_BUG_ON(execlists_active(&engine->execlists) != rq);
|
|
|
|
|
|
|
|
/* Fake a preemption event; failed of course */
|
|
|
|
spin_lock_irq(&engine->active.lock);
|
|
|
|
__unwind_incomplete_requests(engine);
|
|
|
|
spin_unlock_irq(&engine->active.lock);
|
|
|
|
GEM_BUG_ON(rq->engine != ve->engine);
|
|
|
|
|
|
|
|
/* Reset the engine while keeping our active request on hold */
|
|
|
|
execlists_hold(engine, rq);
|
|
|
|
GEM_BUG_ON(!i915_request_on_hold(rq));
|
|
|
|
|
|
|
|
intel_engine_reset(engine, NULL);
|
|
|
|
GEM_BUG_ON(rq->fence.error != -EIO);
|
|
|
|
|
|
|
|
/* Release our grasp on the engine, letting CS flow again */
|
|
|
|
tasklet_enable(&engine->execlists.tasklet);
|
|
|
|
clear_and_wake_up_bit(I915_RESET_ENGINE + engine->id, >->reset.flags);
|
|
|
|
|
|
|
|
/* Check that we do not resubmit the held request */
|
|
|
|
i915_request_get(rq);
|
|
|
|
if (!i915_request_wait(rq, 0, HZ / 5)) {
|
|
|
|
pr_err("%s: on hold request completed!\n",
|
|
|
|
engine->name);
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -EIO;
|
|
|
|
goto out_rq;
|
|
|
|
}
|
|
|
|
GEM_BUG_ON(!i915_request_on_hold(rq));
|
|
|
|
|
|
|
|
/* But is resubmitted on release */
|
|
|
|
execlists_unhold(engine, rq);
|
|
|
|
if (i915_request_wait(rq, 0, HZ / 5) < 0) {
|
|
|
|
pr_err("%s: held request did not complete!\n",
|
|
|
|
engine->name);
|
|
|
|
intel_gt_set_wedged(gt);
|
|
|
|
err = -ETIME;
|
|
|
|
}
|
|
|
|
|
|
|
|
out_rq:
|
|
|
|
i915_request_put(rq);
|
|
|
|
out_heartbeat:
|
|
|
|
for (n = 0; n < nsibling; n++)
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(siblings[n]);
|
2020-01-22 22:02:42 +08:00
|
|
|
|
|
|
|
intel_context_put(ve);
|
|
|
|
out_spin:
|
|
|
|
igt_spinner_fini(&spin);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_virtual_reset(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *siblings[MAX_ENGINE_INSTANCE + 1];
|
2020-05-18 18:29:11 +08:00
|
|
|
unsigned int class;
|
2020-01-22 22:02:42 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that we handle a reset event within a virtual engine.
|
|
|
|
* Only the physical engine is reset, but we have to check the flow
|
|
|
|
* of the virtual requests around the reset, and make sure it is not
|
|
|
|
* forgotten.
|
|
|
|
*/
|
|
|
|
|
2020-02-19 06:33:20 +08:00
|
|
|
if (intel_uc_uses_guc_submission(>->uc))
|
2020-01-22 22:02:42 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!intel_has_reset_engine(gt))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (class = 0; class <= MAX_ENGINE_CLASS; class++) {
|
|
|
|
int nsibling, err;
|
|
|
|
|
2020-05-18 18:29:11 +08:00
|
|
|
nsibling = select_siblings(gt, class, siblings);
|
2020-01-22 22:02:42 +08:00
|
|
|
if (nsibling < 2)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
err = reset_virtual_engine(gt, siblings, nsibling);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-04 17:33:29 +08:00
|
|
|
int intel_execlists_live_selftests(struct drm_i915_private *i915)
|
|
|
|
{
|
|
|
|
static const struct i915_subtest tests[] = {
|
|
|
|
SUBTEST(live_sanitycheck),
|
2019-10-03 02:34:59 +08:00
|
|
|
SUBTEST(live_unlite_switch),
|
|
|
|
SUBTEST(live_unlite_preempt),
|
2020-06-17 02:55:17 +08:00
|
|
|
SUBTEST(live_unlite_ring),
|
2020-02-12 04:56:15 +08:00
|
|
|
SUBTEST(live_pin_rewind),
|
2020-01-17 02:47:53 +08:00
|
|
|
SUBTEST(live_hold_reset),
|
2020-01-29 04:43:15 +08:00
|
|
|
SUBTEST(live_error_interrupt),
|
2019-06-20 22:20:52 +08:00
|
|
|
SUBTEST(live_timeslice_preempt),
|
2020-02-13 22:01:50 +08:00
|
|
|
SUBTEST(live_timeslice_rewind),
|
2019-10-18 15:20:27 +08:00
|
|
|
SUBTEST(live_timeslice_queue),
|
2020-05-28 00:24:18 +08:00
|
|
|
SUBTEST(live_timeslice_nopreempt),
|
2019-03-29 21:40:24 +08:00
|
|
|
SUBTEST(live_busywait_preempt),
|
2018-04-04 17:33:29 +08:00
|
|
|
SUBTEST(live_preempt),
|
|
|
|
SUBTEST(live_late_preempt),
|
2019-07-10 00:42:27 +08:00
|
|
|
SUBTEST(live_nopreempt),
|
2019-10-23 21:31:06 +08:00
|
|
|
SUBTEST(live_preempt_cancel),
|
drm/i915/execlists: Suppress preempting self
In order to avoid preempting ourselves, we currently refuse to schedule
the tasklet if we reschedule an inflight context. However, this glosses
over a few issues such as what happens after a CS completion event and
we then preempt the newly executing context with itself, or if something
else causes a tasklet_schedule triggering the same evaluation to
preempt the active context with itself.
However, when we avoid preempting ELSP[0], we still retain the preemption
value as it may match a second preemption request within the same time period
that we need to resolve after the next CS event. However, since we only
store the maximum preemption priority seen, it may not match the
subsequent event and so we should double check whether or not we
actually do need to trigger a preempt-to-idle by comparing the top
priorities from each queue. Later, this gives us a hook for finer
control over deciding whether the preempt-to-idle is justified.
The sequence of events where we end up preempting for no avail is:
1. Queue requests/contexts A, B
2. Priority boost A; no preemption as it is executing, but keep hint
3. After CS switch, B is less than hint, force preempt-to-idle
4. Resubmit B after idling
v2: We can simplify a bunch of tests based on the knowledge that PI will
ensure that earlier requests along the same context will have the highest
priority.
v3: Demonstrate the stale preemption hint with a selftest
References: a2bf92e8cc16 ("drm/i915/execlists: Avoid kicking priority on the current context")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190129185452.20989-4-chris@chris-wilson.co.uk
2019-01-30 02:54:52 +08:00
|
|
|
SUBTEST(live_suppress_self_preempt),
|
2019-02-05 20:38:35 +08:00
|
|
|
SUBTEST(live_chain_preempt),
|
2020-06-17 07:37:33 +08:00
|
|
|
SUBTEST(live_preempt_ring),
|
2019-11-15 06:57:35 +08:00
|
|
|
SUBTEST(live_preempt_gang),
|
2019-10-23 21:31:05 +08:00
|
|
|
SUBTEST(live_preempt_timeout),
|
2020-04-22 18:09:03 +08:00
|
|
|
SUBTEST(live_preempt_user),
|
2018-09-25 16:31:58 +08:00
|
|
|
SUBTEST(live_preempt_smoke),
|
drm/i915: Load balancing across a virtual engine
Having allowed the user to define a set of engines that they will want
to only use, we go one step further and allow them to bind those engines
into a single virtual instance. Submitting a batch to the virtual engine
will then forward it to any one of the set in a manner as best to
distribute load. The virtual engine has a single timeline across all
engines (it operates as a single queue), so it is not able to concurrently
run batches across multiple engines by itself; that is left up to the user
to submit multiple concurrent batches to multiple queues. Multiple users
will be load balanced across the system.
The mechanism used for load balancing in this patch is a late greedy
balancer. When a request is ready for execution, it is added to each
engine's queue, and when an engine is ready for its next request it
claims it from the virtual engine. The first engine to do so, wins, i.e.
the request is executed at the earliest opportunity (idle moment) in the
system.
As not all HW is created equal, the user is still able to skip the
virtual engine and execute the batch on a specific engine, all within the
same queue. It will then be executed in order on the correct engine,
with execution on other virtual engines being moved away due to the load
detection.
A couple of areas for potential improvement left!
- The virtual engine always take priority over equal-priority tasks.
Mostly broken up by applying FQ_CODEL rules for prioritising new clients,
and hopefully the virtual and real engines are not then congested (i.e.
all work is via virtual engines, or all work is to the real engine).
- We require the breadcrumb irq around every virtual engine request. For
normal engines, we eliminate the need for the slow round trip via
interrupt by using the submit fence and queueing in order. For virtual
engines, we have to allow any job to transfer to a new ring, and cannot
coalesce the submissions, so require the completion fence instead,
forcing the persistent use of interrupts.
- We only drip feed single requests through each virtual engine and onto
the physical engines, even if there was enough work to fill all ELSP,
leaving small stalls with an idle CS event at the end of every request.
Could we be greedy and fill both slots? Being lazy is virtuous for load
distribution on less-than-full workloads though.
Other areas of improvement are more general, such as reducing lock
contention, reducing dispatch overhead, looking at direct submission
rather than bouncing around tasklets etc.
sseu: Lift the restriction to allow sseu to be reconfigured on virtual
engines composed of RENDER_CLASS (rcs).
v2: macroize check_user_mbz()
v3: Cancel virtual engines on wedging
v4: Commence commenting
v5: Replace 64b sibling_mask with a list of class:instance
v6: Drop the one-element array in the uabi
v7: Assert it is an virtual engine in to_virtual_engine()
v8: Skip over holes in [class][inst] so we can selftest with (vcs0, vcs2)
Link: https://github.com/intel/media-driver/pull/283
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190521211134.16117-6-chris@chris-wilson.co.uk
2019-05-22 05:11:30 +08:00
|
|
|
SUBTEST(live_virtual_engine),
|
2019-05-22 05:11:31 +08:00
|
|
|
SUBTEST(live_virtual_mask),
|
2019-10-10 19:02:52 +08:00
|
|
|
SUBTEST(live_virtual_preserved),
|
2020-05-19 21:20:44 +08:00
|
|
|
SUBTEST(live_virtual_slice),
|
2019-05-22 05:11:33 +08:00
|
|
|
SUBTEST(live_virtual_bond),
|
2020-01-22 22:02:42 +08:00
|
|
|
SUBTEST(live_virtual_reset),
|
2018-04-04 17:33:29 +08:00
|
|
|
};
|
2018-05-04 20:42:02 +08:00
|
|
|
|
|
|
|
if (!HAS_EXECLISTS(i915))
|
|
|
|
return 0;
|
|
|
|
|
2019-07-13 03:29:53 +08:00
|
|
|
if (intel_gt_is_wedged(&i915->gt))
|
2018-07-06 19:45:10 +08:00
|
|
|
return 0;
|
|
|
|
|
2019-10-16 20:02:49 +08:00
|
|
|
return intel_gt_live_subtests(tests, &i915->gt);
|
2018-04-04 17:33:29 +08:00
|
|
|
}
|
2019-09-24 22:59:50 +08:00
|
|
|
|
2020-02-19 19:20:03 +08:00
|
|
|
static int emit_semaphore_signal(struct intel_context *ce, void *slot)
|
|
|
|
{
|
|
|
|
const u32 offset =
|
|
|
|
i915_ggtt_offset(ce->engine->status_page.vma) +
|
|
|
|
offset_in_page(slot);
|
|
|
|
struct i915_request *rq;
|
|
|
|
u32 *cs;
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
return PTR_ERR(rq);
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 4);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
return PTR_ERR(cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_DWORD_IMM_GEN4 | MI_USE_GGTT;
|
|
|
|
*cs++ = offset;
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = 1;
|
|
|
|
|
|
|
|
intel_ring_advance(rq, cs);
|
|
|
|
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_BARRIER;
|
|
|
|
i915_request_add(rq);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-02-28 16:23:24 +08:00
|
|
|
static int context_flush(struct intel_context *ce, long timeout)
|
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct dma_fence *fence;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
rq = intel_engine_create_kernel_request(ce->engine);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
return PTR_ERR(rq);
|
|
|
|
|
|
|
|
fence = i915_active_fence_get(&ce->timeline->last_request);
|
|
|
|
if (fence) {
|
|
|
|
i915_request_await_dma_fence(rq, fence);
|
|
|
|
dma_fence_put(fence);
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (i915_request_wait(rq, 0, timeout) < 0)
|
|
|
|
err = -ETIME;
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
rmb(); /* We know the request is written, make sure all state is too! */
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-09-24 22:59:50 +08:00
|
|
|
static int live_lrc_layout(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
2020-01-02 21:17:04 +08:00
|
|
|
u32 *lrc;
|
2019-09-24 22:59:50 +08:00
|
|
|
int err;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check the registers offsets we use to create the initial reg state
|
|
|
|
* match the layout saved by HW.
|
|
|
|
*/
|
|
|
|
|
2020-01-02 21:17:04 +08:00
|
|
|
lrc = kmalloc(PAGE_SIZE, GFP_KERNEL);
|
|
|
|
if (!lrc)
|
2019-09-24 22:59:50 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
err = 0;
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2020-01-02 21:17:04 +08:00
|
|
|
u32 *hw;
|
2019-09-24 22:59:50 +08:00
|
|
|
int dw;
|
|
|
|
|
|
|
|
if (!engine->default_state)
|
|
|
|
continue;
|
|
|
|
|
drm/i915/gt: Keep a no-frills swappable copy of the default context state
We need to keep the default context state around to instantiate new
contexts (aka golden rendercontext), and we also keep it pinned while
the engine is active so that we can quickly reset a hanging context.
However, the default contexts are large enough to merit keeping in
swappable memory as opposed to kernel memory, so we store them inside
shmemfs. Currently, we use the normal GEM objects to create the default
context image, but we can throw away all but the shmemfs file.
This greatly simplifies the tricky power management code which wants to
run underneath the normal GT locking, and we definitely do not want to
use any high level objects that may appear to recurse back into the GT.
Though perhaps the primary advantage of the complex GEM object is that
we aggressively cache the mapping, but here we are recreating the
vm_area everytime time we unpark. At the worst, we add a lightweight
cache, but first find a microbenchmark that is impacted.
Having started to create some utility functions to make working with
shmemfs objects easier, we can start putting them to wider use, where
GEM objects are overkill, such as storing persistent error state.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Matthew Auld <matthew.auld@intel.com>
Cc: Ramalingam C <ramalingam.c@intel.com>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Matthew Auld <matthew.auld@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200429172429.6054-1-chris@chris-wilson.co.uk
2020-04-30 01:24:29 +08:00
|
|
|
hw = shmem_pin_map(engine->default_state);
|
2019-09-24 22:59:50 +08:00
|
|
|
if (IS_ERR(hw)) {
|
|
|
|
err = PTR_ERR(hw);
|
|
|
|
break;
|
|
|
|
}
|
2020-04-24 02:23:52 +08:00
|
|
|
hw += LRC_STATE_OFFSET / sizeof(*hw);
|
2019-09-24 22:59:50 +08:00
|
|
|
|
2020-01-02 21:17:04 +08:00
|
|
|
execlists_init_reg_state(memset(lrc, POISON_INUSE, PAGE_SIZE),
|
2019-09-24 22:59:50 +08:00
|
|
|
engine->kernel_context,
|
|
|
|
engine,
|
|
|
|
engine->kernel_context->ring,
|
|
|
|
true);
|
|
|
|
|
|
|
|
dw = 0;
|
|
|
|
do {
|
|
|
|
u32 lri = hw[dw];
|
|
|
|
|
|
|
|
if (lri == 0) {
|
|
|
|
dw++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-01-02 21:17:03 +08:00
|
|
|
if (lrc[dw] == 0) {
|
|
|
|
pr_debug("%s: skipped instruction %x at dword %d\n",
|
|
|
|
engine->name, lri, dw);
|
|
|
|
dw++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-09-24 22:59:50 +08:00
|
|
|
if ((lri & GENMASK(31, 23)) != MI_INSTR(0x22, 0)) {
|
|
|
|
pr_err("%s: Expected LRI command at dword %d, found %08x\n",
|
|
|
|
engine->name, dw, lri);
|
|
|
|
err = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lrc[dw] != lri) {
|
|
|
|
pr_err("%s: LRI command mismatch at dword %d, expected %08x found %08x\n",
|
|
|
|
engine->name, dw, lri, lrc[dw]);
|
|
|
|
err = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
lri &= 0x7f;
|
|
|
|
lri++;
|
|
|
|
dw++;
|
|
|
|
|
|
|
|
while (lri) {
|
|
|
|
if (hw[dw] != lrc[dw]) {
|
|
|
|
pr_err("%s: Different registers found at dword %d, expected %x, found %x\n",
|
|
|
|
engine->name, dw, hw[dw], lrc[dw]);
|
|
|
|
err = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Skip over the actual register value as we
|
|
|
|
* expect that to differ.
|
|
|
|
*/
|
|
|
|
dw += 2;
|
|
|
|
lri -= 2;
|
|
|
|
}
|
|
|
|
} while ((lrc[dw] & ~BIT(0)) != MI_BATCH_BUFFER_END);
|
|
|
|
|
|
|
|
if (err) {
|
|
|
|
pr_info("%s: HW register image:\n", engine->name);
|
2020-05-04 22:06:29 +08:00
|
|
|
igt_hexdump(hw, PAGE_SIZE);
|
2019-09-24 22:59:50 +08:00
|
|
|
|
|
|
|
pr_info("%s: SW register image:\n", engine->name);
|
2020-05-04 22:06:29 +08:00
|
|
|
igt_hexdump(lrc, PAGE_SIZE);
|
2019-09-24 22:59:50 +08:00
|
|
|
}
|
|
|
|
|
drm/i915/gt: Keep a no-frills swappable copy of the default context state
We need to keep the default context state around to instantiate new
contexts (aka golden rendercontext), and we also keep it pinned while
the engine is active so that we can quickly reset a hanging context.
However, the default contexts are large enough to merit keeping in
swappable memory as opposed to kernel memory, so we store them inside
shmemfs. Currently, we use the normal GEM objects to create the default
context image, but we can throw away all but the shmemfs file.
This greatly simplifies the tricky power management code which wants to
run underneath the normal GT locking, and we definitely do not want to
use any high level objects that may appear to recurse back into the GT.
Though perhaps the primary advantage of the complex GEM object is that
we aggressively cache the mapping, but here we are recreating the
vm_area everytime time we unpark. At the worst, we add a lightweight
cache, but first find a microbenchmark that is impacted.
Having started to create some utility functions to make working with
shmemfs objects easier, we can start putting them to wider use, where
GEM objects are overkill, such as storing persistent error state.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Matthew Auld <matthew.auld@intel.com>
Cc: Ramalingam C <ramalingam.c@intel.com>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Matthew Auld <matthew.auld@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200429172429.6054-1-chris@chris-wilson.co.uk
2020-04-30 01:24:29 +08:00
|
|
|
shmem_unpin_map(engine->default_state, hw);
|
2019-09-24 22:59:50 +08:00
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-01-02 21:17:04 +08:00
|
|
|
kfree(lrc);
|
2019-09-24 22:59:50 +08:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-10-26 16:22:20 +08:00
|
|
|
static int find_offset(const u32 *lri, u32 offset)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < PAGE_SIZE / sizeof(u32); i++)
|
|
|
|
if (lri[i] == offset)
|
|
|
|
return i;
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_lrc_fixed(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check the assumed register offsets match the actual locations in
|
|
|
|
* the context image.
|
|
|
|
*/
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
const struct {
|
|
|
|
u32 reg;
|
|
|
|
u32 offset;
|
|
|
|
const char *name;
|
|
|
|
} tbl[] = {
|
2019-10-28 20:18:03 +08:00
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_START(engine->mmio_base)),
|
2019-11-02 20:57:39 +08:00
|
|
|
CTX_RING_START - 1,
|
2019-10-28 20:18:03 +08:00
|
|
|
"RING_START"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_CTL(engine->mmio_base)),
|
2019-11-02 20:57:39 +08:00
|
|
|
CTX_RING_CTL - 1,
|
2019-10-28 20:18:03 +08:00
|
|
|
"RING_CTL"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_HEAD(engine->mmio_base)),
|
|
|
|
CTX_RING_HEAD - 1,
|
|
|
|
"RING_HEAD"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_TAIL(engine->mmio_base)),
|
|
|
|
CTX_RING_TAIL - 1,
|
|
|
|
"RING_TAIL"
|
|
|
|
},
|
2019-10-26 16:22:20 +08:00
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_MI_MODE(engine->mmio_base)),
|
|
|
|
lrc_ring_mi_mode(engine),
|
2019-10-28 20:18:03 +08:00
|
|
|
"RING_MI_MODE"
|
|
|
|
},
|
|
|
|
{
|
2019-11-02 20:57:39 +08:00
|
|
|
i915_mmio_reg_offset(RING_BBSTATE(engine->mmio_base)),
|
2019-10-28 20:18:03 +08:00
|
|
|
CTX_BB_STATE - 1,
|
|
|
|
"BB_STATE"
|
2019-10-26 16:22:20 +08:00
|
|
|
},
|
2020-04-24 06:41:59 +08:00
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_BB_PER_CTX_PTR(engine->mmio_base)),
|
|
|
|
lrc_ring_wa_bb_per_ctx(engine),
|
|
|
|
"RING_BB_PER_CTX_PTR"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_INDIRECT_CTX(engine->mmio_base)),
|
|
|
|
lrc_ring_indirect_ptr(engine),
|
|
|
|
"RING_INDIRECT_CTX_PTR"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_INDIRECT_CTX_OFFSET(engine->mmio_base)),
|
|
|
|
lrc_ring_indirect_offset(engine),
|
|
|
|
"RING_INDIRECT_CTX_OFFSET"
|
|
|
|
},
|
2020-02-16 21:36:20 +08:00
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_CTX_TIMESTAMP(engine->mmio_base)),
|
|
|
|
CTX_TIMESTAMP - 1,
|
|
|
|
"RING_CTX_TIMESTAMP"
|
|
|
|
},
|
2020-04-25 05:48:38 +08:00
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(GEN8_RING_CS_GPR(engine->mmio_base, 0)),
|
|
|
|
lrc_ring_gpr0(engine),
|
|
|
|
"RING_CS_GPR0"
|
|
|
|
},
|
2020-04-25 07:06:32 +08:00
|
|
|
{
|
|
|
|
i915_mmio_reg_offset(RING_CMD_BUF_CCTL(engine->mmio_base)),
|
|
|
|
lrc_ring_cmd_buf_cctl(engine),
|
|
|
|
"RING_CMD_BUF_CCTL"
|
|
|
|
},
|
2019-10-26 16:22:20 +08:00
|
|
|
{ },
|
|
|
|
}, *t;
|
|
|
|
u32 *hw;
|
|
|
|
|
|
|
|
if (!engine->default_state)
|
|
|
|
continue;
|
|
|
|
|
drm/i915/gt: Keep a no-frills swappable copy of the default context state
We need to keep the default context state around to instantiate new
contexts (aka golden rendercontext), and we also keep it pinned while
the engine is active so that we can quickly reset a hanging context.
However, the default contexts are large enough to merit keeping in
swappable memory as opposed to kernel memory, so we store them inside
shmemfs. Currently, we use the normal GEM objects to create the default
context image, but we can throw away all but the shmemfs file.
This greatly simplifies the tricky power management code which wants to
run underneath the normal GT locking, and we definitely do not want to
use any high level objects that may appear to recurse back into the GT.
Though perhaps the primary advantage of the complex GEM object is that
we aggressively cache the mapping, but here we are recreating the
vm_area everytime time we unpark. At the worst, we add a lightweight
cache, but first find a microbenchmark that is impacted.
Having started to create some utility functions to make working with
shmemfs objects easier, we can start putting them to wider use, where
GEM objects are overkill, such as storing persistent error state.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Matthew Auld <matthew.auld@intel.com>
Cc: Ramalingam C <ramalingam.c@intel.com>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Matthew Auld <matthew.auld@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200429172429.6054-1-chris@chris-wilson.co.uk
2020-04-30 01:24:29 +08:00
|
|
|
hw = shmem_pin_map(engine->default_state);
|
2019-10-26 16:22:20 +08:00
|
|
|
if (IS_ERR(hw)) {
|
|
|
|
err = PTR_ERR(hw);
|
|
|
|
break;
|
|
|
|
}
|
2020-04-24 02:23:52 +08:00
|
|
|
hw += LRC_STATE_OFFSET / sizeof(*hw);
|
2019-10-26 16:22:20 +08:00
|
|
|
|
|
|
|
for (t = tbl; t->name; t++) {
|
|
|
|
int dw = find_offset(hw, t->reg);
|
|
|
|
|
|
|
|
if (dw != t->offset) {
|
|
|
|
pr_err("%s: Offset for %s [0x%x] mismatch, found %x, expected %x\n",
|
|
|
|
engine->name,
|
|
|
|
t->name,
|
|
|
|
t->reg,
|
|
|
|
dw,
|
|
|
|
t->offset);
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
drm/i915/gt: Keep a no-frills swappable copy of the default context state
We need to keep the default context state around to instantiate new
contexts (aka golden rendercontext), and we also keep it pinned while
the engine is active so that we can quickly reset a hanging context.
However, the default contexts are large enough to merit keeping in
swappable memory as opposed to kernel memory, so we store them inside
shmemfs. Currently, we use the normal GEM objects to create the default
context image, but we can throw away all but the shmemfs file.
This greatly simplifies the tricky power management code which wants to
run underneath the normal GT locking, and we definitely do not want to
use any high level objects that may appear to recurse back into the GT.
Though perhaps the primary advantage of the complex GEM object is that
we aggressively cache the mapping, but here we are recreating the
vm_area everytime time we unpark. At the worst, we add a lightweight
cache, but first find a microbenchmark that is impacted.
Having started to create some utility functions to make working with
shmemfs objects easier, we can start putting them to wider use, where
GEM objects are overkill, such as storing persistent error state.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Matthew Auld <matthew.auld@intel.com>
Cc: Ramalingam C <ramalingam.c@intel.com>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Matthew Auld <matthew.auld@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200429172429.6054-1-chris@chris-wilson.co.uk
2020-04-30 01:24:29 +08:00
|
|
|
shmem_unpin_map(engine->default_state, hw);
|
2019-10-26 16:22:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
static int __live_lrc_state(struct intel_engine_cs *engine,
|
2019-10-14 17:07:48 +08:00
|
|
|
struct i915_vma *scratch)
|
|
|
|
{
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
2020-08-19 22:08:59 +08:00
|
|
|
struct i915_gem_ww_ctx ww;
|
2019-10-14 17:07:48 +08:00
|
|
|
enum {
|
|
|
|
RING_START_IDX = 0,
|
|
|
|
RING_TAIL_IDX,
|
|
|
|
MAX_IDX
|
|
|
|
};
|
|
|
|
u32 expected[MAX_IDX];
|
|
|
|
u32 *cs;
|
|
|
|
int err;
|
|
|
|
int n;
|
|
|
|
|
2019-12-22 00:03:24 +08:00
|
|
|
ce = intel_context_create(engine);
|
2019-10-14 17:07:48 +08:00
|
|
|
if (IS_ERR(ce))
|
|
|
|
return PTR_ERR(ce);
|
|
|
|
|
2020-08-19 22:08:59 +08:00
|
|
|
i915_gem_ww_ctx_init(&ww, false);
|
|
|
|
retry:
|
|
|
|
err = i915_gem_object_lock(scratch->obj, &ww);
|
|
|
|
if (!err)
|
|
|
|
err = intel_context_pin_ww(ce, &ww);
|
2019-10-14 17:07:48 +08:00
|
|
|
if (err)
|
|
|
|
goto err_put;
|
|
|
|
|
|
|
|
rq = i915_request_create(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_unpin;
|
|
|
|
}
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 4 * MAX_IDX);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
i915_request_add(rq);
|
|
|
|
goto err_unpin;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_REGISTER_MEM_GEN8 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_mmio_reg_offset(RING_START(engine->mmio_base));
|
|
|
|
*cs++ = i915_ggtt_offset(scratch) + RING_START_IDX * sizeof(u32);
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
expected[RING_START_IDX] = i915_ggtt_offset(ce->ring->vma);
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_REGISTER_MEM_GEN8 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_mmio_reg_offset(RING_TAIL(engine->mmio_base));
|
|
|
|
*cs++ = i915_ggtt_offset(scratch) + RING_TAIL_IDX * sizeof(u32);
|
|
|
|
*cs++ = 0;
|
|
|
|
|
2020-01-31 22:26:10 +08:00
|
|
|
err = i915_request_await_object(rq, scratch->obj, true);
|
|
|
|
if (!err)
|
|
|
|
err = i915_vma_move_to_active(scratch, rq, EXEC_OBJECT_WRITE);
|
|
|
|
|
2019-10-14 17:07:48 +08:00
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
2020-01-31 22:26:10 +08:00
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
2019-10-14 17:07:48 +08:00
|
|
|
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
expected[RING_TAIL_IDX] = ce->ring->tail;
|
|
|
|
|
|
|
|
if (i915_request_wait(rq, 0, HZ / 5) < 0) {
|
|
|
|
err = -ETIME;
|
|
|
|
goto err_rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
cs = i915_gem_object_pin_map(scratch->obj, I915_MAP_WB);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto err_rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (n = 0; n < MAX_IDX; n++) {
|
|
|
|
if (cs[n] != expected[n]) {
|
|
|
|
pr_err("%s: Stored register[%d] value[0x%x] did not match expected[0x%x]\n",
|
|
|
|
engine->name, n, cs[n], expected[n]);
|
|
|
|
err = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_gem_object_unpin_map(scratch->obj);
|
|
|
|
|
|
|
|
err_rq:
|
|
|
|
i915_request_put(rq);
|
|
|
|
err_unpin:
|
|
|
|
intel_context_unpin(ce);
|
|
|
|
err_put:
|
2020-08-19 22:08:59 +08:00
|
|
|
if (err == -EDEADLK) {
|
|
|
|
err = i915_gem_ww_ctx_backoff(&ww);
|
|
|
|
if (!err)
|
|
|
|
goto retry;
|
|
|
|
}
|
|
|
|
i915_gem_ww_ctx_fini(&ww);
|
2019-10-14 17:07:48 +08:00
|
|
|
intel_context_put(ce);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_lrc_state(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct i915_vma *scratch;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check the live register state matches what we expect for this
|
|
|
|
* intel_context.
|
|
|
|
*/
|
|
|
|
|
|
|
|
scratch = create_scratch(gt);
|
2019-12-22 00:03:24 +08:00
|
|
|
if (IS_ERR(scratch))
|
|
|
|
return PTR_ERR(scratch);
|
2019-10-14 17:07:48 +08:00
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2019-12-22 00:03:24 +08:00
|
|
|
err = __live_lrc_state(engine, scratch);
|
2019-10-14 17:07:48 +08:00
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (igt_flush_test(gt->i915))
|
|
|
|
err = -EIO;
|
|
|
|
|
|
|
|
i915_vma_unpin_and_release(&scratch, 0);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-02-19 20:34:18 +08:00
|
|
|
static int gpr_make_dirty(struct intel_context *ce)
|
2019-10-14 17:07:49 +08:00
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
u32 *cs;
|
|
|
|
int n;
|
|
|
|
|
2020-02-19 20:34:18 +08:00
|
|
|
rq = intel_context_create_request(ce);
|
2019-10-14 17:07:49 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
return PTR_ERR(rq);
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 2 * NUM_GPR_DW + 2);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
return PTR_ERR(cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_LOAD_REGISTER_IMM(NUM_GPR_DW);
|
|
|
|
for (n = 0; n < NUM_GPR_DW; n++) {
|
2020-02-19 20:34:18 +08:00
|
|
|
*cs++ = CS_GPR(ce->engine, n);
|
2019-10-14 17:07:49 +08:00
|
|
|
*cs++ = STACK_MAGIC;
|
|
|
|
}
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
|
|
|
|
intel_ring_advance(rq, cs);
|
2020-02-19 20:34:18 +08:00
|
|
|
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_BARRIER;
|
2019-10-14 17:07:49 +08:00
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-02-19 20:34:18 +08:00
|
|
|
static struct i915_request *
|
|
|
|
__gpr_read(struct intel_context *ce, struct i915_vma *scratch, u32 *slot)
|
2019-10-14 17:07:49 +08:00
|
|
|
{
|
2020-02-19 20:34:18 +08:00
|
|
|
const u32 offset =
|
|
|
|
i915_ggtt_offset(ce->engine->status_page.vma) +
|
|
|
|
offset_in_page(slot);
|
2019-10-14 17:07:49 +08:00
|
|
|
struct i915_request *rq;
|
|
|
|
u32 *cs;
|
|
|
|
int err;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
2020-02-19 20:34:18 +08:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
return rq;
|
2019-10-14 17:07:49 +08:00
|
|
|
|
2020-02-19 20:34:18 +08:00
|
|
|
cs = intel_ring_begin(rq, 6 + 4 * NUM_GPR_DW);
|
2019-10-14 17:07:49 +08:00
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_request_add(rq);
|
2020-02-19 20:34:18 +08:00
|
|
|
return ERR_CAST(cs);
|
2019-10-14 17:07:49 +08:00
|
|
|
}
|
|
|
|
|
2020-02-19 20:34:18 +08:00
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_ENABLE;
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
|
|
|
|
*cs++ = MI_SEMAPHORE_WAIT |
|
|
|
|
MI_SEMAPHORE_GLOBAL_GTT |
|
|
|
|
MI_SEMAPHORE_POLL |
|
|
|
|
MI_SEMAPHORE_SAD_NEQ_SDD;
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = offset;
|
|
|
|
*cs++ = 0;
|
|
|
|
|
2019-10-14 17:07:49 +08:00
|
|
|
for (n = 0; n < NUM_GPR_DW; n++) {
|
|
|
|
*cs++ = MI_STORE_REGISTER_MEM_GEN8 | MI_USE_GGTT;
|
2020-02-19 20:34:18 +08:00
|
|
|
*cs++ = CS_GPR(ce->engine, n);
|
2019-10-14 17:07:49 +08:00
|
|
|
*cs++ = i915_ggtt_offset(scratch) + n * sizeof(u32);
|
|
|
|
*cs++ = 0;
|
|
|
|
}
|
|
|
|
|
2020-01-31 22:26:10 +08:00
|
|
|
i915_vma_lock(scratch);
|
|
|
|
err = i915_request_await_object(rq, scratch->obj, true);
|
|
|
|
if (!err)
|
|
|
|
err = i915_vma_move_to_active(scratch, rq, EXEC_OBJECT_WRITE);
|
|
|
|
i915_vma_unlock(scratch);
|
|
|
|
|
2019-10-14 17:07:49 +08:00
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
2020-02-19 20:34:18 +08:00
|
|
|
if (err) {
|
|
|
|
i915_request_put(rq);
|
|
|
|
rq = ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __live_lrc_gpr(struct intel_engine_cs *engine,
|
|
|
|
struct i915_vma *scratch,
|
|
|
|
bool preempt)
|
|
|
|
{
|
|
|
|
u32 *slot = memset32(engine->status_page.addr + 1000, 0, 4);
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
u32 *cs;
|
|
|
|
int err;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (INTEL_GEN(engine->i915) < 9 && engine->class != RENDER_CLASS)
|
|
|
|
return 0; /* GPR only on rcs0 for gen8 */
|
|
|
|
|
|
|
|
err = gpr_make_dirty(engine->kernel_context);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce))
|
|
|
|
return PTR_ERR(ce);
|
|
|
|
|
|
|
|
rq = __gpr_read(ce, scratch, slot);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_put;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = wait_for_submit(engine, rq, HZ / 2);
|
2020-01-31 22:26:10 +08:00
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
2019-10-14 17:07:49 +08:00
|
|
|
|
2020-02-19 20:34:18 +08:00
|
|
|
if (preempt) {
|
|
|
|
err = gpr_make_dirty(engine->kernel_context);
|
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
|
|
|
|
|
|
|
err = emit_semaphore_signal(engine->kernel_context, slot);
|
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
|
|
|
} else {
|
|
|
|
slot[0] = 1;
|
|
|
|
wmb();
|
|
|
|
}
|
|
|
|
|
2019-10-14 17:07:49 +08:00
|
|
|
if (i915_request_wait(rq, 0, HZ / 5) < 0) {
|
|
|
|
err = -ETIME;
|
|
|
|
goto err_rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
cs = i915_gem_object_pin_map(scratch->obj, I915_MAP_WB);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto err_rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (n = 0; n < NUM_GPR_DW; n++) {
|
|
|
|
if (cs[n]) {
|
|
|
|
pr_err("%s: GPR[%d].%s was not zero, found 0x%08x!\n",
|
|
|
|
engine->name,
|
|
|
|
n / 2, n & 1 ? "udw" : "ldw",
|
|
|
|
cs[n]);
|
|
|
|
err = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_gem_object_unpin_map(scratch->obj);
|
|
|
|
|
|
|
|
err_rq:
|
2020-02-19 20:34:18 +08:00
|
|
|
memset32(&slot[0], -1, 4);
|
|
|
|
wmb();
|
2019-10-14 17:07:49 +08:00
|
|
|
i915_request_put(rq);
|
|
|
|
err_put:
|
|
|
|
intel_context_put(ce);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-02-19 20:34:18 +08:00
|
|
|
static int live_lrc_gpr(void *arg)
|
2019-10-14 17:07:49 +08:00
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct i915_vma *scratch;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that GPR registers are cleared in new contexts as we need
|
|
|
|
* to avoid leaking any information from previous contexts.
|
|
|
|
*/
|
|
|
|
|
|
|
|
scratch = create_scratch(gt);
|
2019-12-22 00:03:24 +08:00
|
|
|
if (IS_ERR(scratch))
|
|
|
|
return PTR_ERR(scratch);
|
2019-10-14 17:07:49 +08:00
|
|
|
|
2019-10-17 17:45:00 +08:00
|
|
|
for_each_engine(engine, gt, id) {
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(engine);
|
2020-02-19 20:34:18 +08:00
|
|
|
|
|
|
|
err = __live_lrc_gpr(engine, scratch, false);
|
|
|
|
if (err)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
err = __live_lrc_gpr(engine, scratch, true);
|
|
|
|
if (err)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
err:
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(engine);
|
2020-02-19 20:34:18 +08:00
|
|
|
if (igt_flush_test(gt->i915))
|
|
|
|
err = -EIO;
|
2019-10-14 17:07:49 +08:00
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_vma_unpin_and_release(&scratch, 0);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-02-19 19:20:03 +08:00
|
|
|
static struct i915_request *
|
|
|
|
create_timestamp(struct intel_context *ce, void *slot, int idx)
|
|
|
|
{
|
|
|
|
const u32 offset =
|
|
|
|
i915_ggtt_offset(ce->engine->status_page.vma) +
|
|
|
|
offset_in_page(slot);
|
|
|
|
struct i915_request *rq;
|
|
|
|
u32 *cs;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
return rq;
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 10);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_ENABLE;
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
|
|
|
|
*cs++ = MI_SEMAPHORE_WAIT |
|
|
|
|
MI_SEMAPHORE_GLOBAL_GTT |
|
|
|
|
MI_SEMAPHORE_POLL |
|
|
|
|
MI_SEMAPHORE_SAD_NEQ_SDD;
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = offset;
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_REGISTER_MEM_GEN8 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_mmio_reg_offset(RING_CTX_TIMESTAMP(rq->engine->mmio_base));
|
|
|
|
*cs++ = offset + idx * sizeof(u32);
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
intel_ring_advance(rq, cs);
|
|
|
|
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_MASK;
|
|
|
|
err = 0;
|
|
|
|
err:
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
if (err) {
|
|
|
|
i915_request_put(rq);
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct lrc_timestamp {
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
struct intel_context *ce[2];
|
|
|
|
u32 poison;
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool timestamp_advanced(u32 start, u32 end)
|
|
|
|
{
|
|
|
|
return (s32)(end - start) > 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __lrc_timestamp(const struct lrc_timestamp *arg, bool preempt)
|
|
|
|
{
|
|
|
|
u32 *slot = memset32(arg->engine->status_page.addr + 1000, 0, 4);
|
|
|
|
struct i915_request *rq;
|
|
|
|
u32 timestamp;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
arg->ce[0]->lrc_reg_state[CTX_TIMESTAMP] = arg->poison;
|
|
|
|
rq = create_timestamp(arg->ce[0], slot, 1);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
return PTR_ERR(rq);
|
|
|
|
|
|
|
|
err = wait_for_submit(rq->engine, rq, HZ / 2);
|
|
|
|
if (err)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
if (preempt) {
|
|
|
|
arg->ce[1]->lrc_reg_state[CTX_TIMESTAMP] = 0xdeadbeef;
|
|
|
|
err = emit_semaphore_signal(arg->ce[1], slot);
|
|
|
|
if (err)
|
|
|
|
goto err;
|
|
|
|
} else {
|
|
|
|
slot[0] = 1;
|
|
|
|
wmb();
|
|
|
|
}
|
|
|
|
|
2020-02-28 16:23:24 +08:00
|
|
|
/* And wait for switch to kernel (to save our context to memory) */
|
|
|
|
err = context_flush(arg->ce[0], HZ / 2);
|
|
|
|
if (err)
|
2020-02-19 19:20:03 +08:00
|
|
|
goto err;
|
|
|
|
|
|
|
|
if (!timestamp_advanced(arg->poison, slot[1])) {
|
|
|
|
pr_err("%s(%s): invalid timestamp on restore, context:%x, request:%x\n",
|
|
|
|
arg->engine->name, preempt ? "preempt" : "simple",
|
|
|
|
arg->poison, slot[1]);
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
timestamp = READ_ONCE(arg->ce[0]->lrc_reg_state[CTX_TIMESTAMP]);
|
|
|
|
if (!timestamp_advanced(slot[1], timestamp)) {
|
|
|
|
pr_err("%s(%s): invalid timestamp on save, request:%x, context:%x\n",
|
|
|
|
arg->engine->name, preempt ? "preempt" : "simple",
|
|
|
|
slot[1], timestamp);
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
err:
|
|
|
|
memset32(slot, -1, 4);
|
|
|
|
i915_request_put(rq);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_lrc_timestamp(void *arg)
|
|
|
|
{
|
2020-03-03 22:23:47 +08:00
|
|
|
struct lrc_timestamp data = {};
|
2020-02-19 19:20:03 +08:00
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
const u32 poison[] = {
|
|
|
|
0,
|
|
|
|
S32_MAX,
|
|
|
|
(u32)S32_MAX + 1,
|
|
|
|
U32_MAX,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We want to verify that the timestamp is saved and restore across
|
|
|
|
* context switches and is monotonic.
|
|
|
|
*
|
|
|
|
* So we do this with a little bit of LRC poisoning to check various
|
|
|
|
* boundary conditions, and see what happens if we preempt the context
|
|
|
|
* with a second request (carrying more poison into the timestamp).
|
|
|
|
*/
|
|
|
|
|
|
|
|
for_each_engine(data.engine, gt, id) {
|
|
|
|
int i, err = 0;
|
|
|
|
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_disable(data.engine);
|
2020-02-19 19:20:03 +08:00
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(data.ce); i++) {
|
|
|
|
struct intel_context *tmp;
|
|
|
|
|
|
|
|
tmp = intel_context_create(data.engine);
|
|
|
|
if (IS_ERR(tmp)) {
|
|
|
|
err = PTR_ERR(tmp);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = intel_context_pin(tmp);
|
|
|
|
if (err) {
|
|
|
|
intel_context_put(tmp);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
data.ce[i] = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(poison); i++) {
|
|
|
|
data.poison = poison[i];
|
|
|
|
|
|
|
|
err = __lrc_timestamp(&data, false);
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
|
|
|
|
err = __lrc_timestamp(&data, true);
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
err:
|
2020-06-17 21:09:15 +08:00
|
|
|
st_engine_heartbeat_enable(data.engine);
|
2020-02-19 19:20:03 +08:00
|
|
|
for (i = 0; i < ARRAY_SIZE(data.ce); i++) {
|
|
|
|
if (!data.ce[i])
|
|
|
|
break;
|
|
|
|
|
|
|
|
intel_context_unpin(data.ce[i]);
|
|
|
|
intel_context_put(data.ce[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (igt_flush_test(gt->i915))
|
|
|
|
err = -EIO;
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-02-27 16:57:15 +08:00
|
|
|
static struct i915_vma *
|
|
|
|
create_user_vma(struct i915_address_space *vm, unsigned long size)
|
|
|
|
{
|
|
|
|
struct drm_i915_gem_object *obj;
|
|
|
|
struct i915_vma *vma;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
obj = i915_gem_object_create_internal(vm->i915, size);
|
|
|
|
if (IS_ERR(obj))
|
|
|
|
return ERR_CAST(obj);
|
|
|
|
|
|
|
|
vma = i915_vma_instance(obj, vm, NULL);
|
|
|
|
if (IS_ERR(vma)) {
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
return vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = i915_vma_pin(vma, 0, 0, PIN_USER);
|
|
|
|
if (err) {
|
|
|
|
i915_gem_object_put(obj);
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return vma;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct i915_vma *
|
|
|
|
store_context(struct intel_context *ce, struct i915_vma *scratch)
|
|
|
|
{
|
|
|
|
struct i915_vma *batch;
|
|
|
|
u32 dw, x, *cs, *hw;
|
2020-05-05 02:07:45 +08:00
|
|
|
u32 *defaults;
|
2020-02-27 16:57:15 +08:00
|
|
|
|
|
|
|
batch = create_user_vma(ce->vm, SZ_64K);
|
|
|
|
if (IS_ERR(batch))
|
|
|
|
return batch;
|
|
|
|
|
|
|
|
cs = i915_gem_object_pin_map(batch->obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_vma_put(batch);
|
|
|
|
return ERR_CAST(cs);
|
|
|
|
}
|
|
|
|
|
2020-05-05 02:07:45 +08:00
|
|
|
defaults = shmem_pin_map(ce->engine->default_state);
|
|
|
|
if (!defaults) {
|
|
|
|
i915_gem_object_unpin_map(batch->obj);
|
|
|
|
i915_vma_put(batch);
|
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
}
|
|
|
|
|
2020-02-27 16:57:15 +08:00
|
|
|
x = 0;
|
|
|
|
dw = 0;
|
2020-05-05 02:07:45 +08:00
|
|
|
hw = defaults;
|
2020-04-24 02:23:52 +08:00
|
|
|
hw += LRC_STATE_OFFSET / sizeof(*hw);
|
2020-02-27 16:57:15 +08:00
|
|
|
do {
|
|
|
|
u32 len = hw[dw] & 0x7f;
|
|
|
|
|
|
|
|
if (hw[dw] == 0) {
|
|
|
|
dw++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((hw[dw] & GENMASK(31, 23)) != MI_INSTR(0x22, 0)) {
|
|
|
|
dw += len + 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
dw++;
|
|
|
|
len = (len + 1) / 2;
|
|
|
|
while (len--) {
|
|
|
|
*cs++ = MI_STORE_REGISTER_MEM_GEN8;
|
|
|
|
*cs++ = hw[dw];
|
|
|
|
*cs++ = lower_32_bits(scratch->node.start + x);
|
|
|
|
*cs++ = upper_32_bits(scratch->node.start + x);
|
|
|
|
|
|
|
|
dw += 2;
|
|
|
|
x += 4;
|
|
|
|
}
|
|
|
|
} while (dw < PAGE_SIZE / sizeof(u32) &&
|
|
|
|
(hw[dw] & ~BIT(0)) != MI_BATCH_BUFFER_END);
|
|
|
|
|
|
|
|
*cs++ = MI_BATCH_BUFFER_END;
|
|
|
|
|
2020-05-05 02:07:45 +08:00
|
|
|
shmem_unpin_map(ce->engine->default_state, defaults);
|
|
|
|
|
2020-02-27 16:57:15 +08:00
|
|
|
i915_gem_object_flush_map(batch->obj);
|
|
|
|
i915_gem_object_unpin_map(batch->obj);
|
|
|
|
|
|
|
|
return batch;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int move_to_active(struct i915_request *rq,
|
|
|
|
struct i915_vma *vma,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
i915_vma_lock(vma);
|
|
|
|
err = i915_request_await_object(rq, vma->obj, flags);
|
|
|
|
if (!err)
|
|
|
|
err = i915_vma_move_to_active(vma, rq, flags);
|
|
|
|
i915_vma_unlock(vma);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct i915_request *
|
|
|
|
record_registers(struct intel_context *ce,
|
|
|
|
struct i915_vma *before,
|
|
|
|
struct i915_vma *after,
|
|
|
|
u32 *sema)
|
|
|
|
{
|
|
|
|
struct i915_vma *b_before, *b_after;
|
|
|
|
struct i915_request *rq;
|
|
|
|
u32 *cs;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
b_before = store_context(ce, before);
|
|
|
|
if (IS_ERR(b_before))
|
|
|
|
return ERR_CAST(b_before);
|
|
|
|
|
|
|
|
b_after = store_context(ce, after);
|
|
|
|
if (IS_ERR(b_after)) {
|
|
|
|
rq = ERR_CAST(b_after);
|
|
|
|
goto err_before;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
goto err_after;
|
|
|
|
|
|
|
|
err = move_to_active(rq, before, EXEC_OBJECT_WRITE);
|
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
|
|
|
|
|
|
|
err = move_to_active(rq, b_before, 0);
|
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
|
|
|
|
|
|
|
err = move_to_active(rq, after, EXEC_OBJECT_WRITE);
|
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
|
|
|
|
|
|
|
err = move_to_active(rq, b_after, 0);
|
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 14);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto err_rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_DISABLE;
|
|
|
|
*cs++ = MI_BATCH_BUFFER_START_GEN8 | BIT(8);
|
|
|
|
*cs++ = lower_32_bits(b_before->node.start);
|
|
|
|
*cs++ = upper_32_bits(b_before->node.start);
|
|
|
|
|
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_ENABLE;
|
|
|
|
*cs++ = MI_SEMAPHORE_WAIT |
|
|
|
|
MI_SEMAPHORE_GLOBAL_GTT |
|
|
|
|
MI_SEMAPHORE_POLL |
|
|
|
|
MI_SEMAPHORE_SAD_NEQ_SDD;
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = i915_ggtt_offset(ce->engine->status_page.vma) +
|
|
|
|
offset_in_page(sema);
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = MI_NOOP;
|
|
|
|
|
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_DISABLE;
|
|
|
|
*cs++ = MI_BATCH_BUFFER_START_GEN8 | BIT(8);
|
|
|
|
*cs++ = lower_32_bits(b_after->node.start);
|
|
|
|
*cs++ = upper_32_bits(b_after->node.start);
|
|
|
|
|
|
|
|
intel_ring_advance(rq, cs);
|
|
|
|
|
|
|
|
WRITE_ONCE(*sema, 0);
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
err_after:
|
|
|
|
i915_vma_put(b_after);
|
|
|
|
err_before:
|
|
|
|
i915_vma_put(b_before);
|
|
|
|
return rq;
|
|
|
|
|
|
|
|
err_rq:
|
|
|
|
i915_request_add(rq);
|
|
|
|
rq = ERR_PTR(err);
|
|
|
|
goto err_after;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct i915_vma *load_context(struct intel_context *ce, u32 poison)
|
|
|
|
{
|
|
|
|
struct i915_vma *batch;
|
|
|
|
u32 dw, *cs, *hw;
|
2020-05-05 02:07:45 +08:00
|
|
|
u32 *defaults;
|
2020-02-27 16:57:15 +08:00
|
|
|
|
|
|
|
batch = create_user_vma(ce->vm, SZ_64K);
|
|
|
|
if (IS_ERR(batch))
|
|
|
|
return batch;
|
|
|
|
|
|
|
|
cs = i915_gem_object_pin_map(batch->obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
i915_vma_put(batch);
|
|
|
|
return ERR_CAST(cs);
|
|
|
|
}
|
|
|
|
|
2020-05-05 02:07:45 +08:00
|
|
|
defaults = shmem_pin_map(ce->engine->default_state);
|
|
|
|
if (!defaults) {
|
|
|
|
i915_gem_object_unpin_map(batch->obj);
|
|
|
|
i915_vma_put(batch);
|
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
}
|
|
|
|
|
2020-02-27 16:57:15 +08:00
|
|
|
dw = 0;
|
2020-05-05 02:07:45 +08:00
|
|
|
hw = defaults;
|
2020-04-24 02:23:52 +08:00
|
|
|
hw += LRC_STATE_OFFSET / sizeof(*hw);
|
2020-02-27 16:57:15 +08:00
|
|
|
do {
|
|
|
|
u32 len = hw[dw] & 0x7f;
|
|
|
|
|
|
|
|
if (hw[dw] == 0) {
|
|
|
|
dw++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((hw[dw] & GENMASK(31, 23)) != MI_INSTR(0x22, 0)) {
|
|
|
|
dw += len + 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
dw++;
|
|
|
|
len = (len + 1) / 2;
|
|
|
|
*cs++ = MI_LOAD_REGISTER_IMM(len);
|
|
|
|
while (len--) {
|
|
|
|
*cs++ = hw[dw];
|
|
|
|
*cs++ = poison;
|
|
|
|
dw += 2;
|
|
|
|
}
|
|
|
|
} while (dw < PAGE_SIZE / sizeof(u32) &&
|
|
|
|
(hw[dw] & ~BIT(0)) != MI_BATCH_BUFFER_END);
|
|
|
|
|
|
|
|
*cs++ = MI_BATCH_BUFFER_END;
|
|
|
|
|
2020-05-05 02:07:45 +08:00
|
|
|
shmem_unpin_map(ce->engine->default_state, defaults);
|
|
|
|
|
2020-02-27 16:57:15 +08:00
|
|
|
i915_gem_object_flush_map(batch->obj);
|
|
|
|
i915_gem_object_unpin_map(batch->obj);
|
|
|
|
|
|
|
|
return batch;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int poison_registers(struct intel_context *ce, u32 poison, u32 *sema)
|
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
struct i915_vma *batch;
|
|
|
|
u32 *cs;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
batch = load_context(ce, poison);
|
|
|
|
if (IS_ERR(batch))
|
|
|
|
return PTR_ERR(batch);
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_batch;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = move_to_active(rq, batch, 0);
|
|
|
|
if (err)
|
|
|
|
goto err_rq;
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 8);
|
|
|
|
if (IS_ERR(cs)) {
|
|
|
|
err = PTR_ERR(cs);
|
|
|
|
goto err_rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cs++ = MI_ARB_ON_OFF | MI_ARB_DISABLE;
|
|
|
|
*cs++ = MI_BATCH_BUFFER_START_GEN8 | BIT(8);
|
|
|
|
*cs++ = lower_32_bits(batch->node.start);
|
|
|
|
*cs++ = upper_32_bits(batch->node.start);
|
|
|
|
|
|
|
|
*cs++ = MI_STORE_DWORD_IMM_GEN4 | MI_USE_GGTT;
|
|
|
|
*cs++ = i915_ggtt_offset(ce->engine->status_page.vma) +
|
|
|
|
offset_in_page(sema);
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = 1;
|
|
|
|
|
|
|
|
intel_ring_advance(rq, cs);
|
|
|
|
|
|
|
|
rq->sched.attr.priority = I915_PRIORITY_BARRIER;
|
|
|
|
err_rq:
|
|
|
|
i915_request_add(rq);
|
|
|
|
err_batch:
|
|
|
|
i915_vma_put(batch);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool is_moving(u32 a, u32 b)
|
|
|
|
{
|
|
|
|
return a != b;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int compare_isolation(struct intel_engine_cs *engine,
|
|
|
|
struct i915_vma *ref[2],
|
|
|
|
struct i915_vma *result[2],
|
|
|
|
struct intel_context *ce,
|
|
|
|
u32 poison)
|
|
|
|
{
|
|
|
|
u32 x, dw, *hw, *lrc;
|
|
|
|
u32 *A[2], *B[2];
|
2020-05-05 02:07:45 +08:00
|
|
|
u32 *defaults;
|
2020-02-27 16:57:15 +08:00
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
A[0] = i915_gem_object_pin_map(ref[0]->obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(A[0]))
|
|
|
|
return PTR_ERR(A[0]);
|
|
|
|
|
|
|
|
A[1] = i915_gem_object_pin_map(ref[1]->obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(A[1])) {
|
|
|
|
err = PTR_ERR(A[1]);
|
|
|
|
goto err_A0;
|
|
|
|
}
|
|
|
|
|
|
|
|
B[0] = i915_gem_object_pin_map(result[0]->obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(B[0])) {
|
|
|
|
err = PTR_ERR(B[0]);
|
|
|
|
goto err_A1;
|
|
|
|
}
|
|
|
|
|
|
|
|
B[1] = i915_gem_object_pin_map(result[1]->obj, I915_MAP_WC);
|
|
|
|
if (IS_ERR(B[1])) {
|
|
|
|
err = PTR_ERR(B[1]);
|
|
|
|
goto err_B0;
|
|
|
|
}
|
|
|
|
|
|
|
|
lrc = i915_gem_object_pin_map(ce->state->obj,
|
|
|
|
i915_coherent_map_type(engine->i915));
|
|
|
|
if (IS_ERR(lrc)) {
|
|
|
|
err = PTR_ERR(lrc);
|
|
|
|
goto err_B1;
|
|
|
|
}
|
2020-04-24 02:23:52 +08:00
|
|
|
lrc += LRC_STATE_OFFSET / sizeof(*hw);
|
2020-02-27 16:57:15 +08:00
|
|
|
|
2020-05-05 02:07:45 +08:00
|
|
|
defaults = shmem_pin_map(ce->engine->default_state);
|
|
|
|
if (!defaults) {
|
|
|
|
err = -ENOMEM;
|
|
|
|
goto err_lrc;
|
|
|
|
}
|
|
|
|
|
2020-02-27 16:57:15 +08:00
|
|
|
x = 0;
|
|
|
|
dw = 0;
|
2020-05-05 02:07:45 +08:00
|
|
|
hw = defaults;
|
2020-04-24 02:23:52 +08:00
|
|
|
hw += LRC_STATE_OFFSET / sizeof(*hw);
|
2020-02-27 16:57:15 +08:00
|
|
|
do {
|
|
|
|
u32 len = hw[dw] & 0x7f;
|
|
|
|
|
|
|
|
if (hw[dw] == 0) {
|
|
|
|
dw++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((hw[dw] & GENMASK(31, 23)) != MI_INSTR(0x22, 0)) {
|
|
|
|
dw += len + 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
dw++;
|
|
|
|
len = (len + 1) / 2;
|
|
|
|
while (len--) {
|
|
|
|
if (!is_moving(A[0][x], A[1][x]) &&
|
|
|
|
(A[0][x] != B[0][x] || A[1][x] != B[1][x])) {
|
|
|
|
switch (hw[dw] & 4095) {
|
|
|
|
case 0x30: /* RING_HEAD */
|
|
|
|
case 0x34: /* RING_TAIL */
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
pr_err("%s[%d]: Mismatch for register %4x, default %08x, reference %08x, result (%08x, %08x), poison %08x, context %08x\n",
|
|
|
|
engine->name, dw,
|
|
|
|
hw[dw], hw[dw + 1],
|
|
|
|
A[0][x], B[0][x], B[1][x],
|
|
|
|
poison, lrc[dw + 1]);
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dw += 2;
|
|
|
|
x++;
|
|
|
|
}
|
|
|
|
} while (dw < PAGE_SIZE / sizeof(u32) &&
|
|
|
|
(hw[dw] & ~BIT(0)) != MI_BATCH_BUFFER_END);
|
|
|
|
|
2020-05-05 02:07:45 +08:00
|
|
|
shmem_unpin_map(ce->engine->default_state, defaults);
|
|
|
|
err_lrc:
|
2020-02-27 16:57:15 +08:00
|
|
|
i915_gem_object_unpin_map(ce->state->obj);
|
|
|
|
err_B1:
|
|
|
|
i915_gem_object_unpin_map(result[1]->obj);
|
|
|
|
err_B0:
|
|
|
|
i915_gem_object_unpin_map(result[0]->obj);
|
|
|
|
err_A1:
|
|
|
|
i915_gem_object_unpin_map(ref[1]->obj);
|
|
|
|
err_A0:
|
|
|
|
i915_gem_object_unpin_map(ref[0]->obj);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __lrc_isolation(struct intel_engine_cs *engine, u32 poison)
|
|
|
|
{
|
|
|
|
u32 *sema = memset32(engine->status_page.addr + 1000, 0, 1);
|
|
|
|
struct i915_vma *ref[2], *result[2];
|
|
|
|
struct intel_context *A, *B;
|
|
|
|
struct i915_request *rq;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
A = intel_context_create(engine);
|
|
|
|
if (IS_ERR(A))
|
|
|
|
return PTR_ERR(A);
|
|
|
|
|
|
|
|
B = intel_context_create(engine);
|
|
|
|
if (IS_ERR(B)) {
|
|
|
|
err = PTR_ERR(B);
|
|
|
|
goto err_A;
|
|
|
|
}
|
|
|
|
|
|
|
|
ref[0] = create_user_vma(A->vm, SZ_64K);
|
|
|
|
if (IS_ERR(ref[0])) {
|
|
|
|
err = PTR_ERR(ref[0]);
|
|
|
|
goto err_B;
|
|
|
|
}
|
|
|
|
|
|
|
|
ref[1] = create_user_vma(A->vm, SZ_64K);
|
|
|
|
if (IS_ERR(ref[1])) {
|
|
|
|
err = PTR_ERR(ref[1]);
|
|
|
|
goto err_ref0;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = record_registers(A, ref[0], ref[1], sema);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_ref1;
|
|
|
|
}
|
|
|
|
|
|
|
|
WRITE_ONCE(*sema, 1);
|
|
|
|
wmb();
|
|
|
|
|
|
|
|
if (i915_request_wait(rq, 0, HZ / 2) < 0) {
|
|
|
|
i915_request_put(rq);
|
|
|
|
err = -ETIME;
|
|
|
|
goto err_ref1;
|
|
|
|
}
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
result[0] = create_user_vma(A->vm, SZ_64K);
|
|
|
|
if (IS_ERR(result[0])) {
|
|
|
|
err = PTR_ERR(result[0]);
|
|
|
|
goto err_ref1;
|
|
|
|
}
|
|
|
|
|
|
|
|
result[1] = create_user_vma(A->vm, SZ_64K);
|
|
|
|
if (IS_ERR(result[1])) {
|
|
|
|
err = PTR_ERR(result[1]);
|
|
|
|
goto err_result0;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq = record_registers(A, result[0], result[1], sema);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_result1;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = poison_registers(B, poison, sema);
|
|
|
|
if (err) {
|
|
|
|
WRITE_ONCE(*sema, -1);
|
|
|
|
i915_request_put(rq);
|
|
|
|
goto err_result1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i915_request_wait(rq, 0, HZ / 2) < 0) {
|
|
|
|
i915_request_put(rq);
|
|
|
|
err = -ETIME;
|
|
|
|
goto err_result1;
|
|
|
|
}
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
err = compare_isolation(engine, ref, result, A, poison);
|
|
|
|
|
|
|
|
err_result1:
|
|
|
|
i915_vma_put(result[1]);
|
|
|
|
err_result0:
|
|
|
|
i915_vma_put(result[0]);
|
|
|
|
err_ref1:
|
|
|
|
i915_vma_put(ref[1]);
|
|
|
|
err_ref0:
|
|
|
|
i915_vma_put(ref[0]);
|
|
|
|
err_B:
|
|
|
|
intel_context_put(B);
|
|
|
|
err_A:
|
|
|
|
intel_context_put(A);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool skip_isolation(const struct intel_engine_cs *engine)
|
|
|
|
{
|
|
|
|
if (engine->class == COPY_ENGINE_CLASS && INTEL_GEN(engine->i915) == 9)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (engine->class == RENDER_CLASS && INTEL_GEN(engine->i915) == 11)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_lrc_isolation(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
const u32 poison[] = {
|
|
|
|
STACK_MAGIC,
|
|
|
|
0x3a3a3a3a,
|
2020-03-13 18:28:12 +08:00
|
|
|
0x5c5c5c5c,
|
2020-02-27 16:57:15 +08:00
|
|
|
0xffffffff,
|
2020-03-13 18:28:12 +08:00
|
|
|
0xffff0000,
|
2020-02-27 16:57:15 +08:00
|
|
|
};
|
2020-03-31 21:54:03 +08:00
|
|
|
int err = 0;
|
2020-02-27 16:57:15 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Our goal is try and verify that per-context state cannot be
|
|
|
|
* tampered with by another non-privileged client.
|
|
|
|
*
|
|
|
|
* We take the list of context registers from the LRI in the default
|
|
|
|
* context image and attempt to modify that list from a remote context.
|
|
|
|
*/
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Just don't even ask */
|
|
|
|
if (!IS_ENABLED(CONFIG_DRM_I915_SELFTEST_BROKEN) &&
|
|
|
|
skip_isolation(engine))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
intel_engine_pm_get(engine);
|
2020-05-05 02:07:45 +08:00
|
|
|
for (i = 0; i < ARRAY_SIZE(poison); i++) {
|
|
|
|
int result;
|
2020-03-13 18:28:12 +08:00
|
|
|
|
2020-05-05 02:07:45 +08:00
|
|
|
result = __lrc_isolation(engine, poison[i]);
|
|
|
|
if (result && !err)
|
|
|
|
err = result;
|
2020-03-31 21:54:03 +08:00
|
|
|
|
2020-05-05 02:07:45 +08:00
|
|
|
result = __lrc_isolation(engine, ~poison[i]);
|
|
|
|
if (result && !err)
|
|
|
|
err = result;
|
2020-02-27 16:57:15 +08:00
|
|
|
}
|
|
|
|
intel_engine_pm_put(engine);
|
2020-03-31 21:54:03 +08:00
|
|
|
if (igt_flush_test(gt->i915)) {
|
2020-02-27 16:57:15 +08:00
|
|
|
err = -EIO;
|
2020-03-31 21:54:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-02-27 16:57:15 +08:00
|
|
|
}
|
|
|
|
|
2020-03-31 21:54:03 +08:00
|
|
|
return err;
|
2020-02-27 16:57:15 +08:00
|
|
|
}
|
|
|
|
|
2020-04-25 05:48:40 +08:00
|
|
|
static int indirect_ctx_submit_req(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq))
|
|
|
|
return PTR_ERR(rq);
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
if (i915_request_wait(rq, 0, HZ / 5) < 0)
|
|
|
|
err = -ETIME;
|
|
|
|
|
|
|
|
i915_request_put(rq);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define CTX_BB_CANARY_OFFSET (3 * 1024)
|
|
|
|
#define CTX_BB_CANARY_INDEX (CTX_BB_CANARY_OFFSET / sizeof(u32))
|
|
|
|
|
|
|
|
static u32 *
|
|
|
|
emit_indirect_ctx_bb_canary(const struct intel_context *ce, u32 *cs)
|
|
|
|
{
|
|
|
|
*cs++ = MI_STORE_REGISTER_MEM_GEN8 |
|
|
|
|
MI_SRM_LRM_GLOBAL_GTT |
|
|
|
|
MI_LRI_LRM_CS_MMIO;
|
|
|
|
*cs++ = i915_mmio_reg_offset(RING_START(0));
|
|
|
|
*cs++ = i915_ggtt_offset(ce->state) +
|
|
|
|
context_wa_bb_offset(ce) +
|
|
|
|
CTX_BB_CANARY_OFFSET;
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
return cs;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
indirect_ctx_bb_setup(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
u32 *cs = context_indirect_bb(ce);
|
|
|
|
|
|
|
|
cs[CTX_BB_CANARY_INDEX] = 0xdeadf00d;
|
|
|
|
|
|
|
|
setup_indirect_ctx_bb(ce, ce->engine, emit_indirect_ctx_bb_canary);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool check_ring_start(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
const u32 * const ctx_bb = (void *)(ce->lrc_reg_state) -
|
|
|
|
LRC_STATE_OFFSET + context_wa_bb_offset(ce);
|
|
|
|
|
|
|
|
if (ctx_bb[CTX_BB_CANARY_INDEX] == ce->lrc_reg_state[CTX_RING_START])
|
|
|
|
return true;
|
|
|
|
|
|
|
|
pr_err("ring start mismatch: canary 0x%08x vs state 0x%08x\n",
|
|
|
|
ctx_bb[CTX_BB_CANARY_INDEX],
|
|
|
|
ce->lrc_reg_state[CTX_RING_START]);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int indirect_ctx_bb_check(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = indirect_ctx_submit_req(ce);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
if (!check_ring_start(ce))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __live_lrc_indirect_ctx_bb(struct intel_engine_cs *engine)
|
|
|
|
{
|
|
|
|
struct intel_context *a, *b;
|
2020-04-29 21:24:25 +08:00
|
|
|
int err;
|
2020-04-25 05:48:40 +08:00
|
|
|
|
|
|
|
a = intel_context_create(engine);
|
2020-04-29 21:24:25 +08:00
|
|
|
if (IS_ERR(a))
|
|
|
|
return PTR_ERR(a);
|
2020-04-25 05:48:40 +08:00
|
|
|
err = intel_context_pin(a);
|
|
|
|
if (err)
|
2020-04-29 21:24:25 +08:00
|
|
|
goto put_a;
|
2020-04-25 05:48:40 +08:00
|
|
|
|
2020-04-29 21:24:25 +08:00
|
|
|
b = intel_context_create(engine);
|
|
|
|
if (IS_ERR(b)) {
|
|
|
|
err = PTR_ERR(b);
|
|
|
|
goto unpin_a;
|
2020-04-25 05:48:40 +08:00
|
|
|
}
|
2020-04-29 21:24:25 +08:00
|
|
|
err = intel_context_pin(b);
|
|
|
|
if (err)
|
|
|
|
goto put_b;
|
2020-04-25 05:48:40 +08:00
|
|
|
|
|
|
|
/* We use the already reserved extra page in context state */
|
|
|
|
if (!a->wa_bb_page) {
|
|
|
|
GEM_BUG_ON(b->wa_bb_page);
|
|
|
|
GEM_BUG_ON(INTEL_GEN(engine->i915) == 12);
|
2020-04-29 21:24:25 +08:00
|
|
|
goto unpin_b;
|
2020-04-25 05:48:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* In order to test that our per context bb is truly per context,
|
|
|
|
* and executes at the intended spot on context restoring process,
|
|
|
|
* make the batch store the ring start value to memory.
|
|
|
|
* As ring start is restored apriori of starting the indirect ctx bb and
|
|
|
|
* as it will be different for each context, it fits to this purpose.
|
|
|
|
*/
|
|
|
|
indirect_ctx_bb_setup(a);
|
|
|
|
indirect_ctx_bb_setup(b);
|
|
|
|
|
|
|
|
err = indirect_ctx_bb_check(a);
|
|
|
|
if (err)
|
2020-04-29 21:24:25 +08:00
|
|
|
goto unpin_b;
|
2020-04-25 05:48:40 +08:00
|
|
|
|
|
|
|
err = indirect_ctx_bb_check(b);
|
2020-04-29 21:24:25 +08:00
|
|
|
|
|
|
|
unpin_b:
|
2020-04-25 05:48:40 +08:00
|
|
|
intel_context_unpin(b);
|
2020-04-29 21:24:25 +08:00
|
|
|
put_b:
|
2020-04-25 05:48:40 +08:00
|
|
|
intel_context_put(b);
|
2020-04-29 21:24:25 +08:00
|
|
|
unpin_a:
|
2020-04-25 05:48:40 +08:00
|
|
|
intel_context_unpin(a);
|
2020-04-29 21:24:25 +08:00
|
|
|
put_a:
|
2020-04-25 05:48:40 +08:00
|
|
|
intel_context_put(a);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_lrc_indirect_ctx_bb(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
intel_engine_pm_get(engine);
|
|
|
|
err = __live_lrc_indirect_ctx_bb(engine);
|
|
|
|
intel_engine_pm_put(engine);
|
|
|
|
|
|
|
|
if (igt_flush_test(gt->i915))
|
|
|
|
err = -EIO;
|
|
|
|
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-02-27 16:57:16 +08:00
|
|
|
static void garbage_reset(struct intel_engine_cs *engine,
|
|
|
|
struct i915_request *rq)
|
|
|
|
{
|
|
|
|
const unsigned int bit = I915_RESET_ENGINE + engine->id;
|
|
|
|
unsigned long *lock = &engine->gt->reset.flags;
|
|
|
|
|
|
|
|
if (test_and_set_bit(bit, lock))
|
|
|
|
return;
|
|
|
|
|
|
|
|
tasklet_disable(&engine->execlists.tasklet);
|
|
|
|
|
|
|
|
if (!rq->fence.error)
|
|
|
|
intel_engine_reset(engine, NULL);
|
|
|
|
|
|
|
|
tasklet_enable(&engine->execlists.tasklet);
|
|
|
|
clear_and_wake_up_bit(bit, lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct i915_request *garbage(struct intel_context *ce,
|
|
|
|
struct rnd_state *prng)
|
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = intel_context_pin(ce);
|
|
|
|
if (err)
|
|
|
|
return ERR_PTR(err);
|
|
|
|
|
|
|
|
prandom_bytes_state(prng,
|
|
|
|
ce->lrc_reg_state,
|
|
|
|
ce->engine->context_size -
|
2020-04-24 02:23:52 +08:00
|
|
|
LRC_STATE_OFFSET);
|
2020-02-27 16:57:16 +08:00
|
|
|
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_unpin;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
return rq;
|
|
|
|
|
|
|
|
err_unpin:
|
|
|
|
intel_context_unpin(ce);
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __lrc_garbage(struct intel_engine_cs *engine, struct rnd_state *prng)
|
|
|
|
{
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *hang;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce))
|
|
|
|
return PTR_ERR(ce);
|
|
|
|
|
|
|
|
hang = garbage(ce, prng);
|
|
|
|
if (IS_ERR(hang)) {
|
|
|
|
err = PTR_ERR(hang);
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wait_for_submit(engine, hang, HZ / 2)) {
|
|
|
|
i915_request_put(hang);
|
|
|
|
err = -ETIME;
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
intel_context_set_banned(ce);
|
|
|
|
garbage_reset(engine, hang);
|
|
|
|
|
|
|
|
intel_engine_flush_submission(engine);
|
|
|
|
if (!hang->fence.error) {
|
|
|
|
i915_request_put(hang);
|
|
|
|
pr_err("%s: corrupted context was not reset\n",
|
|
|
|
engine->name);
|
|
|
|
err = -EINVAL;
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i915_request_wait(hang, 0, HZ / 2) < 0) {
|
|
|
|
pr_err("%s: corrupted context did not recover\n",
|
|
|
|
engine->name);
|
|
|
|
i915_request_put(hang);
|
|
|
|
err = -EIO;
|
|
|
|
goto err_ce;
|
|
|
|
}
|
|
|
|
i915_request_put(hang);
|
|
|
|
|
|
|
|
err_ce:
|
|
|
|
intel_context_put(ce);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_lrc_garbage(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Verify that we can recover if one context state is completely
|
|
|
|
* corrupted.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!IS_ENABLED(CONFIG_DRM_I915_SELFTEST_BROKEN))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
I915_RND_STATE(prng);
|
|
|
|
int err = 0, i;
|
|
|
|
|
|
|
|
if (!intel_has_reset_engine(engine->gt))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
intel_engine_pm_get(engine);
|
|
|
|
for (i = 0; i < 3; i++) {
|
|
|
|
err = __lrc_garbage(engine, &prng);
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
intel_engine_pm_put(engine);
|
|
|
|
|
|
|
|
if (igt_flush_test(gt->i915))
|
|
|
|
err = -EIO;
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-02-16 21:36:20 +08:00
|
|
|
static int __live_pphwsp_runtime(struct intel_engine_cs *engine)
|
|
|
|
{
|
|
|
|
struct intel_context *ce;
|
|
|
|
struct i915_request *rq;
|
|
|
|
IGT_TIMEOUT(end_time);
|
|
|
|
int err;
|
|
|
|
|
|
|
|
ce = intel_context_create(engine);
|
|
|
|
if (IS_ERR(ce))
|
|
|
|
return PTR_ERR(ce);
|
|
|
|
|
|
|
|
ce->runtime.num_underflow = 0;
|
|
|
|
ce->runtime.max_underflow = 0;
|
|
|
|
|
|
|
|
do {
|
|
|
|
unsigned int loop = 1024;
|
|
|
|
|
|
|
|
while (loop) {
|
|
|
|
rq = intel_context_create_request(ce);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto err_rq;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (--loop == 0)
|
|
|
|
i915_request_get(rq);
|
|
|
|
|
|
|
|
i915_request_add(rq);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (__igt_timeout(end_time, NULL))
|
|
|
|
break;
|
|
|
|
|
|
|
|
i915_request_put(rq);
|
|
|
|
} while (1);
|
|
|
|
|
|
|
|
err = i915_request_wait(rq, 0, HZ / 5);
|
|
|
|
if (err < 0) {
|
|
|
|
pr_err("%s: request not completed!\n", engine->name);
|
|
|
|
goto err_wait;
|
|
|
|
}
|
|
|
|
|
|
|
|
igt_flush_test(engine->i915);
|
|
|
|
|
|
|
|
pr_info("%s: pphwsp runtime %lluns, average %lluns\n",
|
|
|
|
engine->name,
|
|
|
|
intel_context_get_total_runtime_ns(ce),
|
|
|
|
intel_context_get_avg_runtime_ns(ce));
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
if (ce->runtime.num_underflow) {
|
|
|
|
pr_err("%s: pphwsp underflow %u time(s), max %u cycles!\n",
|
|
|
|
engine->name,
|
|
|
|
ce->runtime.num_underflow,
|
|
|
|
ce->runtime.max_underflow);
|
|
|
|
GEM_TRACE_DUMP();
|
|
|
|
err = -EOVERFLOW;
|
|
|
|
}
|
|
|
|
|
|
|
|
err_wait:
|
|
|
|
i915_request_put(rq);
|
|
|
|
err_rq:
|
|
|
|
intel_context_put(ce);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_pphwsp_runtime(void *arg)
|
|
|
|
{
|
|
|
|
struct intel_gt *gt = arg;
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that cumulative context runtime as stored in the pphwsp[16]
|
|
|
|
* is monotonic.
|
|
|
|
*/
|
|
|
|
|
|
|
|
for_each_engine(engine, gt, id) {
|
|
|
|
err = __live_pphwsp_runtime(engine);
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (igt_flush_test(gt->i915))
|
|
|
|
err = -EIO;
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-09-24 22:59:50 +08:00
|
|
|
int intel_lrc_live_selftests(struct drm_i915_private *i915)
|
|
|
|
{
|
|
|
|
static const struct i915_subtest tests[] = {
|
|
|
|
SUBTEST(live_lrc_layout),
|
2019-10-26 16:22:20 +08:00
|
|
|
SUBTEST(live_lrc_fixed),
|
2019-10-14 17:07:48 +08:00
|
|
|
SUBTEST(live_lrc_state),
|
2020-02-19 20:34:18 +08:00
|
|
|
SUBTEST(live_lrc_gpr),
|
2020-02-27 16:57:15 +08:00
|
|
|
SUBTEST(live_lrc_isolation),
|
2020-02-19 19:20:03 +08:00
|
|
|
SUBTEST(live_lrc_timestamp),
|
2020-02-27 16:57:16 +08:00
|
|
|
SUBTEST(live_lrc_garbage),
|
2020-02-16 21:36:20 +08:00
|
|
|
SUBTEST(live_pphwsp_runtime),
|
2020-04-25 05:48:40 +08:00
|
|
|
SUBTEST(live_lrc_indirect_ctx_bb),
|
2019-09-24 22:59:50 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
if (!HAS_LOGICAL_RING_CONTEXTS(i915))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return intel_gt_live_subtests(tests, &i915->gt);
|
|
|
|
}
|