2019-10-12 14:23:07 +07:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: MIT
|
|
|
|
*
|
|
|
|
* Copyright © 2019 Intel Corporation
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/kref.h>
|
|
|
|
|
|
|
|
#include "gem/i915_gem_pm.h"
|
|
|
|
#include "gt/intel_gt.h"
|
|
|
|
|
|
|
|
#include "i915_selftest.h"
|
|
|
|
|
|
|
|
#include "igt_flush_test.h"
|
|
|
|
#include "lib_sw_fence.h"
|
|
|
|
|
|
|
|
static struct i915_perf_stream *
|
|
|
|
test_stream(struct i915_perf *perf)
|
|
|
|
{
|
|
|
|
struct drm_i915_perf_open_param param = {};
|
|
|
|
struct perf_open_properties props = {
|
|
|
|
.engine = intel_engine_lookup_user(perf->i915,
|
|
|
|
I915_ENGINE_CLASS_RENDER,
|
|
|
|
0),
|
|
|
|
.sample_flags = SAMPLE_OA_REPORT,
|
2019-10-29 21:28:26 +07:00
|
|
|
.oa_format = IS_GEN(perf->i915, 12) ?
|
|
|
|
I915_OA_FORMAT_A32u40_A4u32_B8_C8 : I915_OA_FORMAT_C4_B8,
|
2019-10-12 14:23:07 +07:00
|
|
|
.metrics_set = 1,
|
|
|
|
};
|
|
|
|
struct i915_perf_stream *stream;
|
|
|
|
|
|
|
|
stream = kzalloc(sizeof(*stream), GFP_KERNEL);
|
|
|
|
if (!stream)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
stream->perf = perf;
|
|
|
|
|
|
|
|
mutex_lock(&perf->lock);
|
|
|
|
if (i915_oa_stream_init(stream, ¶m, &props)) {
|
|
|
|
kfree(stream);
|
|
|
|
stream = NULL;
|
|
|
|
}
|
|
|
|
mutex_unlock(&perf->lock);
|
|
|
|
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stream_destroy(struct i915_perf_stream *stream)
|
|
|
|
{
|
|
|
|
struct i915_perf *perf = stream->perf;
|
|
|
|
|
|
|
|
mutex_lock(&perf->lock);
|
|
|
|
i915_perf_destroy_locked(stream);
|
|
|
|
mutex_unlock(&perf->lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_sanitycheck(void *arg)
|
|
|
|
{
|
|
|
|
struct drm_i915_private *i915 = arg;
|
|
|
|
struct i915_perf_stream *stream;
|
|
|
|
|
|
|
|
/* Quick check we can create a perf stream */
|
|
|
|
|
|
|
|
stream = test_stream(&i915->perf);
|
|
|
|
if (!stream)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
stream_destroy(stream);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int write_timestamp(struct i915_request *rq, int slot)
|
|
|
|
{
|
|
|
|
u32 *cs;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
cs = intel_ring_begin(rq, 6);
|
|
|
|
if (IS_ERR(cs))
|
|
|
|
return PTR_ERR(cs);
|
|
|
|
|
|
|
|
len = 5;
|
|
|
|
if (INTEL_GEN(rq->i915) >= 8)
|
|
|
|
len++;
|
|
|
|
|
|
|
|
*cs++ = GFX_OP_PIPE_CONTROL(len);
|
|
|
|
*cs++ = PIPE_CONTROL_GLOBAL_GTT_IVB |
|
|
|
|
PIPE_CONTROL_STORE_DATA_INDEX |
|
|
|
|
PIPE_CONTROL_WRITE_TIMESTAMP;
|
|
|
|
*cs++ = slot * sizeof(u32);
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = 0;
|
|
|
|
*cs++ = 0;
|
|
|
|
|
|
|
|
intel_ring_advance(rq, cs);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ktime_t poll_status(struct i915_request *rq, int slot)
|
|
|
|
{
|
|
|
|
while (!intel_read_status_page(rq->engine, slot) &&
|
|
|
|
!i915_request_completed(rq))
|
|
|
|
cpu_relax();
|
|
|
|
|
|
|
|
return ktime_get();
|
|
|
|
}
|
|
|
|
|
|
|
|
static int live_noa_delay(void *arg)
|
|
|
|
{
|
|
|
|
struct drm_i915_private *i915 = arg;
|
|
|
|
struct i915_perf_stream *stream;
|
|
|
|
struct i915_request *rq;
|
|
|
|
ktime_t t0, t1;
|
|
|
|
u64 expected;
|
|
|
|
u32 delay;
|
|
|
|
int err;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Check that the GPU delays matches expectations */
|
|
|
|
|
|
|
|
stream = test_stream(&i915->perf);
|
|
|
|
if (!stream)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
expected = atomic64_read(&stream->perf->noa_programming_delay);
|
|
|
|
|
|
|
|
if (stream->engine->class != RENDER_CLASS) {
|
|
|
|
err = -ENODEV;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
intel_write_status_page(stream->engine, 0x100 + i, 0);
|
|
|
|
|
|
|
|
rq = i915_request_create(stream->engine->kernel_context);
|
|
|
|
if (IS_ERR(rq)) {
|
|
|
|
err = PTR_ERR(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rq->engine->emit_init_breadcrumb &&
|
|
|
|
i915_request_timeline(rq)->has_initial_breadcrumb) {
|
|
|
|
err = rq->engine->emit_init_breadcrumb(rq);
|
|
|
|
if (err) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = write_timestamp(rq, 0x100);
|
|
|
|
if (err) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = rq->engine->emit_bb_start(rq,
|
|
|
|
i915_ggtt_offset(stream->noa_wait), 0,
|
|
|
|
I915_DISPATCH_SECURE);
|
|
|
|
if (err) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = write_timestamp(rq, 0x102);
|
|
|
|
if (err) {
|
|
|
|
i915_request_add(rq);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_get(rq);
|
|
|
|
i915_request_add(rq);
|
|
|
|
|
|
|
|
preempt_disable();
|
|
|
|
t0 = poll_status(rq, 0x100);
|
|
|
|
t1 = poll_status(rq, 0x102);
|
|
|
|
preempt_enable();
|
|
|
|
|
|
|
|
pr_info("CPU delay: %lluns, expected %lluns\n",
|
|
|
|
ktime_sub(t1, t0), expected);
|
|
|
|
|
|
|
|
delay = intel_read_status_page(stream->engine, 0x102);
|
|
|
|
delay -= intel_read_status_page(stream->engine, 0x100);
|
|
|
|
delay = div_u64(mul_u32_u32(delay, 1000 * 1000),
|
|
|
|
RUNTIME_INFO(i915)->cs_timestamp_frequency_khz);
|
|
|
|
pr_info("GPU delay: %uns, expected %lluns\n",
|
|
|
|
delay, expected);
|
|
|
|
|
|
|
|
if (4 * delay < 3 * expected || 2 * delay > 3 * expected) {
|
|
|
|
pr_err("GPU delay [%uus] outside of expected threshold! [%lluus, %lluus]\n",
|
|
|
|
delay / 1000,
|
|
|
|
div_u64(3 * expected, 4000),
|
|
|
|
div_u64(3 * expected, 2000));
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
i915_request_put(rq);
|
|
|
|
out:
|
|
|
|
stream_destroy(stream);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
int i915_perf_live_selftests(struct drm_i915_private *i915)
|
|
|
|
{
|
|
|
|
static const struct i915_subtest tests[] = {
|
|
|
|
SUBTEST(live_sanitycheck),
|
|
|
|
SUBTEST(live_noa_delay),
|
|
|
|
};
|
|
|
|
struct i915_perf *perf = &i915->perf;
|
|
|
|
|
|
|
|
if (!perf->metrics_kobj || !perf->ops.enable_metric_set)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (intel_gt_is_wedged(&i915->gt))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return i915_subtests(tests, i915);
|
|
|
|
}
|