2019-06-21 15:07:41 +08:00
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
/*
|
|
|
|
* Copyright © 2019 Intel Corporation
|
|
|
|
*/
|
|
|
|
|
2019-06-21 15:07:42 +08:00
|
|
|
#include "i915_drv.h"
|
|
|
|
|
2019-06-21 15:07:41 +08:00
|
|
|
#include "intel_gt.h"
|
2019-06-21 15:07:43 +08:00
|
|
|
#include "intel_gt_pm.h"
|
2019-06-21 15:07:44 +08:00
|
|
|
#include "intel_uncore.h"
|
2019-06-21 15:07:41 +08:00
|
|
|
|
2019-06-21 15:07:42 +08:00
|
|
|
void intel_gt_init_early(struct intel_gt *gt, struct drm_i915_private *i915)
|
2019-06-21 15:07:41 +08:00
|
|
|
{
|
2019-06-21 15:07:42 +08:00
|
|
|
gt->i915 = i915;
|
|
|
|
gt->uncore = &i915->uncore;
|
|
|
|
|
2019-06-21 15:07:41 +08:00
|
|
|
INIT_LIST_HEAD(>->active_rings);
|
|
|
|
INIT_LIST_HEAD(>->closed_vma);
|
|
|
|
|
|
|
|
spin_lock_init(>->closed_lock);
|
2019-06-21 15:07:43 +08:00
|
|
|
|
|
|
|
intel_gt_pm_init_early(gt);
|
2019-06-21 15:07:41 +08:00
|
|
|
}
|
2019-06-21 15:07:44 +08:00
|
|
|
|
|
|
|
static void rmw_set(struct intel_uncore *uncore, i915_reg_t reg, u32 set)
|
|
|
|
{
|
|
|
|
intel_uncore_rmw(uncore, reg, 0, set);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void rmw_clear(struct intel_uncore *uncore, i915_reg_t reg, u32 clr)
|
|
|
|
{
|
|
|
|
intel_uncore_rmw(uncore, reg, clr, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void clear_register(struct intel_uncore *uncore, i915_reg_t reg)
|
|
|
|
{
|
|
|
|
intel_uncore_rmw(uncore, reg, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gen8_clear_engine_error_register(struct intel_engine_cs *engine)
|
|
|
|
{
|
|
|
|
GEN6_RING_FAULT_REG_RMW(engine, RING_FAULT_VALID, 0);
|
|
|
|
GEN6_RING_FAULT_REG_POSTING_READ(engine);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
intel_gt_clear_error_registers(struct intel_gt *gt,
|
|
|
|
intel_engine_mask_t engine_mask)
|
|
|
|
{
|
|
|
|
struct drm_i915_private *i915 = gt->i915;
|
|
|
|
struct intel_uncore *uncore = gt->uncore;
|
|
|
|
u32 eir;
|
|
|
|
|
|
|
|
if (!IS_GEN(i915, 2))
|
|
|
|
clear_register(uncore, PGTBL_ER);
|
|
|
|
|
|
|
|
if (INTEL_GEN(i915) < 4)
|
|
|
|
clear_register(uncore, IPEIR(RENDER_RING_BASE));
|
|
|
|
else
|
|
|
|
clear_register(uncore, IPEIR_I965);
|
|
|
|
|
|
|
|
clear_register(uncore, EIR);
|
|
|
|
eir = intel_uncore_read(uncore, EIR);
|
|
|
|
if (eir) {
|
|
|
|
/*
|
|
|
|
* some errors might have become stuck,
|
|
|
|
* mask them.
|
|
|
|
*/
|
|
|
|
DRM_DEBUG_DRIVER("EIR stuck: 0x%08x, masking\n", eir);
|
|
|
|
rmw_set(uncore, EMR, eir);
|
|
|
|
intel_uncore_write(uncore, GEN2_IIR,
|
|
|
|
I915_MASTER_ERROR_INTERRUPT);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (INTEL_GEN(i915) >= 8) {
|
|
|
|
rmw_clear(uncore, GEN8_RING_FAULT_REG, RING_FAULT_VALID);
|
|
|
|
intel_uncore_posting_read(uncore, GEN8_RING_FAULT_REG);
|
|
|
|
} else if (INTEL_GEN(i915) >= 6) {
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
|
|
|
|
for_each_engine_masked(engine, i915, engine_mask, id)
|
|
|
|
gen8_clear_engine_error_register(engine);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gen6_check_faults(struct intel_gt *gt)
|
|
|
|
{
|
|
|
|
struct intel_engine_cs *engine;
|
|
|
|
enum intel_engine_id id;
|
|
|
|
u32 fault;
|
|
|
|
|
|
|
|
for_each_engine(engine, gt->i915, id) {
|
|
|
|
fault = GEN6_RING_FAULT_REG_READ(engine);
|
|
|
|
if (fault & RING_FAULT_VALID) {
|
|
|
|
DRM_DEBUG_DRIVER("Unexpected fault\n"
|
|
|
|
"\tAddr: 0x%08lx\n"
|
|
|
|
"\tAddress space: %s\n"
|
|
|
|
"\tSource ID: %d\n"
|
|
|
|
"\tType: %d\n",
|
|
|
|
fault & PAGE_MASK,
|
|
|
|
fault & RING_FAULT_GTTSEL_MASK ?
|
|
|
|
"GGTT" : "PPGTT",
|
|
|
|
RING_FAULT_SRCID(fault),
|
|
|
|
RING_FAULT_FAULT_TYPE(fault));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gen8_check_faults(struct intel_gt *gt)
|
|
|
|
{
|
|
|
|
struct intel_uncore *uncore = gt->uncore;
|
|
|
|
u32 fault = intel_uncore_read(uncore, GEN8_RING_FAULT_REG);
|
|
|
|
|
|
|
|
if (fault & RING_FAULT_VALID) {
|
|
|
|
u32 fault_data0, fault_data1;
|
|
|
|
u64 fault_addr;
|
|
|
|
|
|
|
|
fault_data0 = intel_uncore_read(uncore, GEN8_FAULT_TLB_DATA0);
|
|
|
|
fault_data1 = intel_uncore_read(uncore, GEN8_FAULT_TLB_DATA1);
|
|
|
|
fault_addr = ((u64)(fault_data1 & FAULT_VA_HIGH_BITS) << 44) |
|
|
|
|
((u64)fault_data0 << 12);
|
|
|
|
|
|
|
|
DRM_DEBUG_DRIVER("Unexpected fault\n"
|
|
|
|
"\tAddr: 0x%08x_%08x\n"
|
|
|
|
"\tAddress space: %s\n"
|
|
|
|
"\tEngine ID: %d\n"
|
|
|
|
"\tSource ID: %d\n"
|
|
|
|
"\tType: %d\n",
|
|
|
|
upper_32_bits(fault_addr),
|
|
|
|
lower_32_bits(fault_addr),
|
|
|
|
fault_data1 & FAULT_GTT_SEL ? "GGTT" : "PPGTT",
|
|
|
|
GEN8_RING_FAULT_ENGINE_ID(fault),
|
|
|
|
RING_FAULT_SRCID(fault),
|
|
|
|
RING_FAULT_FAULT_TYPE(fault));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void intel_gt_check_and_clear_faults(struct intel_gt *gt)
|
|
|
|
{
|
|
|
|
struct drm_i915_private *i915 = gt->i915;
|
|
|
|
|
|
|
|
/* From GEN8 onwards we only have one 'All Engine Fault Register' */
|
|
|
|
if (INTEL_GEN(i915) >= 8)
|
|
|
|
gen8_check_faults(gt);
|
|
|
|
else if (INTEL_GEN(i915) >= 6)
|
|
|
|
gen6_check_faults(gt);
|
|
|
|
else
|
|
|
|
return;
|
|
|
|
|
|
|
|
intel_gt_clear_error_registers(gt, ALL_ENGINES);
|
|
|
|
}
|