Skip to content

Commit

Permalink
drm/i915/icl: Use hw engine class, instance to find irq handler
Browse files Browse the repository at this point in the history
Interrupt identity register we already read from hardware
contains engine class and instance fields. Leverage
these fields to find correct engine to handle the interrupt.

v3: rebase on top of rps intr
    use correct class / instance limits (Michel)
v4: split engine/other handling
v5: empty iir is not err (Daniele, Michel)

Suggested-by: Daniele Ceraolo Spurio <[email protected]>
Cc: Daniele Ceraolo Spurio <[email protected]>
Cc: Chris Wilson <[email protected]>
Cc: Tvrtko Ursulin <[email protected]>
Cc: Michel Thierry <[email protected]>
Signed-off-by: Mika Kuoppala <[email protected]>
Reviewed-by: Daniele Ceraolo Spurio <[email protected]>
Reviewed-by: Michel Thierry <[email protected]>
Link: https://patchwork.freedesktop.org/patch/msgid/[email protected]
  • Loading branch information
mikuint committed Apr 6, 2018
1 parent e34b034 commit f744dbc
Show file tree
Hide file tree
Showing 2 changed files with 56 additions and 47 deletions.
99 changes: 53 additions & 46 deletions drivers/gpu/drm/i915/i915_irq.c
Original file line number Diff line number Diff line change
Expand Up @@ -2732,47 +2732,9 @@ static void __fini_wedge(struct wedge_me *w)
(W)->i915; \
__fini_wedge((W)))

static void
gen11_gt_engine_irq_handler(struct drm_i915_private * const i915,
const unsigned int bank,
const unsigned int engine_n,
const u16 iir)
{
struct intel_engine_cs ** const engine = i915->engine;

switch (bank) {
case 0:
switch (engine_n) {

case GEN11_RCS0:
return gen8_cs_irq_handler(engine[RCS], iir);

case GEN11_BCS:
return gen8_cs_irq_handler(engine[BCS], iir);
}
case 1:
switch (engine_n) {

case GEN11_VCS(0):
return gen8_cs_irq_handler(engine[_VCS(0)], iir);
case GEN11_VCS(1):
return gen8_cs_irq_handler(engine[_VCS(1)], iir);
case GEN11_VCS(2):
return gen8_cs_irq_handler(engine[_VCS(2)], iir);
case GEN11_VCS(3):
return gen8_cs_irq_handler(engine[_VCS(3)], iir);

case GEN11_VECS(0):
return gen8_cs_irq_handler(engine[_VECS(0)], iir);
case GEN11_VECS(1):
return gen8_cs_irq_handler(engine[_VECS(1)], iir);
}
}
}

static u32
gen11_gt_engine_intr(struct drm_i915_private * const i915,
const unsigned int bank, const unsigned int bit)
gen11_gt_engine_identity(struct drm_i915_private * const i915,
const unsigned int bank, const unsigned int bit)
{
void __iomem * const regs = i915->regs;
u32 timeout_ts;
Expand All @@ -2799,7 +2761,54 @@ gen11_gt_engine_intr(struct drm_i915_private * const i915,
raw_reg_write(regs, GEN11_INTR_IDENTITY_REG(bank),
GEN11_INTR_DATA_VALID);

return ident & GEN11_INTR_ENGINE_MASK;
return ident;
}

static void
gen11_other_irq_handler(struct drm_i915_private * const i915,
const u8 instance, const u16 iir)
{
WARN_ONCE(1, "unhandled other interrupt instance=0x%x, iir=0x%x\n",
instance, iir);
}

static void
gen11_engine_irq_handler(struct drm_i915_private * const i915,
const u8 class, const u8 instance, const u16 iir)
{
struct intel_engine_cs *engine;

if (instance <= MAX_ENGINE_INSTANCE)
engine = i915->engine_class[class][instance];
else
engine = NULL;

if (likely(engine))
return gen8_cs_irq_handler(engine, iir);

WARN_ONCE(1, "unhandled engine interrupt class=0x%x, instance=0x%x\n",
class, instance);
}

static void
gen11_gt_identity_handler(struct drm_i915_private * const i915,
const u32 identity)
{
const u8 class = GEN11_INTR_ENGINE_CLASS(identity);
const u8 instance = GEN11_INTR_ENGINE_INSTANCE(identity);
const u16 intr = GEN11_INTR_ENGINE_INTR(identity);

if (unlikely(!intr))
return;

if (class <= COPY_ENGINE_CLASS)
return gen11_engine_irq_handler(i915, class, instance, intr);

if (class == OTHER_CLASS)
return gen11_other_irq_handler(i915, instance, intr);

WARN_ONCE(1, "unknown interrupt class=0x%x, instance=0x%x, intr=0x%x\n",
class, instance, intr);
}

static void
Expand All @@ -2824,12 +2833,10 @@ gen11_gt_irq_handler(struct drm_i915_private * const i915,
}

for_each_set_bit(bit, &intr_dw, 32) {
const u16 iir = gen11_gt_engine_intr(i915, bank, bit);

if (unlikely(!iir))
continue;
const u32 ident = gen11_gt_engine_identity(i915,
bank, bit);

gen11_gt_engine_irq_handler(i915, bank, bit, iir);
gen11_gt_identity_handler(i915, ident);
}

/* Clear must be after shared has been served for engine */
Expand Down
4 changes: 3 additions & 1 deletion drivers/gpu/drm/i915/i915_reg.h
Original file line number Diff line number Diff line change
Expand Up @@ -6998,7 +6998,9 @@ enum {
#define GEN11_INTR_IDENTITY_REG0 _MMIO(0x190060)
#define GEN11_INTR_IDENTITY_REG1 _MMIO(0x190064)
#define GEN11_INTR_DATA_VALID (1 << 31)
#define GEN11_INTR_ENGINE_MASK (0xffff)
#define GEN11_INTR_ENGINE_CLASS(x) (((x) & GENMASK(18, 16)) >> 16)
#define GEN11_INTR_ENGINE_INSTANCE(x) (((x) & GENMASK(25, 20)) >> 20)
#define GEN11_INTR_ENGINE_INTR(x) ((x) & 0xffff)

#define GEN11_INTR_IDENTITY_REG(x) _MMIO(0x190060 + (x * 4))

Expand Down

0 comments on commit f744dbc

Please sign in to comment.