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: Muralidhara M K <muralidhara.mk@amd.com> Co-developed-by: Naveen Krishna Chatradhi <naveenkrishna.chatradhi@amd.com> Signed-off-by: Naveen Krishna Chatradhi <naveenkrishna.chatradhi@amd.com> Co-developed-by: Yazen Ghannam <yazen.ghannam@amd.com> Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com> Signed-off-by: Borislav Petkov (AMD) <bp@alien8.de> Link: https://lore.kernel.org/r/20230127170419.1824692-17-yazen.ghannam@amd.com
This commit is contained in:
parent
32ecdf8688
commit
eb2bcdfc37
|
@ -3527,48 +3527,21 @@ 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);
|
|
||||||
|
|
||||||
if (umc->umc_cap_hi & UMC_ECC_ENABLED)
|
|
||||||
ecc_en_mask |= BIT(i);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 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);
|
|
||||||
|
|
||||||
/* Assume UMC MCA banks are enabled. */
|
|
||||||
nb_mce_en = true;
|
|
||||||
} else {
|
|
||||||
amd64_read_pci_cfg(pvt->F3, NBCFG, &value);
|
|
||||||
|
|
||||||
ecc_en = !!(value & NBCFG_ECC_ENABLE);
|
|
||||||
|
|
||||||
nb_mce_en = nb_mce_bank_enabled_on_node(nid);
|
|
||||||
if (!nb_mce_en)
|
|
||||||
edac_dbg(0, "NB MCE bank disabled, set MSR 0x%08x[4] on node %d to enable.\n",
|
|
||||||
MSR_IA32_MCG_CTL, 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"));
|
||||||
|
|
||||||
|
@ -3578,6 +3551,40 @@ static bool ecc_enabled(struct amd64_pvt *pvt)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool umc_ecc_enabled(struct amd64_pvt *pvt)
|
||||||
|
{
|
||||||
|
u8 umc_en_mask = 0, ecc_en_mask = 0;
|
||||||
|
u16 nid = pvt->mc_node_id;
|
||||||
|
struct amd64_umc *umc;
|
||||||
|
u8 ecc_en = 0, i;
|
||||||
|
|
||||||
|
for_each_umc(i) {
|
||||||
|
umc = &pvt->umc[i];
|
||||||
|
|
||||||
|
/* Only check enabled UMCs. */
|
||||||
|
if (!(umc->sdp_ctrl & UMC_SDP_INIT))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
umc_en_mask |= BIT(i);
|
||||||
|
|
||||||
|
if (umc->umc_cap_hi & UMC_ECC_ENABLED)
|
||||||
|
ecc_en_mask |= BIT(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 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"));
|
||||||
|
|
||||||
|
if (!ecc_en)
|
||||||
|
return false;
|
||||||
|
else
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static inline void
|
static inline void
|
||||||
f17h_determine_edac_ctl_cap(struct mem_ctl_info *mci, struct amd64_pvt *pvt)
|
f17h_determine_edac_ctl_cap(struct mem_ctl_info *mci, 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 = {
|
||||||
.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)
|
||||||
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 {
|
||||||
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,
|
||||||
|
|
Loading…
Reference in New Issue