1 // SPDX-License-Identifier: GPL-2.0
3 * Copyright © 2018 Intel Corporation
6 #include <linux/sort.h>
8 #include "i915_selftest.h"
9 #include "intel_engine_regs.h"
10 #include "intel_gpu_commands.h"
11 #include "intel_gt_clock_utils.h"
12 #include "selftest_engine.h"
13 #include "selftest_engine_heartbeat.h"
14 #include "selftests/igt_atomic.h"
15 #include "selftests/igt_flush_test.h"
16 #include "selftests/igt_spinner.h"
20 static int cmp_u64(const void *A, const void *B)
22 const u64 *a = A, *b = B;
27 static u64 trifilter(u64 *a)
29 sort(a, COUNT, sizeof(*a), cmp_u64, NULL);
30 return (a[1] + 2 * a[2] + a[3]) >> 2;
33 static u32 *emit_wait(u32 *cs, u32 offset, int op, u32 value)
35 *cs++ = MI_SEMAPHORE_WAIT |
36 MI_SEMAPHORE_GLOBAL_GTT |
46 static u32 *emit_store(u32 *cs, u32 offset, u32 value)
48 *cs++ = MI_STORE_DWORD_IMM_GEN4 | MI_USE_GGTT;
56 static u32 *emit_srm(u32 *cs, i915_reg_t reg, u32 offset)
58 *cs++ = MI_STORE_REGISTER_MEM_GEN8 | MI_USE_GGTT;
59 *cs++ = i915_mmio_reg_offset(reg);
66 static void write_semaphore(u32 *x, u32 value)
68 WRITE_ONCE(*x, value);
72 static int __measure_timestamps(struct intel_context *ce,
73 u64 *dt, u64 *d_ring, u64 *d_ctx)
75 struct intel_engine_cs *engine = ce->engine;
76 u32 *sema = memset32(engine->status_page.addr + 1000, 0, 5);
77 u32 offset = i915_ggtt_offset(engine->status_page.vma);
78 struct i915_request *rq;
81 rq = intel_context_create_request(ce);
85 cs = intel_ring_begin(rq, 28);
91 /* Signal & wait for start */
92 cs = emit_store(cs, offset + 4008, 1);
93 cs = emit_wait(cs, offset + 4008, MI_SEMAPHORE_SAD_NEQ_SDD, 1);
95 cs = emit_srm(cs, RING_TIMESTAMP(engine->mmio_base), offset + 4000);
96 cs = emit_srm(cs, RING_CTX_TIMESTAMP(engine->mmio_base), offset + 4004);
99 cs = emit_wait(cs, offset + 4008, MI_SEMAPHORE_SAD_EQ_SDD, 1);
101 cs = emit_srm(cs, RING_TIMESTAMP(engine->mmio_base), offset + 4016);
102 cs = emit_srm(cs, RING_CTX_TIMESTAMP(engine->mmio_base), offset + 4012);
104 intel_ring_advance(rq, cs);
105 i915_request_get(rq);
106 i915_request_add(rq);
107 intel_engine_flush_submission(engine);
109 /* Wait for the request to start executing, that then waits for us */
110 while (READ_ONCE(sema[2]) == 0)
113 /* Run the request for a 100us, sampling timestamps before/after */
115 write_semaphore(&sema[2], 0);
116 while (READ_ONCE(sema[1]) == 0) /* wait for the gpu to catch up */
120 *dt = local_clock() - *dt;
121 write_semaphore(&sema[2], 1);
124 if (i915_request_wait(rq, 0, HZ / 2) < 0) {
125 i915_request_put(rq);
128 i915_request_put(rq);
130 pr_debug("%s CTX_TIMESTAMP: [%x, %x], RING_TIMESTAMP: [%x, %x]\n",
131 engine->name, sema[1], sema[3], sema[0], sema[4]);
133 *d_ctx = sema[3] - sema[1];
134 *d_ring = sema[4] - sema[0];
138 static int __live_engine_timestamps(struct intel_engine_cs *engine)
140 u64 s_ring[COUNT], s_ctx[COUNT], st[COUNT], d_ring, d_ctx, dt;
141 struct intel_context *ce;
144 ce = intel_context_create(engine);
148 for (i = 0; i < COUNT; i++) {
149 err = __measure_timestamps(ce, &st[i], &s_ring[i], &s_ctx[i]);
153 intel_context_put(ce);
158 d_ring = trifilter(s_ring);
159 d_ctx = trifilter(s_ctx);
161 pr_info("%s elapsed:%lldns, CTX_TIMESTAMP:%lldns, RING_TIMESTAMP:%lldns\n",
163 intel_gt_clock_interval_to_ns(engine->gt, d_ctx),
164 intel_gt_clock_interval_to_ns(engine->gt, d_ring));
166 d_ring = intel_gt_clock_interval_to_ns(engine->gt, d_ring);
167 if (3 * dt > 4 * d_ring || 4 * dt < 3 * d_ring) {
168 pr_err("%s Mismatch between ring timestamp and walltime!\n",
173 d_ring = trifilter(s_ring);
174 d_ctx = trifilter(s_ctx);
176 d_ctx *= engine->gt->clock_frequency;
177 if (GRAPHICS_VER(engine->i915) == 11)
178 d_ring *= 12500000; /* Fixed 80ns for GEN11 ctx timestamp? */
180 d_ring *= engine->gt->clock_frequency;
182 if (3 * d_ctx > 4 * d_ring || 4 * d_ctx < 3 * d_ring) {
183 pr_err("%s Mismatch between ring and context timestamps!\n",
191 static int live_engine_timestamps(void *arg)
193 struct intel_gt *gt = arg;
194 struct intel_engine_cs *engine;
195 enum intel_engine_id id;
198 * Check that CS_TIMESTAMP / CTX_TIMESTAMP are in sync, i.e. share
202 if (GRAPHICS_VER(gt->i915) < 8)
205 for_each_engine(engine, gt, id) {
208 st_engine_heartbeat_disable(engine);
209 err = __live_engine_timestamps(engine);
210 st_engine_heartbeat_enable(engine);
218 static int __spin_until_busier(struct intel_engine_cs *engine, ktime_t busyness)
220 ktime_t start, unused, dt;
222 if (!intel_engine_uses_guc(engine))
226 * In GuC mode of submission, the busyness stats may get updated after
227 * the batch starts running. Poll for a change in busyness and timeout
231 while (intel_engine_get_busy_time(engine, &unused) == busyness) {
232 dt = ktime_get() - start;
234 pr_err("active wait timed out %lld\n", dt);
235 ENGINE_TRACE(engine, "active wait time out %lld\n", dt);
243 static int live_engine_busy_stats(void *arg)
245 struct intel_gt *gt = arg;
246 struct intel_engine_cs *engine;
247 enum intel_engine_id id;
248 struct igt_spinner spin;
252 * Check that if an engine supports busy-stats, they tell the truth.
255 if (igt_spinner_init(&spin, gt))
258 GEM_BUG_ON(intel_gt_pm_is_awake(gt));
259 for_each_engine(engine, gt, id) {
260 struct i915_request *rq;
261 ktime_t busyness, dummy;
265 if (!intel_engine_supports_stats(engine))
268 if (!intel_engine_can_store_dword(engine))
271 if (intel_gt_pm_wait_for_idle(gt)) {
276 st_engine_heartbeat_disable(engine);
278 ENGINE_TRACE(engine, "measuring idle time\n");
280 de = intel_engine_get_busy_time(engine, &t[0]);
282 de = ktime_sub(intel_engine_get_busy_time(engine, &t[1]), de);
284 dt = ktime_sub(t[1], t[0]);
285 if (de < 0 || de > 10) {
286 pr_err("%s: reported %lldns [%d%%] busyness while sleeping [for %lldns]\n",
288 de, (int)div64_u64(100 * de, dt), dt);
295 rq = igt_spinner_create_request(&spin,
296 engine->kernel_context,
302 i915_request_add(rq);
304 busyness = intel_engine_get_busy_time(engine, &dummy);
305 if (!igt_wait_for_spinner(&spin, rq)) {
306 intel_gt_set_wedged(engine->gt);
311 err = __spin_until_busier(engine, busyness);
317 ENGINE_TRACE(engine, "measuring busy time\n");
319 de = intel_engine_get_busy_time(engine, &t[0]);
321 de = ktime_sub(intel_engine_get_busy_time(engine, &t[1]), de);
323 dt = ktime_sub(t[1], t[0]);
324 if (100 * de < 95 * dt || 95 * de > 100 * dt) {
325 pr_err("%s: reported %lldns [%d%%] busyness while spinning [for %lldns]\n",
327 de, (int)div64_u64(100 * de, dt), dt);
334 st_engine_heartbeat_enable(engine);
335 igt_spinner_end(&spin);
336 if (igt_flush_test(gt->i915))
342 igt_spinner_fini(&spin);
343 if (igt_flush_test(gt->i915))
348 static int live_engine_pm(void *arg)
350 struct intel_gt *gt = arg;
351 struct intel_engine_cs *engine;
352 enum intel_engine_id id;
355 * Check we can call intel_engine_pm_put from any context. No
356 * failures are reported directly, but if we mess up lockdep should
359 if (intel_gt_pm_wait_for_idle(gt)) {
360 pr_err("Unable to flush GT pm before test\n");
364 GEM_BUG_ON(intel_gt_pm_is_awake(gt));
365 for_each_engine(engine, gt, id) {
366 const typeof(*igt_atomic_phases) *p;
368 for (p = igt_atomic_phases; p->name; p++) {
370 * Acquisition is always synchronous, except if we
371 * know that the engine is already awake, in which
372 * case we should use intel_engine_pm_get_if_awake()
373 * to atomically grab the wakeref.
376 * intel_engine_pm_get();
377 * intel_engine_pm_put();
378 * occurs in one thread, while simultaneously
379 * intel_engine_pm_get_if_awake();
380 * intel_engine_pm_put();
381 * occurs from atomic context in another.
383 GEM_BUG_ON(intel_engine_pm_is_awake(engine));
384 intel_engine_pm_get(engine);
386 p->critical_section_begin();
387 if (!intel_engine_pm_get_if_awake(engine))
388 pr_err("intel_engine_pm_get_if_awake(%s) failed under %s\n",
389 engine->name, p->name);
391 intel_engine_pm_put_async(engine);
392 intel_engine_pm_put_async(engine);
393 p->critical_section_end();
395 intel_engine_pm_flush(engine);
397 if (intel_engine_pm_is_awake(engine)) {
398 pr_err("%s is still awake after flushing pm\n",
403 /* gt wakeref is async (deferred to workqueue) */
404 if (intel_gt_pm_wait_for_idle(gt)) {
405 pr_err("GT failed to idle\n");
414 int live_engine_pm_selftests(struct intel_gt *gt)
416 static const struct i915_subtest tests[] = {
417 SUBTEST(live_engine_timestamps),
418 SUBTEST(live_engine_busy_stats),
419 SUBTEST(live_engine_pm),
422 return intel_gt_live_subtests(tests, gt);