Commit eefac38a authored by Matt Roper's avatar Matt Roper Committed by Joonas Lahtinen

drm/i915/uncore: Add GSI offset to uncore

GT non-engine registers (referred to as "GSI" registers by the spec)
have the same relative offsets on standalone media as they do on the
primary GT, just with an additional "GSI offset" added to their MMIO
address.  If we store this GSI offset in the standalone media's
intel_uncore structure, it can be automatically applied to all GSI reg
reads/writes that happen on that GT, allowing us to re-use our existing
GT code with minimal changes.

Forcewake and shadowed register tables for the media GT (which will be
added in a future patch) are listed as final addresses that already
include the GSI offset, so we also need to add the GSI offset before
doing lookups of registers in one of those tables.

v2:
 - Add comment on raw_reg_*() macros explaining why we don't bother with
   GSI offsets in them.  (Daniele)

Cc: Daniele Ceraolo Spurio <daniele.ceraolospurio@intel.com>
Signed-off-by: default avatarMatt Roper <matthew.d.roper@intel.com>
Reviewed-by: default avatarDaniele Ceraolo Spurio <daniele.ceraolospurio@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20220908224550.821257-1-matthew.d.roper@intel.comSigned-off-by: default avatarJoonas Lahtinen <joonas.lahtinen@linux.intel.com>
parent 1c66a12a
...@@ -274,6 +274,7 @@ struct intel_gt_definition { ...@@ -274,6 +274,7 @@ struct intel_gt_definition {
enum intel_gt_type type; enum intel_gt_type type;
char *name; char *name;
u32 mapping_base; u32 mapping_base;
u32 gsi_offset;
intel_engine_mask_t engine_mask; intel_engine_mask_t engine_mask;
}; };
......
...@@ -928,6 +928,9 @@ find_fw_domain(struct intel_uncore *uncore, u32 offset) ...@@ -928,6 +928,9 @@ find_fw_domain(struct intel_uncore *uncore, u32 offset)
{ {
const struct intel_forcewake_range *entry; const struct intel_forcewake_range *entry;
if (IS_GSI_REG(offset))
offset += uncore->gsi_offset;
entry = BSEARCH(offset, entry = BSEARCH(offset,
uncore->fw_domains_table, uncore->fw_domains_table,
uncore->fw_domains_table_entries, uncore->fw_domains_table_entries,
...@@ -1143,6 +1146,9 @@ static bool is_shadowed(struct intel_uncore *uncore, u32 offset) ...@@ -1143,6 +1146,9 @@ static bool is_shadowed(struct intel_uncore *uncore, u32 offset)
if (drm_WARN_ON(&uncore->i915->drm, !uncore->shadowed_reg_table)) if (drm_WARN_ON(&uncore->i915->drm, !uncore->shadowed_reg_table))
return false; return false;
if (IS_GSI_REG(offset))
offset += uncore->gsi_offset;
return BSEARCH(offset, return BSEARCH(offset,
uncore->shadowed_reg_table, uncore->shadowed_reg_table,
uncore->shadowed_reg_table_entries, uncore->shadowed_reg_table_entries,
...@@ -1995,8 +2001,8 @@ static int __fw_domain_init(struct intel_uncore *uncore, ...@@ -1995,8 +2001,8 @@ static int __fw_domain_init(struct intel_uncore *uncore,
d->uncore = uncore; d->uncore = uncore;
d->wake_count = 0; d->wake_count = 0;
d->reg_set = uncore->regs + i915_mmio_reg_offset(reg_set); d->reg_set = uncore->regs + i915_mmio_reg_offset(reg_set) + uncore->gsi_offset;
d->reg_ack = uncore->regs + i915_mmio_reg_offset(reg_ack); d->reg_ack = uncore->regs + i915_mmio_reg_offset(reg_ack) + uncore->gsi_offset;
d->id = domain_id; d->id = domain_id;
......
...@@ -136,6 +136,16 @@ struct intel_uncore { ...@@ -136,6 +136,16 @@ struct intel_uncore {
spinlock_t lock; /** lock is also taken in irq contexts. */ spinlock_t lock; /** lock is also taken in irq contexts. */
/*
* Do we need to apply an additional offset to reach the beginning
* of the basic non-engine GT registers (referred to as "GSI" on
* newer platforms, or "GT block" on older platforms)? If so, we'll
* track that here and apply it transparently to registers in the
* appropriate range to maintain compatibility with our existing
* register definitions and GT code.
*/
u32 gsi_offset;
unsigned int flags; unsigned int flags;
#define UNCORE_HAS_FORCEWAKE BIT(0) #define UNCORE_HAS_FORCEWAKE BIT(0)
#define UNCORE_HAS_FPGA_DBG_UNCLAIMED BIT(1) #define UNCORE_HAS_FPGA_DBG_UNCLAIMED BIT(1)
...@@ -294,19 +304,27 @@ intel_wait_for_register_fw(struct intel_uncore *uncore, ...@@ -294,19 +304,27 @@ intel_wait_for_register_fw(struct intel_uncore *uncore,
2, timeout_ms, NULL); 2, timeout_ms, NULL);
} }
#define IS_GSI_REG(reg) ((reg) < 0x40000)
/* register access functions */ /* register access functions */
#define __raw_read(x__, s__) \ #define __raw_read(x__, s__) \
static inline u##x__ __raw_uncore_read##x__(const struct intel_uncore *uncore, \ static inline u##x__ __raw_uncore_read##x__(const struct intel_uncore *uncore, \
i915_reg_t reg) \ i915_reg_t reg) \
{ \ { \
return read##s__(uncore->regs + i915_mmio_reg_offset(reg)); \ u32 offset = i915_mmio_reg_offset(reg); \
if (IS_GSI_REG(offset)) \
offset += uncore->gsi_offset; \
return read##s__(uncore->regs + offset); \
} }
#define __raw_write(x__, s__) \ #define __raw_write(x__, s__) \
static inline void __raw_uncore_write##x__(const struct intel_uncore *uncore, \ static inline void __raw_uncore_write##x__(const struct intel_uncore *uncore, \
i915_reg_t reg, u##x__ val) \ i915_reg_t reg, u##x__ val) \
{ \ { \
write##s__(val, uncore->regs + i915_mmio_reg_offset(reg)); \ u32 offset = i915_mmio_reg_offset(reg); \
if (IS_GSI_REG(offset)) \
offset += uncore->gsi_offset; \
write##s__(val, uncore->regs + offset); \
} }
__raw_read(8, b) __raw_read(8, b)
__raw_read(16, w) __raw_read(16, w)
...@@ -447,6 +465,18 @@ static inline int intel_uncore_write_and_verify(struct intel_uncore *uncore, ...@@ -447,6 +465,18 @@ static inline int intel_uncore_write_and_verify(struct intel_uncore *uncore,
return (reg_val & mask) != expected_val ? -EINVAL : 0; return (reg_val & mask) != expected_val ? -EINVAL : 0;
} }
/*
* The raw_reg_{read,write} macros are intended as a micro-optimization for
* interrupt handlers so that the pointer indirection on uncore->regs can
* be computed once (and presumably cached in a register) instead of generating
* extra load instructions for each MMIO access.
*
* Given that these macros are only intended for non-GSI interrupt registers
* (and the goal is to avoid extra instructions generated by the compiler),
* these macros do not account for uncore->gsi_offset. Any caller that needs
* to use these macros on a GSI register is responsible for adding the
* appropriate GSI offset to the 'base' parameter.
*/
#define raw_reg_read(base, reg) \ #define raw_reg_read(base, reg) \
readl(base + i915_mmio_reg_offset(reg)) readl(base + i915_mmio_reg_offset(reg))
#define raw_reg_write(base, reg, value) \ #define raw_reg_write(base, reg, value) \
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment