c816723b6b
This was done by the following semantic patch: @@ expression i915; @@ - INTEL_GEN(i915) + GRAPHICS_VER(i915) @@ expression i915; expression E; @@ - INTEL_GEN(i915) >= E + GRAPHICS_VER(i915) >= E @@ expression dev_priv; expression E; @@ - !IS_GEN(dev_priv, E) + GRAPHICS_VER(dev_priv) != E @@ expression dev_priv; expression E; @@ - IS_GEN(dev_priv, E) + GRAPHICS_VER(dev_priv) == E @@ expression dev_priv; expression from, until; @@ - IS_GEN_RANGE(dev_priv, from, until) + IS_GRAPHICS_VER(dev_priv, from, until) @def@ expression E; identifier id =~ "^gen$"; @@ - id = GRAPHICS_VER(E) + ver = GRAPHICS_VER(E) @@ identifier def.id; @@ - id + ver It also takes care of renaming the variable we assign to GRAPHICS_VER() so to use "ver" rather than "gen". Signed-off-by: Lucas De Marchi <lucas.demarchi@intel.com> Reviewed-by: Matt Roper <matthew.d.roper@intel.com> Link: https://patchwork.freedesktop.org/patch/msgid/20210605155356.4183026-2-lucas.demarchi@intel.com
1602 lines
35 KiB
C
1602 lines
35 KiB
C
// SPDX-License-Identifier: MIT
|
|
/*
|
|
* Copyright © 2014 Intel Corporation
|
|
*/
|
|
|
|
#include "gem/i915_gem_lmem.h"
|
|
|
|
#include "gen8_engine_cs.h"
|
|
#include "i915_drv.h"
|
|
#include "i915_perf.h"
|
|
#include "intel_engine.h"
|
|
#include "intel_gpu_commands.h"
|
|
#include "intel_gt.h"
|
|
#include "intel_lrc.h"
|
|
#include "intel_lrc_reg.h"
|
|
#include "intel_ring.h"
|
|
#include "shmem_utils.h"
|
|
|
|
static void set_offsets(u32 *regs,
|
|
const u8 *data,
|
|
const struct intel_engine_cs *engine,
|
|
bool close)
|
|
#define NOP(x) (BIT(7) | (x))
|
|
#define LRI(count, flags) ((flags) << 6 | (count) | BUILD_BUG_ON_ZERO(count >= BIT(6)))
|
|
#define POSTED BIT(0)
|
|
#define REG(x) (((x) >> 2) | BUILD_BUG_ON_ZERO(x >= 0x200))
|
|
#define REG16(x) \
|
|
(((x) >> 9) | BIT(7) | BUILD_BUG_ON_ZERO(x >= 0x10000)), \
|
|
(((x) >> 2) & 0x7f)
|
|
#define END 0
|
|
{
|
|
const u32 base = engine->mmio_base;
|
|
|
|
while (*data) {
|
|
u8 count, flags;
|
|
|
|
if (*data & BIT(7)) { /* skip */
|
|
count = *data++ & ~BIT(7);
|
|
regs += count;
|
|
continue;
|
|
}
|
|
|
|
count = *data & 0x3f;
|
|
flags = *data >> 6;
|
|
data++;
|
|
|
|
*regs = MI_LOAD_REGISTER_IMM(count);
|
|
if (flags & POSTED)
|
|
*regs |= MI_LRI_FORCE_POSTED;
|
|
if (GRAPHICS_VER(engine->i915) >= 11)
|
|
*regs |= MI_LRI_LRM_CS_MMIO;
|
|
regs++;
|
|
|
|
GEM_BUG_ON(!count);
|
|
do {
|
|
u32 offset = 0;
|
|
u8 v;
|
|
|
|
do {
|
|
v = *data++;
|
|
offset <<= 7;
|
|
offset |= v & ~BIT(7);
|
|
} while (v & BIT(7));
|
|
|
|
regs[0] = base + (offset << 2);
|
|
regs += 2;
|
|
} while (--count);
|
|
}
|
|
|
|
if (close) {
|
|
/* Close the batch; used mainly by live_lrc_layout() */
|
|
*regs = MI_BATCH_BUFFER_END;
|
|
if (GRAPHICS_VER(engine->i915) >= 10)
|
|
*regs |= BIT(0);
|
|
}
|
|
}
|
|
|
|
static const u8 gen8_xcs_offsets[] = {
|
|
NOP(1),
|
|
LRI(11, 0),
|
|
REG16(0x244),
|
|
REG(0x034),
|
|
REG(0x030),
|
|
REG(0x038),
|
|
REG(0x03c),
|
|
REG(0x168),
|
|
REG(0x140),
|
|
REG(0x110),
|
|
REG(0x11c),
|
|
REG(0x114),
|
|
REG(0x118),
|
|
|
|
NOP(9),
|
|
LRI(9, 0),
|
|
REG16(0x3a8),
|
|
REG16(0x28c),
|
|
REG16(0x288),
|
|
REG16(0x284),
|
|
REG16(0x280),
|
|
REG16(0x27c),
|
|
REG16(0x278),
|
|
REG16(0x274),
|
|
REG16(0x270),
|
|
|
|
NOP(13),
|
|
LRI(2, 0),
|
|
REG16(0x200),
|
|
REG(0x028),
|
|
|
|
END
|
|
};
|
|
|
|
static const u8 gen9_xcs_offsets[] = {
|
|
NOP(1),
|
|
LRI(14, POSTED),
|
|
REG16(0x244),
|
|
REG(0x034),
|
|
REG(0x030),
|
|
REG(0x038),
|
|
REG(0x03c),
|
|
REG(0x168),
|
|
REG(0x140),
|
|
REG(0x110),
|
|
REG(0x11c),
|
|
REG(0x114),
|
|
REG(0x118),
|
|
REG(0x1c0),
|
|
REG(0x1c4),
|
|
REG(0x1c8),
|
|
|
|
NOP(3),
|
|
LRI(9, POSTED),
|
|
REG16(0x3a8),
|
|
REG16(0x28c),
|
|
REG16(0x288),
|
|
REG16(0x284),
|
|
REG16(0x280),
|
|
REG16(0x27c),
|
|
REG16(0x278),
|
|
REG16(0x274),
|
|
REG16(0x270),
|
|
|
|
NOP(13),
|
|
LRI(1, POSTED),
|
|
REG16(0x200),
|
|
|
|
NOP(13),
|
|
LRI(44, POSTED),
|
|
REG(0x028),
|
|
REG(0x09c),
|
|
REG(0x0c0),
|
|
REG(0x178),
|
|
REG(0x17c),
|
|
REG16(0x358),
|
|
REG(0x170),
|
|
REG(0x150),
|
|
REG(0x154),
|
|
REG(0x158),
|
|
REG16(0x41c),
|
|
REG16(0x600),
|
|
REG16(0x604),
|
|
REG16(0x608),
|
|
REG16(0x60c),
|
|
REG16(0x610),
|
|
REG16(0x614),
|
|
REG16(0x618),
|
|
REG16(0x61c),
|
|
REG16(0x620),
|
|
REG16(0x624),
|
|
REG16(0x628),
|
|
REG16(0x62c),
|
|
REG16(0x630),
|
|
REG16(0x634),
|
|
REG16(0x638),
|
|
REG16(0x63c),
|
|
REG16(0x640),
|
|
REG16(0x644),
|
|
REG16(0x648),
|
|
REG16(0x64c),
|
|
REG16(0x650),
|
|
REG16(0x654),
|
|
REG16(0x658),
|
|
REG16(0x65c),
|
|
REG16(0x660),
|
|
REG16(0x664),
|
|
REG16(0x668),
|
|
REG16(0x66c),
|
|
REG16(0x670),
|
|
REG16(0x674),
|
|
REG16(0x678),
|
|
REG16(0x67c),
|
|
REG(0x068),
|
|
|
|
END
|
|
};
|
|
|
|
static const u8 gen12_xcs_offsets[] = {
|
|
NOP(1),
|
|
LRI(13, POSTED),
|
|
REG16(0x244),
|
|
REG(0x034),
|
|
REG(0x030),
|
|
REG(0x038),
|
|
REG(0x03c),
|
|
REG(0x168),
|
|
REG(0x140),
|
|
REG(0x110),
|
|
REG(0x1c0),
|
|
REG(0x1c4),
|
|
REG(0x1c8),
|
|
REG(0x180),
|
|
REG16(0x2b4),
|
|
|
|
NOP(5),
|
|
LRI(9, POSTED),
|
|
REG16(0x3a8),
|
|
REG16(0x28c),
|
|
REG16(0x288),
|
|
REG16(0x284),
|
|
REG16(0x280),
|
|
REG16(0x27c),
|
|
REG16(0x278),
|
|
REG16(0x274),
|
|
REG16(0x270),
|
|
|
|
END
|
|
};
|
|
|
|
static const u8 gen8_rcs_offsets[] = {
|
|
NOP(1),
|
|
LRI(14, POSTED),
|
|
REG16(0x244),
|
|
REG(0x034),
|
|
REG(0x030),
|
|
REG(0x038),
|
|
REG(0x03c),
|
|
REG(0x168),
|
|
REG(0x140),
|
|
REG(0x110),
|
|
REG(0x11c),
|
|
REG(0x114),
|
|
REG(0x118),
|
|
REG(0x1c0),
|
|
REG(0x1c4),
|
|
REG(0x1c8),
|
|
|
|
NOP(3),
|
|
LRI(9, POSTED),
|
|
REG16(0x3a8),
|
|
REG16(0x28c),
|
|
REG16(0x288),
|
|
REG16(0x284),
|
|
REG16(0x280),
|
|
REG16(0x27c),
|
|
REG16(0x278),
|
|
REG16(0x274),
|
|
REG16(0x270),
|
|
|
|
NOP(13),
|
|
LRI(1, 0),
|
|
REG(0x0c8),
|
|
|
|
END
|
|
};
|
|
|
|
static const u8 gen9_rcs_offsets[] = {
|
|
NOP(1),
|
|
LRI(14, POSTED),
|
|
REG16(0x244),
|
|
REG(0x34),
|
|
REG(0x30),
|
|
REG(0x38),
|
|
REG(0x3c),
|
|
REG(0x168),
|
|
REG(0x140),
|
|
REG(0x110),
|
|
REG(0x11c),
|
|
REG(0x114),
|
|
REG(0x118),
|
|
REG(0x1c0),
|
|
REG(0x1c4),
|
|
REG(0x1c8),
|
|
|
|
NOP(3),
|
|
LRI(9, POSTED),
|
|
REG16(0x3a8),
|
|
REG16(0x28c),
|
|
REG16(0x288),
|
|
REG16(0x284),
|
|
REG16(0x280),
|
|
REG16(0x27c),
|
|
REG16(0x278),
|
|
REG16(0x274),
|
|
REG16(0x270),
|
|
|
|
NOP(13),
|
|
LRI(1, 0),
|
|
REG(0xc8),
|
|
|
|
NOP(13),
|
|
LRI(44, POSTED),
|
|
REG(0x28),
|
|
REG(0x9c),
|
|
REG(0xc0),
|
|
REG(0x178),
|
|
REG(0x17c),
|
|
REG16(0x358),
|
|
REG(0x170),
|
|
REG(0x150),
|
|
REG(0x154),
|
|
REG(0x158),
|
|
REG16(0x41c),
|
|
REG16(0x600),
|
|
REG16(0x604),
|
|
REG16(0x608),
|
|
REG16(0x60c),
|
|
REG16(0x610),
|
|
REG16(0x614),
|
|
REG16(0x618),
|
|
REG16(0x61c),
|
|
REG16(0x620),
|
|
REG16(0x624),
|
|
REG16(0x628),
|
|
REG16(0x62c),
|
|
REG16(0x630),
|
|
REG16(0x634),
|
|
REG16(0x638),
|
|
REG16(0x63c),
|
|
REG16(0x640),
|
|
REG16(0x644),
|
|
REG16(0x648),
|
|
REG16(0x64c),
|
|
REG16(0x650),
|
|
REG16(0x654),
|
|
REG16(0x658),
|
|
REG16(0x65c),
|
|
REG16(0x660),
|
|
REG16(0x664),
|
|
REG16(0x668),
|
|
REG16(0x66c),
|
|
REG16(0x670),
|
|
REG16(0x674),
|
|
REG16(0x678),
|
|
REG16(0x67c),
|
|
REG(0x68),
|
|
|
|
END
|
|
};
|
|
|
|
static const u8 gen11_rcs_offsets[] = {
|
|
NOP(1),
|
|
LRI(15, POSTED),
|
|
REG16(0x244),
|
|
REG(0x034),
|
|
REG(0x030),
|
|
REG(0x038),
|
|
REG(0x03c),
|
|
REG(0x168),
|
|
REG(0x140),
|
|
REG(0x110),
|
|
REG(0x11c),
|
|
REG(0x114),
|
|
REG(0x118),
|
|
REG(0x1c0),
|
|
REG(0x1c4),
|
|
REG(0x1c8),
|
|
REG(0x180),
|
|
|
|
NOP(1),
|
|
LRI(9, POSTED),
|
|
REG16(0x3a8),
|
|
REG16(0x28c),
|
|
REG16(0x288),
|
|
REG16(0x284),
|
|
REG16(0x280),
|
|
REG16(0x27c),
|
|
REG16(0x278),
|
|
REG16(0x274),
|
|
REG16(0x270),
|
|
|
|
LRI(1, POSTED),
|
|
REG(0x1b0),
|
|
|
|
NOP(10),
|
|
LRI(1, 0),
|
|
REG(0x0c8),
|
|
|
|
END
|
|
};
|
|
|
|
static const u8 gen12_rcs_offsets[] = {
|
|
NOP(1),
|
|
LRI(13, POSTED),
|
|
REG16(0x244),
|
|
REG(0x034),
|
|
REG(0x030),
|
|
REG(0x038),
|
|
REG(0x03c),
|
|
REG(0x168),
|
|
REG(0x140),
|
|
REG(0x110),
|
|
REG(0x1c0),
|
|
REG(0x1c4),
|
|
REG(0x1c8),
|
|
REG(0x180),
|
|
REG16(0x2b4),
|
|
|
|
NOP(5),
|
|
LRI(9, POSTED),
|
|
REG16(0x3a8),
|
|
REG16(0x28c),
|
|
REG16(0x288),
|
|
REG16(0x284),
|
|
REG16(0x280),
|
|
REG16(0x27c),
|
|
REG16(0x278),
|
|
REG16(0x274),
|
|
REG16(0x270),
|
|
|
|
LRI(3, POSTED),
|
|
REG(0x1b0),
|
|
REG16(0x5a8),
|
|
REG16(0x5ac),
|
|
|
|
NOP(6),
|
|
LRI(1, 0),
|
|
REG(0x0c8),
|
|
NOP(3 + 9 + 1),
|
|
|
|
LRI(51, POSTED),
|
|
REG16(0x588),
|
|
REG16(0x588),
|
|
REG16(0x588),
|
|
REG16(0x588),
|
|
REG16(0x588),
|
|
REG16(0x588),
|
|
REG(0x028),
|
|
REG(0x09c),
|
|
REG(0x0c0),
|
|
REG(0x178),
|
|
REG(0x17c),
|
|
REG16(0x358),
|
|
REG(0x170),
|
|
REG(0x150),
|
|
REG(0x154),
|
|
REG(0x158),
|
|
REG16(0x41c),
|
|
REG16(0x600),
|
|
REG16(0x604),
|
|
REG16(0x608),
|
|
REG16(0x60c),
|
|
REG16(0x610),
|
|
REG16(0x614),
|
|
REG16(0x618),
|
|
REG16(0x61c),
|
|
REG16(0x620),
|
|
REG16(0x624),
|
|
REG16(0x628),
|
|
REG16(0x62c),
|
|
REG16(0x630),
|
|
REG16(0x634),
|
|
REG16(0x638),
|
|
REG16(0x63c),
|
|
REG16(0x640),
|
|
REG16(0x644),
|
|
REG16(0x648),
|
|
REG16(0x64c),
|
|
REG16(0x650),
|
|
REG16(0x654),
|
|
REG16(0x658),
|
|
REG16(0x65c),
|
|
REG16(0x660),
|
|
REG16(0x664),
|
|
REG16(0x668),
|
|
REG16(0x66c),
|
|
REG16(0x670),
|
|
REG16(0x674),
|
|
REG16(0x678),
|
|
REG16(0x67c),
|
|
REG(0x068),
|
|
REG(0x084),
|
|
NOP(1),
|
|
|
|
END
|
|
};
|
|
|
|
#undef END
|
|
#undef REG16
|
|
#undef REG
|
|
#undef LRI
|
|
#undef NOP
|
|
|
|
static const u8 *reg_offsets(const struct intel_engine_cs *engine)
|
|
{
|
|
/*
|
|
* The gen12+ lists only have the registers we program in the basic
|
|
* default state. We rely on the context image using relative
|
|
* addressing to automatic fixup the register state between the
|
|
* physical engines for virtual engine.
|
|
*/
|
|
GEM_BUG_ON(GRAPHICS_VER(engine->i915) >= 12 &&
|
|
!intel_engine_has_relative_mmio(engine));
|
|
|
|
if (engine->class == RENDER_CLASS) {
|
|
if (GRAPHICS_VER(engine->i915) >= 12)
|
|
return gen12_rcs_offsets;
|
|
else if (GRAPHICS_VER(engine->i915) >= 11)
|
|
return gen11_rcs_offsets;
|
|
else if (GRAPHICS_VER(engine->i915) >= 9)
|
|
return gen9_rcs_offsets;
|
|
else
|
|
return gen8_rcs_offsets;
|
|
} else {
|
|
if (GRAPHICS_VER(engine->i915) >= 12)
|
|
return gen12_xcs_offsets;
|
|
else if (GRAPHICS_VER(engine->i915) >= 9)
|
|
return gen9_xcs_offsets;
|
|
else
|
|
return gen8_xcs_offsets;
|
|
}
|
|
}
|
|
|
|
static int lrc_ring_mi_mode(const struct intel_engine_cs *engine)
|
|
{
|
|
if (GRAPHICS_VER(engine->i915) >= 12)
|
|
return 0x60;
|
|
else if (GRAPHICS_VER(engine->i915) >= 9)
|
|
return 0x54;
|
|
else if (engine->class == RENDER_CLASS)
|
|
return 0x58;
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
static int lrc_ring_gpr0(const struct intel_engine_cs *engine)
|
|
{
|
|
if (GRAPHICS_VER(engine->i915) >= 12)
|
|
return 0x74;
|
|
else if (GRAPHICS_VER(engine->i915) >= 9)
|
|
return 0x68;
|
|
else if (engine->class == RENDER_CLASS)
|
|
return 0xd8;
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
static int lrc_ring_wa_bb_per_ctx(const struct intel_engine_cs *engine)
|
|
{
|
|
if (GRAPHICS_VER(engine->i915) >= 12)
|
|
return 0x12;
|
|
else if (GRAPHICS_VER(engine->i915) >= 9 || engine->class == RENDER_CLASS)
|
|
return 0x18;
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
static int lrc_ring_indirect_ptr(const struct intel_engine_cs *engine)
|
|
{
|
|
int x;
|
|
|
|
x = lrc_ring_wa_bb_per_ctx(engine);
|
|
if (x < 0)
|
|
return x;
|
|
|
|
return x + 2;
|
|
}
|
|
|
|
static int lrc_ring_indirect_offset(const struct intel_engine_cs *engine)
|
|
{
|
|
int x;
|
|
|
|
x = lrc_ring_indirect_ptr(engine);
|
|
if (x < 0)
|
|
return x;
|
|
|
|
return x + 2;
|
|
}
|
|
|
|
static int lrc_ring_cmd_buf_cctl(const struct intel_engine_cs *engine)
|
|
{
|
|
if (engine->class != RENDER_CLASS)
|
|
return -1;
|
|
|
|
if (GRAPHICS_VER(engine->i915) >= 12)
|
|
return 0xb6;
|
|
else if (GRAPHICS_VER(engine->i915) >= 11)
|
|
return 0xaa;
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
static u32
|
|
lrc_ring_indirect_offset_default(const struct intel_engine_cs *engine)
|
|
{
|
|
switch (GRAPHICS_VER(engine->i915)) {
|
|
default:
|
|
MISSING_CASE(GRAPHICS_VER(engine->i915));
|
|
fallthrough;
|
|
case 12:
|
|
return GEN12_CTX_RCS_INDIRECT_CTX_OFFSET_DEFAULT;
|
|
case 11:
|
|
return GEN11_CTX_RCS_INDIRECT_CTX_OFFSET_DEFAULT;
|
|
case 10:
|
|
return GEN10_CTX_RCS_INDIRECT_CTX_OFFSET_DEFAULT;
|
|
case 9:
|
|
return GEN9_CTX_RCS_INDIRECT_CTX_OFFSET_DEFAULT;
|
|
case 8:
|
|
return GEN8_CTX_RCS_INDIRECT_CTX_OFFSET_DEFAULT;
|
|
}
|
|
}
|
|
|
|
static void
|
|
lrc_setup_indirect_ctx(u32 *regs,
|
|
const struct intel_engine_cs *engine,
|
|
u32 ctx_bb_ggtt_addr,
|
|
u32 size)
|
|
{
|
|
GEM_BUG_ON(!size);
|
|
GEM_BUG_ON(!IS_ALIGNED(size, CACHELINE_BYTES));
|
|
GEM_BUG_ON(lrc_ring_indirect_ptr(engine) == -1);
|
|
regs[lrc_ring_indirect_ptr(engine) + 1] =
|
|
ctx_bb_ggtt_addr | (size / CACHELINE_BYTES);
|
|
|
|
GEM_BUG_ON(lrc_ring_indirect_offset(engine) == -1);
|
|
regs[lrc_ring_indirect_offset(engine) + 1] =
|
|
lrc_ring_indirect_offset_default(engine) << 6;
|
|
}
|
|
|
|
static void init_common_regs(u32 * const regs,
|
|
const struct intel_context *ce,
|
|
const struct intel_engine_cs *engine,
|
|
bool inhibit)
|
|
{
|
|
u32 ctl;
|
|
|
|
ctl = _MASKED_BIT_ENABLE(CTX_CTRL_INHIBIT_SYN_CTX_SWITCH);
|
|
ctl |= _MASKED_BIT_DISABLE(CTX_CTRL_ENGINE_CTX_RESTORE_INHIBIT);
|
|
if (inhibit)
|
|
ctl |= CTX_CTRL_ENGINE_CTX_RESTORE_INHIBIT;
|
|
if (GRAPHICS_VER(engine->i915) < 11)
|
|
ctl |= _MASKED_BIT_DISABLE(CTX_CTRL_ENGINE_CTX_SAVE_INHIBIT |
|
|
CTX_CTRL_RS_CTX_ENABLE);
|
|
regs[CTX_CONTEXT_CONTROL] = ctl;
|
|
|
|
regs[CTX_TIMESTAMP] = ce->runtime.last;
|
|
}
|
|
|
|
static void init_wa_bb_regs(u32 * const regs,
|
|
const struct intel_engine_cs *engine)
|
|
{
|
|
const struct i915_ctx_workarounds * const wa_ctx = &engine->wa_ctx;
|
|
|
|
if (wa_ctx->per_ctx.size) {
|
|
const u32 ggtt_offset = i915_ggtt_offset(wa_ctx->vma);
|
|
|
|
GEM_BUG_ON(lrc_ring_wa_bb_per_ctx(engine) == -1);
|
|
regs[lrc_ring_wa_bb_per_ctx(engine) + 1] =
|
|
(ggtt_offset + wa_ctx->per_ctx.offset) | 0x01;
|
|
}
|
|
|
|
if (wa_ctx->indirect_ctx.size) {
|
|
lrc_setup_indirect_ctx(regs, engine,
|
|
i915_ggtt_offset(wa_ctx->vma) +
|
|
wa_ctx->indirect_ctx.offset,
|
|
wa_ctx->indirect_ctx.size);
|
|
}
|
|
}
|
|
|
|
static void init_ppgtt_regs(u32 *regs, const struct i915_ppgtt *ppgtt)
|
|
{
|
|
if (i915_vm_is_4lvl(&ppgtt->vm)) {
|
|
/* 64b PPGTT (48bit canonical)
|
|
* PDP0_DESCRIPTOR contains the base address to PML4 and
|
|
* other PDP Descriptors are ignored.
|
|
*/
|
|
ASSIGN_CTX_PML4(ppgtt, regs);
|
|
} else {
|
|
ASSIGN_CTX_PDP(ppgtt, regs, 3);
|
|
ASSIGN_CTX_PDP(ppgtt, regs, 2);
|
|
ASSIGN_CTX_PDP(ppgtt, regs, 1);
|
|
ASSIGN_CTX_PDP(ppgtt, regs, 0);
|
|
}
|
|
}
|
|
|
|
static struct i915_ppgtt *vm_alias(struct i915_address_space *vm)
|
|
{
|
|
if (i915_is_ggtt(vm))
|
|
return i915_vm_to_ggtt(vm)->alias;
|
|
else
|
|
return i915_vm_to_ppgtt(vm);
|
|
}
|
|
|
|
static void __reset_stop_ring(u32 *regs, const struct intel_engine_cs *engine)
|
|
{
|
|
int x;
|
|
|
|
x = lrc_ring_mi_mode(engine);
|
|
if (x != -1) {
|
|
regs[x + 1] &= ~STOP_RING;
|
|
regs[x + 1] |= STOP_RING << 16;
|
|
}
|
|
}
|
|
|
|
static void __lrc_init_regs(u32 *regs,
|
|
const struct intel_context *ce,
|
|
const struct intel_engine_cs *engine,
|
|
bool inhibit)
|
|
{
|
|
/*
|
|
* A context is actually a big batch buffer with several
|
|
* MI_LOAD_REGISTER_IMM commands followed by (reg, value) pairs. The
|
|
* values we are setting here are only for the first context restore:
|
|
* on a subsequent save, the GPU will recreate this batchbuffer with new
|
|
* values (including all the missing MI_LOAD_REGISTER_IMM commands that
|
|
* we are not initializing here).
|
|
*
|
|
* Must keep consistent with virtual_update_register_offsets().
|
|
*/
|
|
|
|
if (inhibit)
|
|
memset(regs, 0, PAGE_SIZE);
|
|
|
|
set_offsets(regs, reg_offsets(engine), engine, inhibit);
|
|
|
|
init_common_regs(regs, ce, engine, inhibit);
|
|
init_ppgtt_regs(regs, vm_alias(ce->vm));
|
|
|
|
init_wa_bb_regs(regs, engine);
|
|
|
|
__reset_stop_ring(regs, engine);
|
|
}
|
|
|
|
void lrc_init_regs(const struct intel_context *ce,
|
|
const struct intel_engine_cs *engine,
|
|
bool inhibit)
|
|
{
|
|
__lrc_init_regs(ce->lrc_reg_state, ce, engine, inhibit);
|
|
}
|
|
|
|
void lrc_reset_regs(const struct intel_context *ce,
|
|
const struct intel_engine_cs *engine)
|
|
{
|
|
__reset_stop_ring(ce->lrc_reg_state, engine);
|
|
}
|
|
|
|
static void
|
|
set_redzone(void *vaddr, const struct intel_engine_cs *engine)
|
|
{
|
|
if (!IS_ENABLED(CONFIG_DRM_I915_DEBUG_GEM))
|
|
return;
|
|
|
|
vaddr += engine->context_size;
|
|
|
|
memset(vaddr, CONTEXT_REDZONE, I915_GTT_PAGE_SIZE);
|
|
}
|
|
|
|
static void
|
|
check_redzone(const void *vaddr, const struct intel_engine_cs *engine)
|
|
{
|
|
if (!IS_ENABLED(CONFIG_DRM_I915_DEBUG_GEM))
|
|
return;
|
|
|
|
vaddr += engine->context_size;
|
|
|
|
if (memchr_inv(vaddr, CONTEXT_REDZONE, I915_GTT_PAGE_SIZE))
|
|
drm_err_once(&engine->i915->drm,
|
|
"%s context redzone overwritten!\n",
|
|
engine->name);
|
|
}
|
|
|
|
void lrc_init_state(struct intel_context *ce,
|
|
struct intel_engine_cs *engine,
|
|
void *state)
|
|
{
|
|
bool inhibit = true;
|
|
|
|
set_redzone(state, engine);
|
|
|
|
if (engine->default_state) {
|
|
shmem_read(engine->default_state, 0,
|
|
state, engine->context_size);
|
|
__set_bit(CONTEXT_VALID_BIT, &ce->flags);
|
|
inhibit = false;
|
|
}
|
|
|
|
/* Clear the ppHWSP (inc. per-context counters) */
|
|
memset(state, 0, PAGE_SIZE);
|
|
|
|
/*
|
|
* The second page of the context object contains some registers which
|
|
* must be set up prior to the first execution.
|
|
*/
|
|
__lrc_init_regs(state + LRC_STATE_OFFSET, ce, engine, inhibit);
|
|
}
|
|
|
|
static struct i915_vma *
|
|
__lrc_alloc_state(struct intel_context *ce, struct intel_engine_cs *engine)
|
|
{
|
|
struct drm_i915_gem_object *obj;
|
|
struct i915_vma *vma;
|
|
u32 context_size;
|
|
|
|
context_size = round_up(engine->context_size, I915_GTT_PAGE_SIZE);
|
|
|
|
if (IS_ENABLED(CONFIG_DRM_I915_DEBUG_GEM))
|
|
context_size += I915_GTT_PAGE_SIZE; /* for redzone */
|
|
|
|
if (GRAPHICS_VER(engine->i915) == 12) {
|
|
ce->wa_bb_page = context_size / PAGE_SIZE;
|
|
context_size += PAGE_SIZE;
|
|
}
|
|
|
|
obj = i915_gem_object_create_lmem(engine->i915, context_size, 0);
|
|
if (IS_ERR(obj))
|
|
obj = i915_gem_object_create_shmem(engine->i915, context_size);
|
|
if (IS_ERR(obj))
|
|
return ERR_CAST(obj);
|
|
|
|
vma = i915_vma_instance(obj, &engine->gt->ggtt->vm, NULL);
|
|
if (IS_ERR(vma)) {
|
|
i915_gem_object_put(obj);
|
|
return vma;
|
|
}
|
|
|
|
return vma;
|
|
}
|
|
|
|
static struct intel_timeline *
|
|
pinned_timeline(struct intel_context *ce, struct intel_engine_cs *engine)
|
|
{
|
|
struct intel_timeline *tl = fetch_and_zero(&ce->timeline);
|
|
|
|
return intel_timeline_create_from_engine(engine, page_unmask_bits(tl));
|
|
}
|
|
|
|
int lrc_alloc(struct intel_context *ce, struct intel_engine_cs *engine)
|
|
{
|
|
struct intel_ring *ring;
|
|
struct i915_vma *vma;
|
|
int err;
|
|
|
|
GEM_BUG_ON(ce->state);
|
|
|
|
vma = __lrc_alloc_state(ce, engine);
|
|
if (IS_ERR(vma))
|
|
return PTR_ERR(vma);
|
|
|
|
ring = intel_engine_create_ring(engine, (unsigned long)ce->ring);
|
|
if (IS_ERR(ring)) {
|
|
err = PTR_ERR(ring);
|
|
goto err_vma;
|
|
}
|
|
|
|
if (!page_mask_bits(ce->timeline)) {
|
|
struct intel_timeline *tl;
|
|
|
|
/*
|
|
* Use the static global HWSP for the kernel context, and
|
|
* a dynamically allocated cacheline for everyone else.
|
|
*/
|
|
if (unlikely(ce->timeline))
|
|
tl = pinned_timeline(ce, engine);
|
|
else
|
|
tl = intel_timeline_create(engine->gt);
|
|
if (IS_ERR(tl)) {
|
|
err = PTR_ERR(tl);
|
|
goto err_ring;
|
|
}
|
|
|
|
ce->timeline = tl;
|
|
}
|
|
|
|
ce->ring = ring;
|
|
ce->state = vma;
|
|
|
|
return 0;
|
|
|
|
err_ring:
|
|
intel_ring_put(ring);
|
|
err_vma:
|
|
i915_vma_put(vma);
|
|
return err;
|
|
}
|
|
|
|
void lrc_reset(struct intel_context *ce)
|
|
{
|
|
GEM_BUG_ON(!intel_context_is_pinned(ce));
|
|
|
|
intel_ring_reset(ce->ring, ce->ring->emit);
|
|
|
|
/* Scrub away the garbage */
|
|
lrc_init_regs(ce, ce->engine, true);
|
|
ce->lrc.lrca = lrc_update_regs(ce, ce->engine, ce->ring->tail);
|
|
}
|
|
|
|
int
|
|
lrc_pre_pin(struct intel_context *ce,
|
|
struct intel_engine_cs *engine,
|
|
struct i915_gem_ww_ctx *ww,
|
|
void **vaddr)
|
|
{
|
|
GEM_BUG_ON(!ce->state);
|
|
GEM_BUG_ON(!i915_vma_is_pinned(ce->state));
|
|
|
|
*vaddr = i915_gem_object_pin_map(ce->state->obj,
|
|
i915_coherent_map_type(ce->engine->i915,
|
|
ce->state->obj,
|
|
false) |
|
|
I915_MAP_OVERRIDE);
|
|
|
|
return PTR_ERR_OR_ZERO(*vaddr);
|
|
}
|
|
|
|
int
|
|
lrc_pin(struct intel_context *ce,
|
|
struct intel_engine_cs *engine,
|
|
void *vaddr)
|
|
{
|
|
ce->lrc_reg_state = vaddr + LRC_STATE_OFFSET;
|
|
|
|
if (!__test_and_set_bit(CONTEXT_INIT_BIT, &ce->flags))
|
|
lrc_init_state(ce, engine, vaddr);
|
|
|
|
ce->lrc.lrca = lrc_update_regs(ce, engine, ce->ring->tail);
|
|
return 0;
|
|
}
|
|
|
|
void lrc_unpin(struct intel_context *ce)
|
|
{
|
|
check_redzone((void *)ce->lrc_reg_state - LRC_STATE_OFFSET,
|
|
ce->engine);
|
|
}
|
|
|
|
void lrc_post_unpin(struct intel_context *ce)
|
|
{
|
|
i915_gem_object_unpin_map(ce->state->obj);
|
|
}
|
|
|
|
void lrc_fini(struct intel_context *ce)
|
|
{
|
|
if (!ce->state)
|
|
return;
|
|
|
|
intel_ring_put(fetch_and_zero(&ce->ring));
|
|
i915_vma_put(fetch_and_zero(&ce->state));
|
|
}
|
|
|
|
void lrc_destroy(struct kref *kref)
|
|
{
|
|
struct intel_context *ce = container_of(kref, typeof(*ce), ref);
|
|
|
|
GEM_BUG_ON(!i915_active_is_idle(&ce->active));
|
|
GEM_BUG_ON(intel_context_is_pinned(ce));
|
|
|
|
lrc_fini(ce);
|
|
|
|
intel_context_fini(ce);
|
|
intel_context_free(ce);
|
|
}
|
|
|
|
static u32 *
|
|
gen12_emit_timestamp_wa(const struct intel_context *ce, u32 *cs)
|
|
{
|
|
*cs++ = MI_LOAD_REGISTER_MEM_GEN8 |
|
|
MI_SRM_LRM_GLOBAL_GTT |
|
|
MI_LRI_LRM_CS_MMIO;
|
|
*cs++ = i915_mmio_reg_offset(GEN8_RING_CS_GPR(0, 0));
|
|
*cs++ = i915_ggtt_offset(ce->state) + LRC_STATE_OFFSET +
|
|
CTX_TIMESTAMP * sizeof(u32);
|
|
*cs++ = 0;
|
|
|
|
*cs++ = MI_LOAD_REGISTER_REG |
|
|
MI_LRR_SOURCE_CS_MMIO |
|
|
MI_LRI_LRM_CS_MMIO;
|
|
*cs++ = i915_mmio_reg_offset(GEN8_RING_CS_GPR(0, 0));
|
|
*cs++ = i915_mmio_reg_offset(RING_CTX_TIMESTAMP(0));
|
|
|
|
*cs++ = MI_LOAD_REGISTER_REG |
|
|
MI_LRR_SOURCE_CS_MMIO |
|
|
MI_LRI_LRM_CS_MMIO;
|
|
*cs++ = i915_mmio_reg_offset(GEN8_RING_CS_GPR(0, 0));
|
|
*cs++ = i915_mmio_reg_offset(RING_CTX_TIMESTAMP(0));
|
|
|
|
return cs;
|
|
}
|
|
|
|
static u32 *
|
|
gen12_emit_restore_scratch(const struct intel_context *ce, u32 *cs)
|
|
{
|
|
GEM_BUG_ON(lrc_ring_gpr0(ce->engine) == -1);
|
|
|
|
*cs++ = MI_LOAD_REGISTER_MEM_GEN8 |
|
|
MI_SRM_LRM_GLOBAL_GTT |
|
|
MI_LRI_LRM_CS_MMIO;
|
|
*cs++ = i915_mmio_reg_offset(GEN8_RING_CS_GPR(0, 0));
|
|
*cs++ = i915_ggtt_offset(ce->state) + LRC_STATE_OFFSET +
|
|
(lrc_ring_gpr0(ce->engine) + 1) * sizeof(u32);
|
|
*cs++ = 0;
|
|
|
|
return cs;
|
|
}
|
|
|
|
static u32 *
|
|
gen12_emit_cmd_buf_wa(const struct intel_context *ce, u32 *cs)
|
|
{
|
|
GEM_BUG_ON(lrc_ring_cmd_buf_cctl(ce->engine) == -1);
|
|
|
|
*cs++ = MI_LOAD_REGISTER_MEM_GEN8 |
|
|
MI_SRM_LRM_GLOBAL_GTT |
|
|
MI_LRI_LRM_CS_MMIO;
|
|
*cs++ = i915_mmio_reg_offset(GEN8_RING_CS_GPR(0, 0));
|
|
*cs++ = i915_ggtt_offset(ce->state) + LRC_STATE_OFFSET +
|
|
(lrc_ring_cmd_buf_cctl(ce->engine) + 1) * sizeof(u32);
|
|
*cs++ = 0;
|
|
|
|
*cs++ = MI_LOAD_REGISTER_REG |
|
|
MI_LRR_SOURCE_CS_MMIO |
|
|
MI_LRI_LRM_CS_MMIO;
|
|
*cs++ = i915_mmio_reg_offset(GEN8_RING_CS_GPR(0, 0));
|
|
*cs++ = i915_mmio_reg_offset(RING_CMD_BUF_CCTL(0));
|
|
|
|
return cs;
|
|
}
|
|
|
|
static u32 *
|
|
gen12_emit_indirect_ctx_rcs(const struct intel_context *ce, u32 *cs)
|
|
{
|
|
cs = gen12_emit_timestamp_wa(ce, cs);
|
|
cs = gen12_emit_cmd_buf_wa(ce, cs);
|
|
cs = gen12_emit_restore_scratch(ce, cs);
|
|
|
|
return cs;
|
|
}
|
|
|
|
static u32 *
|
|
gen12_emit_indirect_ctx_xcs(const struct intel_context *ce, u32 *cs)
|
|
{
|
|
cs = gen12_emit_timestamp_wa(ce, cs);
|
|
cs = gen12_emit_restore_scratch(ce, cs);
|
|
|
|
return cs;
|
|
}
|
|
|
|
static u32 context_wa_bb_offset(const struct intel_context *ce)
|
|
{
|
|
return PAGE_SIZE * ce->wa_bb_page;
|
|
}
|
|
|
|
static u32 *context_indirect_bb(const struct intel_context *ce)
|
|
{
|
|
void *ptr;
|
|
|
|
GEM_BUG_ON(!ce->wa_bb_page);
|
|
|
|
ptr = ce->lrc_reg_state;
|
|
ptr -= LRC_STATE_OFFSET; /* back to start of context image */
|
|
ptr += context_wa_bb_offset(ce);
|
|
|
|
return ptr;
|
|
}
|
|
|
|
static void
|
|
setup_indirect_ctx_bb(const struct intel_context *ce,
|
|
const struct intel_engine_cs *engine,
|
|
u32 *(*emit)(const struct intel_context *, u32 *))
|
|
{
|
|
u32 * const start = context_indirect_bb(ce);
|
|
u32 *cs;
|
|
|
|
cs = emit(ce, start);
|
|
GEM_BUG_ON(cs - start > I915_GTT_PAGE_SIZE / sizeof(*cs));
|
|
while ((unsigned long)cs % CACHELINE_BYTES)
|
|
*cs++ = MI_NOOP;
|
|
|
|
lrc_setup_indirect_ctx(ce->lrc_reg_state, engine,
|
|
i915_ggtt_offset(ce->state) +
|
|
context_wa_bb_offset(ce),
|
|
(cs - start) * sizeof(*cs));
|
|
}
|
|
|
|
/*
|
|
* The context descriptor encodes various attributes of a context,
|
|
* including its GTT address and some flags. Because it's fairly
|
|
* expensive to calculate, we'll just do it once and cache the result,
|
|
* which remains valid until the context is unpinned.
|
|
*
|
|
* This is what a descriptor looks like, from LSB to MSB::
|
|
*
|
|
* bits 0-11: flags, GEN8_CTX_* (cached in ctx->desc_template)
|
|
* bits 12-31: LRCA, GTT address of (the HWSP of) this context
|
|
* bits 32-52: ctx ID, a globally unique tag (highest bit used by GuC)
|
|
* bits 53-54: mbz, reserved for use by hardware
|
|
* bits 55-63: group ID, currently unused and set to 0
|
|
*
|
|
* Starting from Gen11, the upper dword of the descriptor has a new format:
|
|
*
|
|
* bits 32-36: reserved
|
|
* bits 37-47: SW context ID
|
|
* bits 48:53: engine instance
|
|
* bit 54: mbz, reserved for use by hardware
|
|
* bits 55-60: SW counter
|
|
* bits 61-63: engine class
|
|
*
|
|
* engine info, SW context ID and SW counter need to form a unique number
|
|
* (Context ID) per lrc.
|
|
*/
|
|
static u32 lrc_descriptor(const struct intel_context *ce)
|
|
{
|
|
u32 desc;
|
|
|
|
desc = INTEL_LEGACY_32B_CONTEXT;
|
|
if (i915_vm_is_4lvl(ce->vm))
|
|
desc = INTEL_LEGACY_64B_CONTEXT;
|
|
desc <<= GEN8_CTX_ADDRESSING_MODE_SHIFT;
|
|
|
|
desc |= GEN8_CTX_VALID | GEN8_CTX_PRIVILEGE;
|
|
if (GRAPHICS_VER(ce->vm->i915) == 8)
|
|
desc |= GEN8_CTX_L3LLC_COHERENT;
|
|
|
|
return i915_ggtt_offset(ce->state) | desc;
|
|
}
|
|
|
|
u32 lrc_update_regs(const struct intel_context *ce,
|
|
const struct intel_engine_cs *engine,
|
|
u32 head)
|
|
{
|
|
struct intel_ring *ring = ce->ring;
|
|
u32 *regs = ce->lrc_reg_state;
|
|
|
|
GEM_BUG_ON(!intel_ring_offset_valid(ring, head));
|
|
GEM_BUG_ON(!intel_ring_offset_valid(ring, ring->tail));
|
|
|
|
regs[CTX_RING_START] = i915_ggtt_offset(ring->vma);
|
|
regs[CTX_RING_HEAD] = head;
|
|
regs[CTX_RING_TAIL] = ring->tail;
|
|
regs[CTX_RING_CTL] = RING_CTL_SIZE(ring->size) | RING_VALID;
|
|
|
|
/* RPCS */
|
|
if (engine->class == RENDER_CLASS) {
|
|
regs[CTX_R_PWR_CLK_STATE] =
|
|
intel_sseu_make_rpcs(engine->gt, &ce->sseu);
|
|
|
|
i915_oa_init_reg_state(ce, engine);
|
|
}
|
|
|
|
if (ce->wa_bb_page) {
|
|
u32 *(*fn)(const struct intel_context *ce, u32 *cs);
|
|
|
|
fn = gen12_emit_indirect_ctx_xcs;
|
|
if (ce->engine->class == RENDER_CLASS)
|
|
fn = gen12_emit_indirect_ctx_rcs;
|
|
|
|
/* Mutually exclusive wrt to global indirect bb */
|
|
GEM_BUG_ON(engine->wa_ctx.indirect_ctx.size);
|
|
setup_indirect_ctx_bb(ce, engine, fn);
|
|
}
|
|
|
|
return lrc_descriptor(ce) | CTX_DESC_FORCE_RESTORE;
|
|
}
|
|
|
|
void lrc_update_offsets(struct intel_context *ce,
|
|
struct intel_engine_cs *engine)
|
|
{
|
|
set_offsets(ce->lrc_reg_state, reg_offsets(engine), engine, false);
|
|
}
|
|
|
|
void lrc_check_regs(const struct intel_context *ce,
|
|
const struct intel_engine_cs *engine,
|
|
const char *when)
|
|
{
|
|
const struct intel_ring *ring = ce->ring;
|
|
u32 *regs = ce->lrc_reg_state;
|
|
bool valid = true;
|
|
int x;
|
|
|
|
if (regs[CTX_RING_START] != i915_ggtt_offset(ring->vma)) {
|
|
pr_err("%s: context submitted with incorrect RING_START [%08x], expected %08x\n",
|
|
engine->name,
|
|
regs[CTX_RING_START],
|
|
i915_ggtt_offset(ring->vma));
|
|
regs[CTX_RING_START] = i915_ggtt_offset(ring->vma);
|
|
valid = false;
|
|
}
|
|
|
|
if ((regs[CTX_RING_CTL] & ~(RING_WAIT | RING_WAIT_SEMAPHORE)) !=
|
|
(RING_CTL_SIZE(ring->size) | RING_VALID)) {
|
|
pr_err("%s: context submitted with incorrect RING_CTL [%08x], expected %08x\n",
|
|
engine->name,
|
|
regs[CTX_RING_CTL],
|
|
(u32)(RING_CTL_SIZE(ring->size) | RING_VALID));
|
|
regs[CTX_RING_CTL] = RING_CTL_SIZE(ring->size) | RING_VALID;
|
|
valid = false;
|
|
}
|
|
|
|
x = lrc_ring_mi_mode(engine);
|
|
if (x != -1 && regs[x + 1] & (regs[x + 1] >> 16) & STOP_RING) {
|
|
pr_err("%s: context submitted with STOP_RING [%08x] in RING_MI_MODE\n",
|
|
engine->name, regs[x + 1]);
|
|
regs[x + 1] &= ~STOP_RING;
|
|
regs[x + 1] |= STOP_RING << 16;
|
|
valid = false;
|
|
}
|
|
|
|
WARN_ONCE(!valid, "Invalid lrc state found %s submission\n", when);
|
|
}
|
|
|
|
/*
|
|
* In this WA we need to set GEN8_L3SQCREG4[21:21] and reset it after
|
|
* PIPE_CONTROL instruction. This is required for the flush to happen correctly
|
|
* but there is a slight complication as this is applied in WA batch where the
|
|
* values are only initialized once so we cannot take register value at the
|
|
* beginning and reuse it further; hence we save its value to memory, upload a
|
|
* constant value with bit21 set and then we restore it back with the saved value.
|
|
* To simplify the WA, a constant value is formed by using the default value
|
|
* of this register. This shouldn't be a problem because we are only modifying
|
|
* it for a short period and this batch in non-premptible. We can ofcourse
|
|
* use additional instructions that read the actual value of the register
|
|
* at that time and set our bit of interest but it makes the WA complicated.
|
|
*
|
|
* This WA is also required for Gen9 so extracting as a function avoids
|
|
* code duplication.
|
|
*/
|
|
static u32 *
|
|
gen8_emit_flush_coherentl3_wa(struct intel_engine_cs *engine, u32 *batch)
|
|
{
|
|
/* NB no one else is allowed to scribble over scratch + 256! */
|
|
*batch++ = MI_STORE_REGISTER_MEM_GEN8 | MI_SRM_LRM_GLOBAL_GTT;
|
|
*batch++ = i915_mmio_reg_offset(GEN8_L3SQCREG4);
|
|
*batch++ = intel_gt_scratch_offset(engine->gt,
|
|
INTEL_GT_SCRATCH_FIELD_COHERENTL3_WA);
|
|
*batch++ = 0;
|
|
|
|
*batch++ = MI_LOAD_REGISTER_IMM(1);
|
|
*batch++ = i915_mmio_reg_offset(GEN8_L3SQCREG4);
|
|
*batch++ = 0x40400000 | GEN8_LQSC_FLUSH_COHERENT_LINES;
|
|
|
|
batch = gen8_emit_pipe_control(batch,
|
|
PIPE_CONTROL_CS_STALL |
|
|
PIPE_CONTROL_DC_FLUSH_ENABLE,
|
|
0);
|
|
|
|
*batch++ = MI_LOAD_REGISTER_MEM_GEN8 | MI_SRM_LRM_GLOBAL_GTT;
|
|
*batch++ = i915_mmio_reg_offset(GEN8_L3SQCREG4);
|
|
*batch++ = intel_gt_scratch_offset(engine->gt,
|
|
INTEL_GT_SCRATCH_FIELD_COHERENTL3_WA);
|
|
*batch++ = 0;
|
|
|
|
return batch;
|
|
}
|
|
|
|
/*
|
|
* Typically we only have one indirect_ctx and per_ctx batch buffer which are
|
|
* initialized at the beginning and shared across all contexts but this field
|
|
* helps us to have multiple batches at different offsets and select them based
|
|
* on a criteria. At the moment this batch always start at the beginning of the page
|
|
* and at this point we don't have multiple wa_ctx batch buffers.
|
|
*
|
|
* The number of WA applied are not known at the beginning; we use this field
|
|
* to return the no of DWORDS written.
|
|
*
|
|
* It is to be noted that this batch does not contain MI_BATCH_BUFFER_END
|
|
* so it adds NOOPs as padding to make it cacheline aligned.
|
|
* MI_BATCH_BUFFER_END will be added to perctx batch and both of them together
|
|
* makes a complete batch buffer.
|
|
*/
|
|
static u32 *gen8_init_indirectctx_bb(struct intel_engine_cs *engine, u32 *batch)
|
|
{
|
|
/* WaDisableCtxRestoreArbitration:bdw,chv */
|
|
*batch++ = MI_ARB_ON_OFF | MI_ARB_DISABLE;
|
|
|
|
/* WaFlushCoherentL3CacheLinesAtContextSwitch:bdw */
|
|
if (IS_BROADWELL(engine->i915))
|
|
batch = gen8_emit_flush_coherentl3_wa(engine, batch);
|
|
|
|
/* WaClearSlmSpaceAtContextSwitch:bdw,chv */
|
|
/* Actual scratch location is at 128 bytes offset */
|
|
batch = gen8_emit_pipe_control(batch,
|
|
PIPE_CONTROL_FLUSH_L3 |
|
|
PIPE_CONTROL_STORE_DATA_INDEX |
|
|
PIPE_CONTROL_CS_STALL |
|
|
PIPE_CONTROL_QW_WRITE,
|
|
LRC_PPHWSP_SCRATCH_ADDR);
|
|
|
|
*batch++ = MI_ARB_ON_OFF | MI_ARB_ENABLE;
|
|
|
|
/* Pad to end of cacheline */
|
|
while ((unsigned long)batch % CACHELINE_BYTES)
|
|
*batch++ = MI_NOOP;
|
|
|
|
/*
|
|
* MI_BATCH_BUFFER_END is not required in Indirect ctx BB because
|
|
* execution depends on the length specified in terms of cache lines
|
|
* in the register CTX_RCS_INDIRECT_CTX
|
|
*/
|
|
|
|
return batch;
|
|
}
|
|
|
|
struct lri {
|
|
i915_reg_t reg;
|
|
u32 value;
|
|
};
|
|
|
|
static u32 *emit_lri(u32 *batch, const struct lri *lri, unsigned int count)
|
|
{
|
|
GEM_BUG_ON(!count || count > 63);
|
|
|
|
*batch++ = MI_LOAD_REGISTER_IMM(count);
|
|
do {
|
|
*batch++ = i915_mmio_reg_offset(lri->reg);
|
|
*batch++ = lri->value;
|
|
} while (lri++, --count);
|
|
*batch++ = MI_NOOP;
|
|
|
|
return batch;
|
|
}
|
|
|
|
static u32 *gen9_init_indirectctx_bb(struct intel_engine_cs *engine, u32 *batch)
|
|
{
|
|
static const struct lri lri[] = {
|
|
/* WaDisableGatherAtSetShaderCommonSlice:skl,bxt,kbl,glk */
|
|
{
|
|
COMMON_SLICE_CHICKEN2,
|
|
__MASKED_FIELD(GEN9_DISABLE_GATHER_AT_SET_SHADER_COMMON_SLICE,
|
|
0),
|
|
},
|
|
|
|
/* BSpec: 11391 */
|
|
{
|
|
FF_SLICE_CHICKEN,
|
|
__MASKED_FIELD(FF_SLICE_CHICKEN_CL_PROVOKING_VERTEX_FIX,
|
|
FF_SLICE_CHICKEN_CL_PROVOKING_VERTEX_FIX),
|
|
},
|
|
|
|
/* BSpec: 11299 */
|
|
{
|
|
_3D_CHICKEN3,
|
|
__MASKED_FIELD(_3D_CHICKEN_SF_PROVOKING_VERTEX_FIX,
|
|
_3D_CHICKEN_SF_PROVOKING_VERTEX_FIX),
|
|
}
|
|
};
|
|
|
|
*batch++ = MI_ARB_ON_OFF | MI_ARB_DISABLE;
|
|
|
|
/* WaFlushCoherentL3CacheLinesAtContextSwitch:skl,bxt,glk */
|
|
batch = gen8_emit_flush_coherentl3_wa(engine, batch);
|
|
|
|
/* WaClearSlmSpaceAtContextSwitch:skl,bxt,kbl,glk,cfl */
|
|
batch = gen8_emit_pipe_control(batch,
|
|
PIPE_CONTROL_FLUSH_L3 |
|
|
PIPE_CONTROL_STORE_DATA_INDEX |
|
|
PIPE_CONTROL_CS_STALL |
|
|
PIPE_CONTROL_QW_WRITE,
|
|
LRC_PPHWSP_SCRATCH_ADDR);
|
|
|
|
batch = emit_lri(batch, lri, ARRAY_SIZE(lri));
|
|
|
|
/* WaMediaPoolStateCmdInWABB:bxt,glk */
|
|
if (HAS_POOLED_EU(engine->i915)) {
|
|
/*
|
|
* EU pool configuration is setup along with golden context
|
|
* during context initialization. This value depends on
|
|
* device type (2x6 or 3x6) and needs to be updated based
|
|
* on which subslice is disabled especially for 2x6
|
|
* devices, however it is safe to load default
|
|
* configuration of 3x6 device instead of masking off
|
|
* corresponding bits because HW ignores bits of a disabled
|
|
* subslice and drops down to appropriate config. Please
|
|
* see render_state_setup() in i915_gem_render_state.c for
|
|
* possible configurations, to avoid duplication they are
|
|
* not shown here again.
|
|
*/
|
|
*batch++ = GEN9_MEDIA_POOL_STATE;
|
|
*batch++ = GEN9_MEDIA_POOL_ENABLE;
|
|
*batch++ = 0x00777000;
|
|
*batch++ = 0;
|
|
*batch++ = 0;
|
|
*batch++ = 0;
|
|
}
|
|
|
|
*batch++ = MI_ARB_ON_OFF | MI_ARB_ENABLE;
|
|
|
|
/* Pad to end of cacheline */
|
|
while ((unsigned long)batch % CACHELINE_BYTES)
|
|
*batch++ = MI_NOOP;
|
|
|
|
return batch;
|
|
}
|
|
|
|
static u32 *
|
|
gen10_init_indirectctx_bb(struct intel_engine_cs *engine, u32 *batch)
|
|
{
|
|
int i;
|
|
|
|
/*
|
|
* WaPipeControlBefore3DStateSamplePattern: cnl
|
|
*
|
|
* Ensure the engine is idle prior to programming a
|
|
* 3DSTATE_SAMPLE_PATTERN during a context restore.
|
|
*/
|
|
batch = gen8_emit_pipe_control(batch,
|
|
PIPE_CONTROL_CS_STALL,
|
|
0);
|
|
/*
|
|
* WaPipeControlBefore3DStateSamplePattern says we need 4 dwords for
|
|
* the PIPE_CONTROL followed by 12 dwords of 0x0, so 16 dwords in
|
|
* total. However, a PIPE_CONTROL is 6 dwords long, not 4, which is
|
|
* confusing. Since gen8_emit_pipe_control() already advances the
|
|
* batch by 6 dwords, we advance the other 10 here, completing a
|
|
* cacheline. It's not clear if the workaround requires this padding
|
|
* before other commands, or if it's just the regular padding we would
|
|
* already have for the workaround bb, so leave it here for now.
|
|
*/
|
|
for (i = 0; i < 10; i++)
|
|
*batch++ = MI_NOOP;
|
|
|
|
/* Pad to end of cacheline */
|
|
while ((unsigned long)batch % CACHELINE_BYTES)
|
|
*batch++ = MI_NOOP;
|
|
|
|
return batch;
|
|
}
|
|
|
|
#define CTX_WA_BB_SIZE (PAGE_SIZE)
|
|
|
|
static int lrc_create_wa_ctx(struct intel_engine_cs *engine)
|
|
{
|
|
struct drm_i915_gem_object *obj;
|
|
struct i915_vma *vma;
|
|
int err;
|
|
|
|
obj = i915_gem_object_create_shmem(engine->i915, CTX_WA_BB_SIZE);
|
|
if (IS_ERR(obj))
|
|
return PTR_ERR(obj);
|
|
|
|
vma = i915_vma_instance(obj, &engine->gt->ggtt->vm, NULL);
|
|
if (IS_ERR(vma)) {
|
|
err = PTR_ERR(vma);
|
|
goto err;
|
|
}
|
|
|
|
engine->wa_ctx.vma = vma;
|
|
return 0;
|
|
|
|
err:
|
|
i915_gem_object_put(obj);
|
|
return err;
|
|
}
|
|
|
|
void lrc_fini_wa_ctx(struct intel_engine_cs *engine)
|
|
{
|
|
i915_vma_unpin_and_release(&engine->wa_ctx.vma, 0);
|
|
}
|
|
|
|
typedef u32 *(*wa_bb_func_t)(struct intel_engine_cs *engine, u32 *batch);
|
|
|
|
void lrc_init_wa_ctx(struct intel_engine_cs *engine)
|
|
{
|
|
struct i915_ctx_workarounds *wa_ctx = &engine->wa_ctx;
|
|
struct i915_wa_ctx_bb *wa_bb[] = {
|
|
&wa_ctx->indirect_ctx, &wa_ctx->per_ctx
|
|
};
|
|
wa_bb_func_t wa_bb_fn[ARRAY_SIZE(wa_bb)];
|
|
struct i915_gem_ww_ctx ww;
|
|
void *batch, *batch_ptr;
|
|
unsigned int i;
|
|
int err;
|
|
|
|
if (engine->class != RENDER_CLASS)
|
|
return;
|
|
|
|
switch (GRAPHICS_VER(engine->i915)) {
|
|
case 12:
|
|
case 11:
|
|
return;
|
|
case 10:
|
|
wa_bb_fn[0] = gen10_init_indirectctx_bb;
|
|
wa_bb_fn[1] = NULL;
|
|
break;
|
|
case 9:
|
|
wa_bb_fn[0] = gen9_init_indirectctx_bb;
|
|
wa_bb_fn[1] = NULL;
|
|
break;
|
|
case 8:
|
|
wa_bb_fn[0] = gen8_init_indirectctx_bb;
|
|
wa_bb_fn[1] = NULL;
|
|
break;
|
|
default:
|
|
MISSING_CASE(GRAPHICS_VER(engine->i915));
|
|
return;
|
|
}
|
|
|
|
err = lrc_create_wa_ctx(engine);
|
|
if (err) {
|
|
/*
|
|
* We continue even if we fail to initialize WA batch
|
|
* because we only expect rare glitches but nothing
|
|
* critical to prevent us from using GPU
|
|
*/
|
|
drm_err(&engine->i915->drm,
|
|
"Ignoring context switch w/a allocation error:%d\n",
|
|
err);
|
|
return;
|
|
}
|
|
|
|
if (!engine->wa_ctx.vma)
|
|
return;
|
|
|
|
i915_gem_ww_ctx_init(&ww, true);
|
|
retry:
|
|
err = i915_gem_object_lock(wa_ctx->vma->obj, &ww);
|
|
if (!err)
|
|
err = i915_ggtt_pin(wa_ctx->vma, &ww, 0, PIN_HIGH);
|
|
if (err)
|
|
goto err;
|
|
|
|
batch = i915_gem_object_pin_map(wa_ctx->vma->obj, I915_MAP_WB);
|
|
if (IS_ERR(batch)) {
|
|
err = PTR_ERR(batch);
|
|
goto err_unpin;
|
|
}
|
|
|
|
/*
|
|
* Emit the two workaround batch buffers, recording the offset from the
|
|
* start of the workaround batch buffer object for each and their
|
|
* respective sizes.
|
|
*/
|
|
batch_ptr = batch;
|
|
for (i = 0; i < ARRAY_SIZE(wa_bb_fn); i++) {
|
|
wa_bb[i]->offset = batch_ptr - batch;
|
|
if (GEM_DEBUG_WARN_ON(!IS_ALIGNED(wa_bb[i]->offset,
|
|
CACHELINE_BYTES))) {
|
|
err = -EINVAL;
|
|
break;
|
|
}
|
|
if (wa_bb_fn[i])
|
|
batch_ptr = wa_bb_fn[i](engine, batch_ptr);
|
|
wa_bb[i]->size = batch_ptr - (batch + wa_bb[i]->offset);
|
|
}
|
|
GEM_BUG_ON(batch_ptr - batch > CTX_WA_BB_SIZE);
|
|
|
|
__i915_gem_object_flush_map(wa_ctx->vma->obj, 0, batch_ptr - batch);
|
|
__i915_gem_object_release_map(wa_ctx->vma->obj);
|
|
|
|
/* Verify that we can handle failure to setup the wa_ctx */
|
|
if (!err)
|
|
err = i915_inject_probe_error(engine->i915, -ENODEV);
|
|
|
|
err_unpin:
|
|
if (err)
|
|
i915_vma_unpin(wa_ctx->vma);
|
|
err:
|
|
if (err == -EDEADLK) {
|
|
err = i915_gem_ww_ctx_backoff(&ww);
|
|
if (!err)
|
|
goto retry;
|
|
}
|
|
i915_gem_ww_ctx_fini(&ww);
|
|
|
|
if (err) {
|
|
i915_vma_put(engine->wa_ctx.vma);
|
|
|
|
/* Clear all flags to prevent further use */
|
|
memset(wa_ctx, 0, sizeof(*wa_ctx));
|
|
}
|
|
}
|
|
|
|
static void st_update_runtime_underflow(struct intel_context *ce, s32 dt)
|
|
{
|
|
#if IS_ENABLED(CONFIG_DRM_I915_SELFTEST)
|
|
ce->runtime.num_underflow++;
|
|
ce->runtime.max_underflow = max_t(u32, ce->runtime.max_underflow, -dt);
|
|
#endif
|
|
}
|
|
|
|
void lrc_update_runtime(struct intel_context *ce)
|
|
{
|
|
u32 old;
|
|
s32 dt;
|
|
|
|
if (intel_context_is_barrier(ce))
|
|
return;
|
|
|
|
old = ce->runtime.last;
|
|
ce->runtime.last = lrc_get_runtime(ce);
|
|
dt = ce->runtime.last - old;
|
|
|
|
if (unlikely(dt < 0)) {
|
|
CE_TRACE(ce, "runtime underflow: last=%u, new=%u, delta=%d\n",
|
|
old, ce->runtime.last, dt);
|
|
st_update_runtime_underflow(ce, dt);
|
|
return;
|
|
}
|
|
|
|
ewma_runtime_add(&ce->runtime.avg, dt);
|
|
ce->runtime.total += dt;
|
|
}
|
|
|
|
#if IS_ENABLED(CONFIG_DRM_I915_SELFTEST)
|
|
#include "selftest_lrc.c"
|
|
#endif
|