Commit b87d3901 authored by Matt Roper's avatar Matt Roper

drm/i915/sseu: Disassociate internal subslice mask representation from uapi

As with EU masks, it's easier to store subslice/DSS masks internally in
a format that's more natural for the driver to work with, and then only
covert into the u8[] uapi form when the query ioctl is invoked.  Since
the hardware design changed significantly with Xe_HP, we'll use a union
to choose between the old "hsw-style" subslice masks or the newer xehp
mask.  HSW-style masks will be stored in an array of u8's, indexed by
slice (there's never more than 6 subslices per slice on older
platforms).  For Xe_HP and beyond where slices no longer exist, we only
need a single bitmask.  However we already know that this mask is
eventually going to grow too large for a simple u64 to hold, so we'll
represent it in a manner that can be operated on by the utilities in
linux/bitmap.h.

v2:
 - Fix typo: BIT(s) -> BIT(ss) in gen9_sseu_device_status()

v3:
 - Eliminate sseu->ss_stride and just calculate the stride while
   specifically handling uapi.  (Tvrtko)
 - Use BITMAP_BITS() macro to refer to size of masks rather than
   passing I915_MAX_SS_FUSE_BITS directly.  (Tvrtko)
 - Report compute/geometry DSS masks separately when dumping Xe_HP SSEU
   info.  (Tvrtko)
 - Restore dropped range checks to intel_sseu_has_subslice().  (Tvrtko)

v4:
 - Make the bitmap size macro check the size of the .xehp field rather
   than the containing union.  (Tvrtko)
 - Don't add GEM_BUG_ON() intel_sseu_has_subslice()'s check for whether
   slice or subslice ID exceed sseu->max_[sub]slices; various loops
   in the driver are expected to exceed these, so we should just
   silently return 'false.'

v5:
 - Move XEHP_BITMAP_BITS() to the header so that we can also replace a
   usage of I915_MAX_SS_FUSE_BITS in one of the inline functions.
   (Bala)
 - Change the local variable in intel_slicemask_from_xehp_dssmask() from
   u16 to 'unsigned long' to make it a bit more future-proof.

Cc: Tvrtko Ursulin <tvrtko.ursulin@linux.intel.com>
Cc: Balasubramani Vivekanandan <balasubramani.vivekanandan@intel.com>
Signed-off-by: default avatarMatt Roper <matthew.d.roper@intel.com>
Acked-by: default avatarTvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: default avatarBalasubramani Vivekanandan <balasubramani.vivekanandan@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20220601150725.521468-6-matthew.d.roper@intel.com
parent bc3c5e08
...@@ -1875,6 +1875,7 @@ i915_gem_user_to_context_sseu(struct intel_gt *gt, ...@@ -1875,6 +1875,7 @@ i915_gem_user_to_context_sseu(struct intel_gt *gt,
{ {
const struct sseu_dev_info *device = &gt->info.sseu; const struct sseu_dev_info *device = &gt->info.sseu;
struct drm_i915_private *i915 = gt->i915; struct drm_i915_private *i915 = gt->i915;
unsigned int dev_subslice_mask = intel_sseu_get_hsw_subslices(device, 0);
/* No zeros in any field. */ /* No zeros in any field. */
if (!user->slice_mask || !user->subslice_mask || if (!user->slice_mask || !user->subslice_mask ||
...@@ -1901,7 +1902,7 @@ i915_gem_user_to_context_sseu(struct intel_gt *gt, ...@@ -1901,7 +1902,7 @@ i915_gem_user_to_context_sseu(struct intel_gt *gt,
if (user->slice_mask & ~device->slice_mask) if (user->slice_mask & ~device->slice_mask)
return -EINVAL; return -EINVAL;
if (user->subslice_mask & ~device->subslice_mask[0]) if (user->subslice_mask & ~dev_subslice_mask)
return -EINVAL; return -EINVAL;
if (user->max_eus_per_subslice > device->max_eus_per_subslice) if (user->max_eus_per_subslice > device->max_eus_per_subslice)
...@@ -1915,7 +1916,7 @@ i915_gem_user_to_context_sseu(struct intel_gt *gt, ...@@ -1915,7 +1916,7 @@ i915_gem_user_to_context_sseu(struct intel_gt *gt,
/* Part specific restrictions. */ /* Part specific restrictions. */
if (GRAPHICS_VER(i915) == 11) { if (GRAPHICS_VER(i915) == 11) {
unsigned int hw_s = hweight8(device->slice_mask); unsigned int hw_s = hweight8(device->slice_mask);
unsigned int hw_ss_per_s = hweight8(device->subslice_mask[0]); unsigned int hw_ss_per_s = hweight8(dev_subslice_mask);
unsigned int req_s = hweight8(context->slice_mask); unsigned int req_s = hweight8(context->slice_mask);
unsigned int req_ss = hweight8(context->subslice_mask); unsigned int req_ss = hweight8(context->subslice_mask);
......
...@@ -674,8 +674,8 @@ static void engine_mask_apply_compute_fuses(struct intel_gt *gt) ...@@ -674,8 +674,8 @@ static void engine_mask_apply_compute_fuses(struct intel_gt *gt)
if (GRAPHICS_VER_FULL(i915) < IP_VER(12, 50)) if (GRAPHICS_VER_FULL(i915) < IP_VER(12, 50))
return; return;
ccs_mask = intel_slicemask_from_dssmask(intel_sseu_get_compute_subslices(&info->sseu), ccs_mask = intel_slicemask_from_xehp_dssmask(info->sseu.compute_subslice_mask,
ss_per_ccs); ss_per_ccs);
/* /*
* If all DSS in a quadrant are fused off, the corresponding CCS * If all DSS in a quadrant are fused off, the corresponding CCS
* engine is not available for use. * engine is not available for use.
......
...@@ -133,13 +133,6 @@ static const struct intel_mmio_range dg2_lncf_steering_table[] = { ...@@ -133,13 +133,6 @@ static const struct intel_mmio_range dg2_lncf_steering_table[] = {
{}, {},
}; };
static u16 slicemask(struct intel_gt *gt, int count)
{
u64 dss_mask = intel_sseu_get_subslices(&gt->info.sseu, 0);
return intel_slicemask_from_dssmask(dss_mask, count);
}
int intel_gt_init_mmio(struct intel_gt *gt) int intel_gt_init_mmio(struct intel_gt *gt)
{ {
struct drm_i915_private *i915 = gt->i915; struct drm_i915_private *i915 = gt->i915;
...@@ -155,9 +148,12 @@ int intel_gt_init_mmio(struct intel_gt *gt) ...@@ -155,9 +148,12 @@ int intel_gt_init_mmio(struct intel_gt *gt)
*/ */
if (HAS_MSLICES(i915)) { if (HAS_MSLICES(i915)) {
gt->info.mslice_mask = gt->info.mslice_mask =
slicemask(gt, GEN_DSS_PER_MSLICE) | intel_slicemask_from_xehp_dssmask(gt->info.sseu.subslice_mask,
GEN_DSS_PER_MSLICE);
gt->info.mslice_mask |=
(intel_uncore_read(gt->uncore, GEN10_MIRROR_FUSE3) & (intel_uncore_read(gt->uncore, GEN10_MIRROR_FUSE3) &
GEN12_MEML3_EN_MASK); GEN12_MEML3_EN_MASK);
if (!gt->info.mslice_mask) /* should be impossible! */ if (!gt->info.mslice_mask) /* should be impossible! */
drm_warn(&i915->drm, "mslice mask all zero!\n"); drm_warn(&i915->drm, "mslice mask all zero!\n");
} }
......
This diff is collapsed.
...@@ -25,12 +25,16 @@ struct drm_printer; ...@@ -25,12 +25,16 @@ struct drm_printer;
/* /*
* Maximum number of subslices that can exist within a HSW-style slice. This * Maximum number of subslices that can exist within a HSW-style slice. This
* is only relevant to pre-Xe_HP platforms (Xe_HP and beyond use the * is only relevant to pre-Xe_HP platforms (Xe_HP and beyond use the
* GEN_MAX_DSS value below). * I915_MAX_SS_FUSE_BITS value below).
*/ */
#define GEN_MAX_SS_PER_HSW_SLICE 6 #define GEN_MAX_SS_PER_HSW_SLICE 6
/* Maximum number of DSS on newer platforms (Xe_HP and beyond). */ /*
#define GEN_MAX_DSS 32 * Maximum number of 32-bit registers used by hardware to express the
* enabled/disabled subslices.
*/
#define I915_MAX_SS_FUSE_REGS 1
#define I915_MAX_SS_FUSE_BITS (I915_MAX_SS_FUSE_REGS * 32)
/* Maximum number of EUs that can exist within a subslice or DSS. */ /* Maximum number of EUs that can exist within a subslice or DSS. */
#define GEN_MAX_EUS_PER_SS 16 #define GEN_MAX_EUS_PER_SS 16
...@@ -38,7 +42,7 @@ struct drm_printer; ...@@ -38,7 +42,7 @@ struct drm_printer;
#define SSEU_MAX(a, b) ((a) > (b) ? (a) : (b)) #define SSEU_MAX(a, b) ((a) > (b) ? (a) : (b))
/* The maximum number of bits needed to express each subslice/DSS independently */ /* The maximum number of bits needed to express each subslice/DSS independently */
#define GEN_SS_MASK_SIZE SSEU_MAX(GEN_MAX_DSS, \ #define GEN_SS_MASK_SIZE SSEU_MAX(I915_MAX_SS_FUSE_BITS, \
GEN_MAX_HSW_SLICES * GEN_MAX_SS_PER_HSW_SLICE) GEN_MAX_HSW_SLICES * GEN_MAX_SS_PER_HSW_SLICE)
#define GEN_SSEU_STRIDE(max_entries) DIV_ROUND_UP(max_entries, BITS_PER_BYTE) #define GEN_SSEU_STRIDE(max_entries) DIV_ROUND_UP(max_entries, BITS_PER_BYTE)
...@@ -49,17 +53,26 @@ struct drm_printer; ...@@ -49,17 +53,26 @@ struct drm_printer;
#define GEN_DSS_PER_CSLICE 8 #define GEN_DSS_PER_CSLICE 8
#define GEN_DSS_PER_MSLICE 8 #define GEN_DSS_PER_MSLICE 8
#define GEN_MAX_GSLICES (GEN_MAX_DSS / GEN_DSS_PER_GSLICE) #define GEN_MAX_GSLICES (I915_MAX_SS_FUSE_BITS / GEN_DSS_PER_GSLICE)
#define GEN_MAX_CSLICES (GEN_MAX_DSS / GEN_DSS_PER_CSLICE) #define GEN_MAX_CSLICES (I915_MAX_SS_FUSE_BITS / GEN_DSS_PER_CSLICE)
typedef union {
u8 hsw[GEN_MAX_HSW_SLICES];
/* Bitmap compatible with linux/bitmap.h; may exceed size of u64 */
unsigned long xehp[BITS_TO_LONGS(I915_MAX_SS_FUSE_BITS)];
} intel_sseu_ss_mask_t;
#define XEHP_BITMAP_BITS(mask) ((int)BITS_PER_TYPE(typeof(mask.xehp)))
struct sseu_dev_info { struct sseu_dev_info {
u8 slice_mask; u8 slice_mask;
u8 subslice_mask[GEN_SS_MASK_SIZE]; intel_sseu_ss_mask_t subslice_mask;
u8 geometry_subslice_mask[GEN_SS_MASK_SIZE]; intel_sseu_ss_mask_t geometry_subslice_mask;
u8 compute_subslice_mask[GEN_SS_MASK_SIZE]; intel_sseu_ss_mask_t compute_subslice_mask;
union { union {
u16 hsw[GEN_MAX_HSW_SLICES][GEN_MAX_SS_PER_HSW_SLICE]; u16 hsw[GEN_MAX_HSW_SLICES][GEN_MAX_SS_PER_HSW_SLICE];
u16 xehp[GEN_MAX_DSS]; u16 xehp[I915_MAX_SS_FUSE_BITS];
} eu_mask; } eu_mask;
u16 eu_total; u16 eu_total;
...@@ -80,8 +93,6 @@ struct sseu_dev_info { ...@@ -80,8 +93,6 @@ struct sseu_dev_info {
u8 max_slices; u8 max_slices;
u8 max_subslices; u8 max_subslices;
u8 max_eus_per_subslice; u8 max_eus_per_subslice;
u8 ss_stride;
}; };
/* /*
...@@ -99,7 +110,7 @@ intel_sseu_from_device_info(const struct sseu_dev_info *sseu) ...@@ -99,7 +110,7 @@ intel_sseu_from_device_info(const struct sseu_dev_info *sseu)
{ {
struct intel_sseu value = { struct intel_sseu value = {
.slice_mask = sseu->slice_mask, .slice_mask = sseu->slice_mask,
.subslice_mask = sseu->subslice_mask[0], .subslice_mask = sseu->subslice_mask.hsw[0],
.min_eus_per_subslice = sseu->max_eus_per_subslice, .min_eus_per_subslice = sseu->max_eus_per_subslice,
.max_eus_per_subslice = sseu->max_eus_per_subslice, .max_eus_per_subslice = sseu->max_eus_per_subslice,
}; };
...@@ -111,18 +122,28 @@ static inline bool ...@@ -111,18 +122,28 @@ static inline bool
intel_sseu_has_subslice(const struct sseu_dev_info *sseu, int slice, intel_sseu_has_subslice(const struct sseu_dev_info *sseu, int slice,
int subslice) int subslice)
{ {
u8 mask;
int ss_idx = subslice / BITS_PER_BYTE;
if (slice >= sseu->max_slices || if (slice >= sseu->max_slices ||
subslice >= sseu->max_subslices) subslice >= sseu->max_subslices)
return false; return false;
GEM_BUG_ON(ss_idx >= sseu->ss_stride); if (sseu->has_xehp_dss)
return test_bit(subslice, sseu->subslice_mask.xehp);
mask = sseu->subslice_mask[slice * sseu->ss_stride + ss_idx]; else
return sseu->subslice_mask.hsw[slice] & BIT(subslice);
}
return mask & BIT(subslice % BITS_PER_BYTE); /*
* Used to obtain the index of the first DSS. Can start searching from the
* beginning of a specific dss group (e.g., gslice, cslice, etc.) if
* groupsize and groupnum are non-zero.
*/
static inline unsigned int
intel_sseu_find_first_xehp_dss(const struct sseu_dev_info *sseu, int groupsize,
int groupnum)
{
return find_next_bit(sseu->subslice_mask.xehp,
XEHP_BITMAP_BITS(sseu->subslice_mask),
groupnum * groupsize);
} }
void intel_sseu_set_info(struct sseu_dev_info *sseu, u8 max_slices, void intel_sseu_set_info(struct sseu_dev_info *sseu, u8 max_slices,
...@@ -132,14 +153,10 @@ unsigned int ...@@ -132,14 +153,10 @@ unsigned int
intel_sseu_subslice_total(const struct sseu_dev_info *sseu); intel_sseu_subslice_total(const struct sseu_dev_info *sseu);
unsigned int unsigned int
intel_sseu_subslices_per_slice(const struct sseu_dev_info *sseu, u8 slice); intel_sseu_get_hsw_subslices(const struct sseu_dev_info *sseu, u8 slice);
u32 intel_sseu_get_subslices(const struct sseu_dev_info *sseu, u8 slice); intel_sseu_ss_mask_t
intel_sseu_get_compute_subslices(const struct sseu_dev_info *sseu);
u32 intel_sseu_get_compute_subslices(const struct sseu_dev_info *sseu);
void intel_sseu_set_subslices(struct sseu_dev_info *sseu, int slice,
u8 *subslice_mask, u32 ss_mask);
void intel_sseu_info_init(struct intel_gt *gt); void intel_sseu_info_init(struct intel_gt *gt);
...@@ -151,9 +168,15 @@ void intel_sseu_print_topology(struct drm_i915_private *i915, ...@@ -151,9 +168,15 @@ void intel_sseu_print_topology(struct drm_i915_private *i915,
const struct sseu_dev_info *sseu, const struct sseu_dev_info *sseu,
struct drm_printer *p); struct drm_printer *p);
u16 intel_slicemask_from_dssmask(u64 dss_mask, int dss_per_slice); u16 intel_slicemask_from_xehp_dssmask(intel_sseu_ss_mask_t dss_mask, int dss_per_slice);
int intel_sseu_copy_eumask_to_user(void __user *to, int intel_sseu_copy_eumask_to_user(void __user *to,
const struct sseu_dev_info *sseu); const struct sseu_dev_info *sseu);
int intel_sseu_copy_ssmask_to_user(void __user *to,
const struct sseu_dev_info *sseu);
void intel_sseu_print_ss_info(const char *type,
const struct sseu_dev_info *sseu,
struct seq_file *m);
#endif /* __INTEL_SSEU_H__ */ #endif /* __INTEL_SSEU_H__ */
...@@ -4,6 +4,7 @@ ...@@ -4,6 +4,7 @@
* Copyright © 2020 Intel Corporation * Copyright © 2020 Intel Corporation
*/ */
#include <linux/bitmap.h>
#include <linux/string_helpers.h> #include <linux/string_helpers.h>
#include "i915_drv.h" #include "i915_drv.h"
...@@ -11,14 +12,6 @@ ...@@ -11,14 +12,6 @@
#include "intel_gt_regs.h" #include "intel_gt_regs.h"
#include "intel_sseu_debugfs.h" #include "intel_sseu_debugfs.h"
static void sseu_copy_subslices(const struct sseu_dev_info *sseu,
int slice, u8 *to_mask)
{
int offset = slice * sseu->ss_stride;
memcpy(&to_mask[offset], &sseu->subslice_mask[offset], sseu->ss_stride);
}
static void cherryview_sseu_device_status(struct intel_gt *gt, static void cherryview_sseu_device_status(struct intel_gt *gt,
struct sseu_dev_info *sseu) struct sseu_dev_info *sseu)
{ {
...@@ -41,7 +34,7 @@ static void cherryview_sseu_device_status(struct intel_gt *gt, ...@@ -41,7 +34,7 @@ static void cherryview_sseu_device_status(struct intel_gt *gt,
continue; continue;
sseu->slice_mask = BIT(0); sseu->slice_mask = BIT(0);
sseu->subslice_mask[0] |= BIT(ss); sseu->subslice_mask.hsw[0] |= BIT(ss);
eu_cnt = ((sig1[ss] & CHV_EU08_PG_ENABLE) ? 0 : 2) + eu_cnt = ((sig1[ss] & CHV_EU08_PG_ENABLE) ? 0 : 2) +
((sig1[ss] & CHV_EU19_PG_ENABLE) ? 0 : 2) + ((sig1[ss] & CHV_EU19_PG_ENABLE) ? 0 : 2) +
((sig1[ss] & CHV_EU210_PG_ENABLE) ? 0 : 2) + ((sig1[ss] & CHV_EU210_PG_ENABLE) ? 0 : 2) +
...@@ -92,7 +85,7 @@ static void gen11_sseu_device_status(struct intel_gt *gt, ...@@ -92,7 +85,7 @@ static void gen11_sseu_device_status(struct intel_gt *gt,
continue; continue;
sseu->slice_mask |= BIT(s); sseu->slice_mask |= BIT(s);
sseu_copy_subslices(&info->sseu, s, sseu->subslice_mask); sseu->subslice_mask.hsw[s] = info->sseu.subslice_mask.hsw[s];
for (ss = 0; ss < info->sseu.max_subslices; ss++) { for (ss = 0; ss < info->sseu.max_subslices; ss++) {
unsigned int eu_cnt; unsigned int eu_cnt;
...@@ -147,21 +140,17 @@ static void gen9_sseu_device_status(struct intel_gt *gt, ...@@ -147,21 +140,17 @@ static void gen9_sseu_device_status(struct intel_gt *gt,
sseu->slice_mask |= BIT(s); sseu->slice_mask |= BIT(s);
if (IS_GEN9_BC(gt->i915)) if (IS_GEN9_BC(gt->i915))
sseu_copy_subslices(&info->sseu, s, sseu->subslice_mask.hsw[s] = info->sseu.subslice_mask.hsw[s];
sseu->subslice_mask);
for (ss = 0; ss < info->sseu.max_subslices; ss++) { for (ss = 0; ss < info->sseu.max_subslices; ss++) {
unsigned int eu_cnt; unsigned int eu_cnt;
u8 ss_idx = s * info->sseu.ss_stride +
ss / BITS_PER_BYTE;
if (IS_GEN9_LP(gt->i915)) { if (IS_GEN9_LP(gt->i915)) {
if (!(s_reg[s] & (GEN9_PGCTL_SS_ACK(ss)))) if (!(s_reg[s] & (GEN9_PGCTL_SS_ACK(ss))))
/* skip disabled subslice */ /* skip disabled subslice */
continue; continue;
sseu->subslice_mask[ss_idx] |= sseu->subslice_mask.hsw[s] |= BIT(ss);
BIT(ss % BITS_PER_BYTE);
} }
eu_cnt = eu_reg[2 * s + ss / 2] & eu_mask[ss % 2]; eu_cnt = eu_reg[2 * s + ss / 2] & eu_mask[ss % 2];
...@@ -188,8 +177,7 @@ static void bdw_sseu_device_status(struct intel_gt *gt, ...@@ -188,8 +177,7 @@ static void bdw_sseu_device_status(struct intel_gt *gt,
if (sseu->slice_mask) { if (sseu->slice_mask) {
sseu->eu_per_subslice = info->sseu.eu_per_subslice; sseu->eu_per_subslice = info->sseu.eu_per_subslice;
for (s = 0; s < fls(sseu->slice_mask); s++) for (s = 0; s < fls(sseu->slice_mask); s++)
sseu_copy_subslices(&info->sseu, s, sseu->subslice_mask.hsw[s] = info->sseu.subslice_mask.hsw[s];
sseu->subslice_mask);
sseu->eu_total = sseu->eu_per_subslice * sseu->eu_total = sseu->eu_per_subslice *
intel_sseu_subslice_total(sseu); intel_sseu_subslice_total(sseu);
...@@ -208,7 +196,6 @@ static void i915_print_sseu_info(struct seq_file *m, ...@@ -208,7 +196,6 @@ static void i915_print_sseu_info(struct seq_file *m,
const struct sseu_dev_info *sseu) const struct sseu_dev_info *sseu)
{ {
const char *type = is_available_info ? "Available" : "Enabled"; const char *type = is_available_info ? "Available" : "Enabled";
int s;
seq_printf(m, " %s Slice Mask: %04x\n", type, seq_printf(m, " %s Slice Mask: %04x\n", type,
sseu->slice_mask); sseu->slice_mask);
...@@ -216,10 +203,7 @@ static void i915_print_sseu_info(struct seq_file *m, ...@@ -216,10 +203,7 @@ static void i915_print_sseu_info(struct seq_file *m,
hweight8(sseu->slice_mask)); hweight8(sseu->slice_mask));
seq_printf(m, " %s Subslice Total: %u\n", type, seq_printf(m, " %s Subslice Total: %u\n", type,
intel_sseu_subslice_total(sseu)); intel_sseu_subslice_total(sseu));
for (s = 0; s < fls(sseu->slice_mask); s++) { intel_sseu_print_ss_info(type, sseu, m);
seq_printf(m, " %s Slice%i subslices: %u\n", type,
s, intel_sseu_subslices_per_slice(sseu, s));
}
seq_printf(m, " %s EU Total: %u\n", type, seq_printf(m, " %s EU Total: %u\n", type,
sseu->eu_total); sseu->eu_total);
seq_printf(m, " %s EU Per Subslice: %u\n", type, seq_printf(m, " %s EU Per Subslice: %u\n", type,
......
...@@ -950,8 +950,8 @@ gen9_wa_init_mcr(struct drm_i915_private *i915, struct i915_wa_list *wal) ...@@ -950,8 +950,8 @@ gen9_wa_init_mcr(struct drm_i915_private *i915, struct i915_wa_list *wal)
* on s/ss combo, the read should be done with read_subslice_reg. * on s/ss combo, the read should be done with read_subslice_reg.
*/ */
slice = ffs(sseu->slice_mask) - 1; slice = ffs(sseu->slice_mask) - 1;
GEM_BUG_ON(slice >= ARRAY_SIZE(sseu->subslice_mask)); GEM_BUG_ON(slice >= ARRAY_SIZE(sseu->subslice_mask.hsw));
subslice = ffs(intel_sseu_get_subslices(sseu, slice)); subslice = ffs(intel_sseu_get_hsw_subslices(sseu, slice));
GEM_BUG_ON(!subslice); GEM_BUG_ON(!subslice);
subslice--; subslice--;
...@@ -1089,11 +1089,10 @@ static void ...@@ -1089,11 +1089,10 @@ static void
icl_wa_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal) icl_wa_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal)
{ {
const struct sseu_dev_info *sseu = &gt->info.sseu; const struct sseu_dev_info *sseu = &gt->info.sseu;
unsigned int slice, subslice; unsigned int subslice;
GEM_BUG_ON(GRAPHICS_VER(gt->i915) < 11); GEM_BUG_ON(GRAPHICS_VER(gt->i915) < 11);
GEM_BUG_ON(hweight8(sseu->slice_mask) > 1); GEM_BUG_ON(hweight8(sseu->slice_mask) > 1);
slice = 0;
/* /*
* Although a platform may have subslices, we need to always steer * Although a platform may have subslices, we need to always steer
...@@ -1104,7 +1103,7 @@ icl_wa_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal) ...@@ -1104,7 +1103,7 @@ icl_wa_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal)
* one of the higher subslices, we run the risk of reading back 0's or * one of the higher subslices, we run the risk of reading back 0's or
* random garbage. * random garbage.
*/ */
subslice = __ffs(intel_sseu_get_subslices(sseu, slice)); subslice = __ffs(intel_sseu_get_hsw_subslices(sseu, 0));
/* /*
* If the subslice we picked above also steers us to a valid L3 bank, * If the subslice we picked above also steers us to a valid L3 bank,
...@@ -1114,7 +1113,7 @@ icl_wa_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal) ...@@ -1114,7 +1113,7 @@ icl_wa_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal)
if (gt->info.l3bank_mask & BIT(subslice)) if (gt->info.l3bank_mask & BIT(subslice))
gt->steering_table[L3BANK] = NULL; gt->steering_table[L3BANK] = NULL;
__add_mcr_wa(gt, wal, slice, subslice); __add_mcr_wa(gt, wal, 0, subslice);
} }
static void static void
...@@ -1122,7 +1121,6 @@ xehp_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal) ...@@ -1122,7 +1121,6 @@ xehp_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal)
{ {
const struct sseu_dev_info *sseu = &gt->info.sseu; const struct sseu_dev_info *sseu = &gt->info.sseu;
unsigned long slice, subslice = 0, slice_mask = 0; unsigned long slice, subslice = 0, slice_mask = 0;
u64 dss_mask = 0;
u32 lncf_mask = 0; u32 lncf_mask = 0;
int i; int i;
...@@ -1153,8 +1151,8 @@ xehp_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal) ...@@ -1153,8 +1151,8 @@ xehp_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal)
*/ */
/* Find the potential gslice candidates */ /* Find the potential gslice candidates */
dss_mask = intel_sseu_get_subslices(sseu, 0); slice_mask = intel_slicemask_from_xehp_dssmask(sseu->subslice_mask,
slice_mask = intel_slicemask_from_dssmask(dss_mask, GEN_DSS_PER_GSLICE); GEN_DSS_PER_GSLICE);
/* /*
* Find the potential LNCF candidates. Either LNCF within a valid * Find the potential LNCF candidates. Either LNCF within a valid
...@@ -1179,9 +1177,8 @@ xehp_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal) ...@@ -1179,9 +1177,8 @@ xehp_init_mcr(struct intel_gt *gt, struct i915_wa_list *wal)
} }
slice = __ffs(slice_mask); slice = __ffs(slice_mask);
subslice = __ffs(dss_mask >> (slice * GEN_DSS_PER_GSLICE)); subslice = intel_sseu_find_first_xehp_dss(sseu, GEN_DSS_PER_GSLICE, slice);
WARN_ON(subslice > GEN_DSS_PER_GSLICE); WARN_ON(subslice > GEN_DSS_PER_GSLICE);
WARN_ON(dss_mask >> (slice * GEN_DSS_PER_GSLICE) == 0);
__add_mcr_wa(gt, wal, slice, subslice); __add_mcr_wa(gt, wal, slice, subslice);
...@@ -2062,9 +2059,8 @@ engine_fake_wa_init(struct intel_engine_cs *engine, struct i915_wa_list *wal) ...@@ -2062,9 +2059,8 @@ engine_fake_wa_init(struct intel_engine_cs *engine, struct i915_wa_list *wal)
static bool needs_wa_1308578152(struct intel_engine_cs *engine) static bool needs_wa_1308578152(struct intel_engine_cs *engine)
{ {
u64 dss_mask = intel_sseu_get_subslices(&engine->gt->info.sseu, 0); return intel_sseu_find_first_xehp_dss(&engine->gt->info.sseu, 0, 0) >
GEN_DSS_PER_GSLICE;
return (dss_mask & GENMASK(GEN_DSS_PER_GSLICE - 1, 0)) == 0;
} }
static void static void
......
...@@ -162,8 +162,7 @@ int i915_getparam_ioctl(struct drm_device *dev, void *data, ...@@ -162,8 +162,7 @@ int i915_getparam_ioctl(struct drm_device *dev, void *data,
return -EINVAL; return -EINVAL;
/* Only copy bits from the first slice */ /* Only copy bits from the first slice */
memcpy(&value, sseu->subslice_mask, value = intel_sseu_get_hsw_subslices(sseu, 0);
min(sseu->ss_stride, (u8)sizeof(value)));
if (!value) if (!value)
return -ENODEV; return -ENODEV;
break; break;
......
...@@ -31,10 +31,11 @@ static int copy_query_item(void *query_hdr, size_t query_sz, ...@@ -31,10 +31,11 @@ static int copy_query_item(void *query_hdr, size_t query_sz,
static int fill_topology_info(const struct sseu_dev_info *sseu, static int fill_topology_info(const struct sseu_dev_info *sseu,
struct drm_i915_query_item *query_item, struct drm_i915_query_item *query_item,
const u8 *subslice_mask) intel_sseu_ss_mask_t subslice_mask)
{ {
struct drm_i915_query_topology_info topo; struct drm_i915_query_topology_info topo;
u32 slice_length, subslice_length, eu_length, total_length; u32 slice_length, subslice_length, eu_length, total_length;
int ss_stride = GEN_SSEU_STRIDE(sseu->max_subslices);
int eu_stride = GEN_SSEU_STRIDE(sseu->max_eus_per_subslice); int eu_stride = GEN_SSEU_STRIDE(sseu->max_eus_per_subslice);
int ret; int ret;
...@@ -44,7 +45,7 @@ static int fill_topology_info(const struct sseu_dev_info *sseu, ...@@ -44,7 +45,7 @@ static int fill_topology_info(const struct sseu_dev_info *sseu,
return -ENODEV; return -ENODEV;
slice_length = sizeof(sseu->slice_mask); slice_length = sizeof(sseu->slice_mask);
subslice_length = sseu->max_slices * sseu->ss_stride; subslice_length = sseu->max_slices * ss_stride;
eu_length = sseu->max_slices * sseu->max_subslices * eu_stride; eu_length = sseu->max_slices * sseu->max_subslices * eu_stride;
total_length = sizeof(topo) + slice_length + subslice_length + total_length = sizeof(topo) + slice_length + subslice_length +
eu_length; eu_length;
...@@ -60,7 +61,7 @@ static int fill_topology_info(const struct sseu_dev_info *sseu, ...@@ -60,7 +61,7 @@ static int fill_topology_info(const struct sseu_dev_info *sseu,
topo.max_eus_per_subslice = sseu->max_eus_per_subslice; topo.max_eus_per_subslice = sseu->max_eus_per_subslice;
topo.subslice_offset = slice_length; topo.subslice_offset = slice_length;
topo.subslice_stride = sseu->ss_stride; topo.subslice_stride = ss_stride;
topo.eu_offset = slice_length + subslice_length; topo.eu_offset = slice_length + subslice_length;
topo.eu_stride = eu_stride; topo.eu_stride = eu_stride;
...@@ -72,9 +73,9 @@ static int fill_topology_info(const struct sseu_dev_info *sseu, ...@@ -72,9 +73,9 @@ static int fill_topology_info(const struct sseu_dev_info *sseu,
&sseu->slice_mask, slice_length)) &sseu->slice_mask, slice_length))
return -EFAULT; return -EFAULT;
if (copy_to_user(u64_to_user_ptr(query_item->data_ptr + if (intel_sseu_copy_ssmask_to_user(u64_to_user_ptr(query_item->data_ptr +
sizeof(topo) + slice_length), sizeof(topo) + slice_length),
subslice_mask, subslice_length)) sseu))
return -EFAULT; return -EFAULT;
if (intel_sseu_copy_eumask_to_user(u64_to_user_ptr(query_item->data_ptr + if (intel_sseu_copy_eumask_to_user(u64_to_user_ptr(query_item->data_ptr +
......
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