mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2024-12-28 11:18:45 +07:00
drm/i915/selftests: Add a simple rollover test for the kernel context
Exercise the seqno wrap paths on the kernel context to provide a small amount of sanity checking and ensure that they are visible to lockdep. Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk> Cc: Maarten Lankhorst <maarten.lankhorst@linux.intel.com> Reviewed-by: Matthew Auld <matthew.auld@intel.com> Link: https://patchwork.freedesktop.org/patch/msgid/20200204094102.876636-1-chris@chris-wilson.co.uk
This commit is contained in:
parent
1a2466fe13
commit
bb5e4397a6
@ -6,6 +6,8 @@
|
|||||||
|
|
||||||
#include <linux/prime_numbers.h>
|
#include <linux/prime_numbers.h>
|
||||||
|
|
||||||
|
#include "intel_context.h"
|
||||||
|
#include "intel_engine_heartbeat.h"
|
||||||
#include "intel_engine_pm.h"
|
#include "intel_engine_pm.h"
|
||||||
#include "intel_gt.h"
|
#include "intel_gt.h"
|
||||||
#include "intel_gt_requests.h"
|
#include "intel_gt_requests.h"
|
||||||
@ -750,6 +752,189 @@ static int live_hwsp_wrap(void *arg)
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void engine_heartbeat_disable(struct intel_engine_cs *engine,
|
||||||
|
unsigned long *saved)
|
||||||
|
{
|
||||||
|
*saved = engine->props.heartbeat_interval_ms;
|
||||||
|
engine->props.heartbeat_interval_ms = 0;
|
||||||
|
|
||||||
|
intel_engine_pm_get(engine);
|
||||||
|
intel_engine_park_heartbeat(engine);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void engine_heartbeat_enable(struct intel_engine_cs *engine,
|
||||||
|
unsigned long saved)
|
||||||
|
{
|
||||||
|
intel_engine_pm_put(engine);
|
||||||
|
|
||||||
|
engine->props.heartbeat_interval_ms = saved;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int live_hwsp_rollover_kernel(void *arg)
|
||||||
|
{
|
||||||
|
struct intel_gt *gt = arg;
|
||||||
|
struct intel_engine_cs *engine;
|
||||||
|
enum intel_engine_id id;
|
||||||
|
int err = 0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Run the host for long enough, and even the kernel context will
|
||||||
|
* see a seqno rollover.
|
||||||
|
*/
|
||||||
|
|
||||||
|
for_each_engine(engine, gt, id) {
|
||||||
|
struct intel_context *ce = engine->kernel_context;
|
||||||
|
struct intel_timeline *tl = ce->timeline;
|
||||||
|
struct i915_request *rq[3] = {};
|
||||||
|
unsigned long heartbeat;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
engine_heartbeat_disable(engine, &heartbeat);
|
||||||
|
if (intel_gt_wait_for_idle(gt, HZ / 2)) {
|
||||||
|
err = -EIO;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
GEM_BUG_ON(i915_active_fence_isset(&tl->last_request));
|
||||||
|
tl->seqno = 0;
|
||||||
|
timeline_rollback(tl);
|
||||||
|
timeline_rollback(tl);
|
||||||
|
WRITE_ONCE(*(u32 *)tl->hwsp_seqno, tl->seqno);
|
||||||
|
|
||||||
|
for (i = 0; i < ARRAY_SIZE(rq); i++) {
|
||||||
|
struct i915_request *this;
|
||||||
|
|
||||||
|
this = i915_request_create(ce);
|
||||||
|
if (IS_ERR(this)) {
|
||||||
|
err = PTR_ERR(this);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
pr_debug("%s: create fence.seqnp:%d\n",
|
||||||
|
engine->name,
|
||||||
|
lower_32_bits(this->fence.seqno));
|
||||||
|
|
||||||
|
GEM_BUG_ON(rcu_access_pointer(this->timeline) != tl);
|
||||||
|
|
||||||
|
rq[i] = i915_request_get(this);
|
||||||
|
i915_request_add(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We expected a wrap! */
|
||||||
|
GEM_BUG_ON(rq[2]->fence.seqno > rq[0]->fence.seqno);
|
||||||
|
|
||||||
|
if (i915_request_wait(rq[2], 0, HZ / 5) < 0) {
|
||||||
|
pr_err("Wait for timeline wrap timed out!\n");
|
||||||
|
err = -EIO;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < ARRAY_SIZE(rq); i++) {
|
||||||
|
if (!i915_request_completed(rq[i])) {
|
||||||
|
pr_err("Pre-wrap request not completed!\n");
|
||||||
|
err = -EINVAL;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
|
for (i = 0; i < ARRAY_SIZE(rq); i++)
|
||||||
|
i915_request_put(rq[i]);
|
||||||
|
engine_heartbeat_enable(engine, heartbeat);
|
||||||
|
if (err)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (igt_flush_test(gt->i915))
|
||||||
|
err = -EIO;
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int live_hwsp_rollover_user(void *arg)
|
||||||
|
{
|
||||||
|
struct intel_gt *gt = arg;
|
||||||
|
struct intel_engine_cs *engine;
|
||||||
|
enum intel_engine_id id;
|
||||||
|
int err = 0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Simulate a long running user context, and force the seqno wrap
|
||||||
|
* on the user's timeline.
|
||||||
|
*/
|
||||||
|
|
||||||
|
for_each_engine(engine, gt, id) {
|
||||||
|
struct i915_request *rq[3] = {};
|
||||||
|
struct intel_timeline *tl;
|
||||||
|
struct intel_context *ce;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
ce = intel_context_create(engine);
|
||||||
|
if (IS_ERR(ce))
|
||||||
|
return PTR_ERR(ce);
|
||||||
|
|
||||||
|
err = intel_context_alloc_state(ce);
|
||||||
|
if (err)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
tl = ce->timeline;
|
||||||
|
if (!tl->has_initial_breadcrumb || !tl->hwsp_cacheline)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
timeline_rollback(tl);
|
||||||
|
timeline_rollback(tl);
|
||||||
|
WRITE_ONCE(*(u32 *)tl->hwsp_seqno, tl->seqno);
|
||||||
|
|
||||||
|
for (i = 0; i < ARRAY_SIZE(rq); i++) {
|
||||||
|
struct i915_request *this;
|
||||||
|
|
||||||
|
this = intel_context_create_request(ce);
|
||||||
|
if (IS_ERR(this)) {
|
||||||
|
err = PTR_ERR(this);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
pr_debug("%s: create fence.seqnp:%d\n",
|
||||||
|
engine->name,
|
||||||
|
lower_32_bits(this->fence.seqno));
|
||||||
|
|
||||||
|
GEM_BUG_ON(rcu_access_pointer(this->timeline) != tl);
|
||||||
|
|
||||||
|
rq[i] = i915_request_get(this);
|
||||||
|
i915_request_add(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We expected a wrap! */
|
||||||
|
GEM_BUG_ON(rq[2]->fence.seqno > rq[0]->fence.seqno);
|
||||||
|
|
||||||
|
if (i915_request_wait(rq[2], 0, HZ / 5) < 0) {
|
||||||
|
pr_err("Wait for timeline wrap timed out!\n");
|
||||||
|
err = -EIO;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < ARRAY_SIZE(rq); i++) {
|
||||||
|
if (!i915_request_completed(rq[i])) {
|
||||||
|
pr_err("Pre-wrap request not completed!\n");
|
||||||
|
err = -EINVAL;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
|
for (i = 0; i < ARRAY_SIZE(rq); i++)
|
||||||
|
i915_request_put(rq[i]);
|
||||||
|
intel_context_put(ce);
|
||||||
|
if (err)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (igt_flush_test(gt->i915))
|
||||||
|
err = -EIO;
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
static int live_hwsp_recycle(void *arg)
|
static int live_hwsp_recycle(void *arg)
|
||||||
{
|
{
|
||||||
struct intel_gt *gt = arg;
|
struct intel_gt *gt = arg;
|
||||||
@ -827,6 +1012,8 @@ int intel_timeline_live_selftests(struct drm_i915_private *i915)
|
|||||||
SUBTEST(live_hwsp_engine),
|
SUBTEST(live_hwsp_engine),
|
||||||
SUBTEST(live_hwsp_alternate),
|
SUBTEST(live_hwsp_alternate),
|
||||||
SUBTEST(live_hwsp_wrap),
|
SUBTEST(live_hwsp_wrap),
|
||||||
|
SUBTEST(live_hwsp_rollover_kernel),
|
||||||
|
SUBTEST(live_hwsp_rollover_user),
|
||||||
};
|
};
|
||||||
|
|
||||||
if (intel_gt_is_wedged(&i915->gt))
|
if (intel_gt_is_wedged(&i915->gt))
|
||||||
|
Loading…
Reference in New Issue
Block a user