Commit eb2bcdfc authored by Muralidhara M K's avatar Muralidhara M K Committed by Borislav Petkov (AMD)

EDAC/amd64: Split ecc_enabled() into dct/umc functions

Call them using a function pointer in pvt->ops. The "ECC enabled"
check is done outside of the hardware information gathering done in
hw_info_get(). So a high-level function pointer is needed to separate
the legacy and modern paths.

No functional change is intended.

  [Yazen: rebased/reworked patch and reworded commit message. ]
Signed-off-by: default avatarMuralidhara M K <muralidhara.mk@amd.com>
Co-developed-by: default avatarNaveen Krishna Chatradhi <naveenkrishna.chatradhi@amd.com>
Signed-off-by: default avatarNaveen Krishna Chatradhi <naveenkrishna.chatradhi@amd.com>
Co-developed-by: default avatarYazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: default avatarYazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: default avatarBorislav Petkov (AMD) <bp@alien8.de>
Link: https://lore.kernel.org/r/20230127170419.1824692-17-yazen.ghannam@amd.com
parent 32ecdf86
...@@ -3527,52 +3527,59 @@ static void restore_ecc_error_reporting(struct ecc_settings *s, u16 nid, ...@@ -3527,52 +3527,59 @@ static void restore_ecc_error_reporting(struct ecc_settings *s, u16 nid,
amd64_warn("Error restoring NB MCGCTL settings!\n"); amd64_warn("Error restoring NB MCGCTL settings!\n");
} }
static bool ecc_enabled(struct amd64_pvt *pvt) static bool dct_ecc_enabled(struct amd64_pvt *pvt)
{ {
u16 nid = pvt->mc_node_id; u16 nid = pvt->mc_node_id;
bool nb_mce_en = false; bool nb_mce_en = false;
u8 ecc_en = 0, i; u8 ecc_en = 0;
u32 value; u32 value;
if (boot_cpu_data.x86 >= 0x17) { amd64_read_pci_cfg(pvt->F3, NBCFG, &value);
u8 umc_en_mask = 0, ecc_en_mask = 0;
struct amd64_umc *umc;
for_each_umc(i) { ecc_en = !!(value & NBCFG_ECC_ENABLE);
umc = &pvt->umc[i];
/* Only check enabled UMCs. */ nb_mce_en = nb_mce_bank_enabled_on_node(nid);
if (!(umc->sdp_ctrl & UMC_SDP_INIT)) if (!nb_mce_en)
continue; edac_dbg(0, "NB MCE bank disabled, set MSR 0x%08x[4] on node %d to enable.\n",
MSR_IA32_MCG_CTL, nid);
umc_en_mask |= BIT(i); edac_dbg(3, "Node %d: DRAM ECC %s.\n", nid, (ecc_en ? "enabled" : "disabled"));
if (umc->umc_cap_hi & UMC_ECC_ENABLED) if (!ecc_en || !nb_mce_en)
ecc_en_mask |= BIT(i); return false;
} else
return true;
}
/* Check whether at least one UMC is enabled: */ static bool umc_ecc_enabled(struct amd64_pvt *pvt)
if (umc_en_mask) {
ecc_en = umc_en_mask == ecc_en_mask; u8 umc_en_mask = 0, ecc_en_mask = 0;
else u16 nid = pvt->mc_node_id;
edac_dbg(0, "Node %d: No enabled UMCs.\n", nid); struct amd64_umc *umc;
u8 ecc_en = 0, i;
/* Assume UMC MCA banks are enabled. */ for_each_umc(i) {
nb_mce_en = true; umc = &pvt->umc[i];
} else {
amd64_read_pci_cfg(pvt->F3, NBCFG, &value); /* Only check enabled UMCs. */
if (!(umc->sdp_ctrl & UMC_SDP_INIT))
continue;
ecc_en = !!(value & NBCFG_ECC_ENABLE); umc_en_mask |= BIT(i);
nb_mce_en = nb_mce_bank_enabled_on_node(nid); if (umc->umc_cap_hi & UMC_ECC_ENABLED)
if (!nb_mce_en) ecc_en_mask |= BIT(i);
edac_dbg(0, "NB MCE bank disabled, set MSR 0x%08x[4] on node %d to enable.\n",
MSR_IA32_MCG_CTL, nid);
} }
/* Check whether at least one UMC is enabled: */
if (umc_en_mask)
ecc_en = umc_en_mask == ecc_en_mask;
else
edac_dbg(0, "Node %d: No enabled UMCs.\n", nid);
edac_dbg(3, "Node %d: DRAM ECC %s.\n", nid, (ecc_en ? "enabled" : "disabled")); edac_dbg(3, "Node %d: DRAM ECC %s.\n", nid, (ecc_en ? "enabled" : "disabled"));
if (!ecc_en || !nb_mce_en) if (!ecc_en)
return false; return false;
else else
return true; return true;
...@@ -3678,6 +3685,7 @@ static void hw_info_put(struct amd64_pvt *pvt) ...@@ -3678,6 +3685,7 @@ static void hw_info_put(struct amd64_pvt *pvt)
static struct low_ops umc_ops = { static struct low_ops umc_ops = {
.hw_info_get = umc_hw_info_get, .hw_info_get = umc_hw_info_get,
.ecc_enabled = umc_ecc_enabled,
}; };
/* Use Family 16h versions for defaults and adjust as needed below. */ /* Use Family 16h versions for defaults and adjust as needed below. */
...@@ -3685,6 +3693,7 @@ static struct low_ops dct_ops = { ...@@ -3685,6 +3693,7 @@ static struct low_ops dct_ops = {
.map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow, .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
.dbam_to_cs = f16_dbam_to_chip_select, .dbam_to_cs = f16_dbam_to_chip_select,
.hw_info_get = dct_hw_info_get, .hw_info_get = dct_hw_info_get,
.ecc_enabled = dct_ecc_enabled,
}; };
static int per_family_init(struct amd64_pvt *pvt) static int per_family_init(struct amd64_pvt *pvt)
...@@ -3910,7 +3919,7 @@ static int probe_one_instance(unsigned int nid) ...@@ -3910,7 +3919,7 @@ static int probe_one_instance(unsigned int nid)
goto err_enable; goto err_enable;
} }
if (!ecc_enabled(pvt)) { if (!pvt->ops->ecc_enabled(pvt)) {
ret = -ENODEV; ret = -ENODEV;
if (!ecc_enable_override) if (!ecc_enable_override)
......
...@@ -467,6 +467,7 @@ struct low_ops { ...@@ -467,6 +467,7 @@ struct low_ops {
int (*dbam_to_cs)(struct amd64_pvt *pvt, u8 dct, int (*dbam_to_cs)(struct amd64_pvt *pvt, u8 dct,
unsigned int cs_mode, int cs_mask_nr); unsigned int cs_mode, int cs_mask_nr);
int (*hw_info_get)(struct amd64_pvt *pvt); int (*hw_info_get)(struct amd64_pvt *pvt);
bool (*ecc_enabled)(struct amd64_pvt *pvt);
}; };
int __amd64_read_pci_cfg_dword(struct pci_dev *pdev, int offset, int __amd64_read_pci_cfg_dword(struct pci_dev *pdev, int offset,
......
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