aboutsummaryrefslogtreecommitdiffstats
path: root/meta-amd-bsp/recipes-kernel/linux/linux-yocto-5.15/0086-EDAC-amd64-Add-new-register-offset-support-and-relat.patch
diff options
context:
space:
mode:
Diffstat (limited to 'meta-amd-bsp/recipes-kernel/linux/linux-yocto-5.15/0086-EDAC-amd64-Add-new-register-offset-support-and-relat.patch')
-rw-r--r--meta-amd-bsp/recipes-kernel/linux/linux-yocto-5.15/0086-EDAC-amd64-Add-new-register-offset-support-and-relat.patch229
1 files changed, 229 insertions, 0 deletions
diff --git a/meta-amd-bsp/recipes-kernel/linux/linux-yocto-5.15/0086-EDAC-amd64-Add-new-register-offset-support-and-relat.patch b/meta-amd-bsp/recipes-kernel/linux/linux-yocto-5.15/0086-EDAC-amd64-Add-new-register-offset-support-and-relat.patch
new file mode 100644
index 00000000..72db2450
--- /dev/null
+++ b/meta-amd-bsp/recipes-kernel/linux/linux-yocto-5.15/0086-EDAC-amd64-Add-new-register-offset-support-and-relat.patch
@@ -0,0 +1,229 @@
+From 68170266b364b39efd0e6eba8f06b8f9e1255bb4 Mon Sep 17 00:00:00 2001
+From: Yazen Ghannam <yazen.ghannam@amd.com>
+Date: Wed, 2 Feb 2022 14:43:07 +0000
+Subject: [PATCH 86/86] EDAC/amd64: Add new register offset support and related
+ changes
+
+commit 2151c84ece920dc55942495004a823cbecb921e5 upstream
+
+Introduce a "family flags" bitmask that can be used to indicate any
+special behavior needed on a per-family basis.
+
+Add a flag to indicate a system uses the new register offsets introduced
+with Family 19h Model 10h.
+
+Use this flag to account for register offset changes, a new bitfield
+indicating DDR5 use on a memory controller, and to set the proper number
+of chip select masks.
+
+Rework f17_addr_mask_to_cs_size() to properly handle the change in chip
+select masks. And update code comments to reflect the updated Chip
+Select, DIMM, and Mask relationships.
+
+[uninitialized variable warning]
+Reported-by: kernel test robot <lkp@intel.com>
+Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com>
+Signed-off-by: Borislav Petkov <bp@suse.de>
+Reviewed-by: William Roche <william.roche@oracle.com>
+Link: https://lore.kernel.org/r/20220202144307.2678405-3-yazen.ghannam@amd.com
+Signed-off-by: Zhaolong Zhang <zhaolong.zhang@windriver.com>
+---
+ drivers/edac/amd64_edac.c | 80 +++++++++++++++++++++++++++++++--------
+ drivers/edac/amd64_edac.h | 14 +++++++
+ 2 files changed, 78 insertions(+), 16 deletions(-)
+
+diff --git a/drivers/edac/amd64_edac.c b/drivers/edac/amd64_edac.c
+index 771b30d8c77f..009e4fe5df8f 100644
+--- a/drivers/edac/amd64_edac.c
++++ b/drivers/edac/amd64_edac.c
+@@ -15,6 +15,21 @@ static struct msr __percpu *msrs;
+
+ static struct amd64_family_type *fam_type;
+
++static inline u32 get_umc_reg(u32 reg)
++{
++ if (!fam_type->flags.zn_regs_v2)
++ return reg;
++
++ switch (reg) {
++ case UMCCH_ADDR_CFG: return UMCCH_ADDR_CFG_DDR5;
++ case UMCCH_ADDR_MASK_SEC: return UMCCH_ADDR_MASK_SEC_DDR5;
++ case UMCCH_DIMM_CFG: return UMCCH_DIMM_CFG_DDR5;
++ }
++
++ WARN_ONCE(1, "%s: unknown register 0x%x", __func__, reg);
++ return 0;
++}
++
+ /* Per-node stuff */
+ static struct ecc_settings **ecc_stngs;
+
+@@ -1429,8 +1444,10 @@ static void __dump_misc_regs_df(struct amd64_pvt *pvt)
+ edac_dbg(1, "UMC%d x16 DIMMs present: %s\n",
+ i, (umc->dimm_cfg & BIT(7)) ? "yes" : "no");
+
+- if (umc->dram_type == MEM_LRDDR4) {
+- amd_smn_read(pvt->mc_node_id, umc_base + UMCCH_ADDR_CFG, &tmp);
++ if (umc->dram_type == MEM_LRDDR4 || umc->dram_type == MEM_LRDDR5) {
++ amd_smn_read(pvt->mc_node_id,
++ umc_base + get_umc_reg(UMCCH_ADDR_CFG),
++ &tmp);
+ edac_dbg(1, "UMC%d LRDIMM %dx rank multiply\n",
+ i, 1 << ((tmp >> 4) & 0x3));
+ }
+@@ -1505,7 +1522,7 @@ static void prep_chip_selects(struct amd64_pvt *pvt)
+
+ for_each_umc(umc) {
+ pvt->csels[umc].b_cnt = 4;
+- pvt->csels[umc].m_cnt = 2;
++ pvt->csels[umc].m_cnt = fam_type->flags.zn_regs_v2 ? 4 : 2;
+ }
+
+ } else {
+@@ -1545,7 +1562,7 @@ static void read_umc_base_mask(struct amd64_pvt *pvt)
+ }
+
+ umc_mask_reg = get_umc_base(umc) + UMCCH_ADDR_MASK;
+- umc_mask_reg_sec = get_umc_base(umc) + UMCCH_ADDR_MASK_SEC;
++ umc_mask_reg_sec = get_umc_base(umc) + get_umc_reg(UMCCH_ADDR_MASK_SEC);
+
+ for_each_chip_select_mask(cs, umc, pvt) {
+ mask = &pvt->csels[umc].csmasks[cs];
+@@ -1629,12 +1646,25 @@ static void determine_memory_type_df(struct amd64_pvt *pvt)
+ continue;
+ }
+
+- if (umc->dimm_cfg & BIT(5))
+- umc->dram_type = MEM_LRDDR4;
+- else if (umc->dimm_cfg & BIT(4))
+- umc->dram_type = MEM_RDDR4;
+- else
+- umc->dram_type = MEM_DDR4;
++ /*
++ * Check if the system supports the "DDR Type" field in UMC Config
++ * and has DDR5 DIMMs in use.
++ */
++ if (fam_type->flags.zn_regs_v2 && ((umc->umc_cfg & GENMASK(2, 0)) == 0x1)) {
++ if (umc->dimm_cfg & BIT(5))
++ umc->dram_type = MEM_LRDDR5;
++ else if (umc->dimm_cfg & BIT(4))
++ umc->dram_type = MEM_RDDR5;
++ else
++ umc->dram_type = MEM_DDR5;
++ } else {
++ if (umc->dimm_cfg & BIT(5))
++ umc->dram_type = MEM_LRDDR4;
++ else if (umc->dimm_cfg & BIT(4))
++ umc->dram_type = MEM_RDDR4;
++ else
++ umc->dram_type = MEM_DDR4;
++ }
+
+ edac_dbg(1, " UMC%d DIMM type: %s\n", i, edac_mem_types[umc->dram_type]);
+ }
+@@ -2166,6 +2196,7 @@ static int f17_addr_mask_to_cs_size(struct amd64_pvt *pvt, u8 umc,
+ {
+ u32 addr_mask_orig, addr_mask_deinterleaved;
+ u32 msb, weight, num_zero_bits;
++ int cs_mask_nr = csrow_nr;
+ int dimm, size = 0;
+
+ /* No Chip Selects are enabled. */
+@@ -2181,17 +2212,33 @@ static int f17_addr_mask_to_cs_size(struct amd64_pvt *pvt, u8 umc,
+ return size;
+
+ /*
+- * There is one mask per DIMM, and two Chip Selects per DIMM.
+- * CS0 and CS1 -> DIMM0
+- * CS2 and CS3 -> DIMM1
++ * Family 17h introduced systems with one mask per DIMM,
++ * and two Chip Selects per DIMM.
++ *
++ * CS0 and CS1 -> MASK0 / DIMM0
++ * CS2 and CS3 -> MASK1 / DIMM1
++ *
++ * Family 19h Model 10h introduced systems with one mask per Chip Select,
++ * and two Chip Selects per DIMM.
++ *
++ * CS0 -> MASK0 -> DIMM0
++ * CS1 -> MASK1 -> DIMM0
++ * CS2 -> MASK2 -> DIMM1
++ * CS3 -> MASK3 -> DIMM1
++ *
++ * Keep the mask number equal to the Chip Select number for newer systems,
++ * and shift the mask number for older systems.
+ */
+ dimm = csrow_nr >> 1;
+
++ if (!fam_type->flags.zn_regs_v2)
++ cs_mask_nr >>= 1;
++
+ /* Asymmetric dual-rank DIMM support. */
+ if ((csrow_nr & 1) && (cs_mode & CS_ODD_SECONDARY))
+- addr_mask_orig = pvt->csels[umc].csmasks_sec[dimm];
++ addr_mask_orig = pvt->csels[umc].csmasks_sec[cs_mask_nr];
+ else
+- addr_mask_orig = pvt->csels[umc].csmasks[dimm];
++ addr_mask_orig = pvt->csels[umc].csmasks[cs_mask_nr];
+
+ /*
+ * The number of zero bits in the mask is equal to the number of bits
+@@ -2947,6 +2994,7 @@ static struct amd64_family_type family_types[] = {
+ .f0_id = PCI_DEVICE_ID_AMD_19H_M10H_DF_F0,
+ .f6_id = PCI_DEVICE_ID_AMD_19H_M10H_DF_F6,
+ .max_mcs = 12,
++ .flags.zn_regs_v2 = 1,
+ .ops = {
+ .early_channel_count = f17_early_channel_count,
+ .dbam_to_cs = f17_addr_mask_to_cs_size,
+@@ -3375,7 +3423,7 @@ static void __read_mc_regs_df(struct amd64_pvt *pvt)
+ umc_base = get_umc_base(i);
+ umc = &pvt->umc[i];
+
+- amd_smn_read(nid, umc_base + UMCCH_DIMM_CFG, &umc->dimm_cfg);
++ amd_smn_read(nid, umc_base + get_umc_reg(UMCCH_DIMM_CFG), &umc->dimm_cfg);
+ amd_smn_read(nid, umc_base + UMCCH_UMC_CFG, &umc->umc_cfg);
+ amd_smn_read(nid, umc_base + UMCCH_SDP_CTRL, &umc->sdp_ctrl);
+ amd_smn_read(nid, umc_base + UMCCH_ECC_CTRL, &umc->ecc_ctrl);
+diff --git a/drivers/edac/amd64_edac.h b/drivers/edac/amd64_edac.h
+index 7691a0baacdd..8d7b168f8e64 100644
+--- a/drivers/edac/amd64_edac.h
++++ b/drivers/edac/amd64_edac.h
+@@ -271,8 +271,11 @@
+ #define UMCCH_BASE_ADDR_SEC 0x10
+ #define UMCCH_ADDR_MASK 0x20
+ #define UMCCH_ADDR_MASK_SEC 0x28
++#define UMCCH_ADDR_MASK_SEC_DDR5 0x30
+ #define UMCCH_ADDR_CFG 0x30
++#define UMCCH_ADDR_CFG_DDR5 0x40
+ #define UMCCH_DIMM_CFG 0x80
++#define UMCCH_DIMM_CFG_DDR5 0x90
+ #define UMCCH_UMC_CFG 0x100
+ #define UMCCH_SDP_CTRL 0x104
+ #define UMCCH_ECC_CTRL 0x14C
+@@ -485,11 +488,22 @@ struct low_ops {
+ unsigned cs_mode, int cs_mask_nr);
+ };
+
++struct amd64_family_flags {
++ /*
++ * Indicates that the system supports the new register offsets, etc.
++ * first introduced with Family 19h Model 10h.
++ */
++ __u64 zn_regs_v2 : 1,
++
++ __reserved : 63;
++};
++
+ struct amd64_family_type {
+ const char *ctl_name;
+ u16 f0_id, f1_id, f2_id, f6_id;
+ /* Maximum number of memory controllers per die/node. */
+ u8 max_mcs;
++ struct amd64_family_flags flags;
+ struct low_ops ops;
+ };
+
+--
+2.37.3
+