1 // SPDX-License-Identifier: GPL-2.0
3 * Driver for Intel(R) 10nm server memory controller.
4 * Copyright (c) 2019, Intel Corporation.
8 #include <linux/kernel.h>
9 #include <asm/cpu_device_id.h>
10 #include <asm/intel-family.h>
12 #include "edac_module.h"
13 #include "skx_common.h"
15 #define I10NM_REVISION "v0.0.3"
16 #define EDAC_MOD_STR "i10nm_edac"
19 #define i10nm_printk(level, fmt, arg...) \
20 edac_printk(level, "i10nm", fmt, ##arg)
22 #define I10NM_GET_SCK_BAR(d, reg) \
23 pci_read_config_dword((d)->uracu, 0xd0, &(reg))
24 #define I10NM_GET_IMC_BAR(d, i, reg) \
25 pci_read_config_dword((d)->uracu, 0xd8 + (i) * 4, &(reg))
26 #define I10NM_GET_DIMMMTR(m, i, j) \
27 (*(u32 *)((m)->mbase + 0x2080c + (i) * 0x4000 + (j) * 4))
28 #define I10NM_GET_MCDDRTCFG(m, i, j) \
29 (*(u32 *)((m)->mbase + 0x20970 + (i) * 0x4000 + (j) * 4))
31 #define I10NM_GET_SCK_MMIO_BASE(reg) (GET_BITFIELD(reg, 0, 28) << 23)
32 #define I10NM_GET_IMC_MMIO_OFFSET(reg) (GET_BITFIELD(reg, 0, 10) << 12)
33 #define I10NM_GET_IMC_MMIO_SIZE(reg) ((GET_BITFIELD(reg, 13, 23) - \
34 GET_BITFIELD(reg, 0, 10) + 1) << 12)
36 static struct list_head
*i10nm_edac_list
;
38 static struct pci_dev
*pci_get_dev_wrapper(int dom
, unsigned int bus
,
39 unsigned int dev
, unsigned int fun
)
43 pdev
= pci_get_domain_bus_and_slot(dom
, bus
, PCI_DEVFN(dev
, fun
));
45 edac_dbg(2, "No device %02x:%02x.%x\n",
50 if (unlikely(pci_enable_device(pdev
) < 0)) {
51 edac_dbg(2, "Failed to enable device %02x:%02x.%x\n",
61 static int i10nm_get_all_munits(void)
71 list_for_each_entry(d
, i10nm_edac_list
, list
) {
72 d
->util_all
= pci_get_dev_wrapper(d
->seg
, d
->bus
[1], 29, 1);
76 d
->uracu
= pci_get_dev_wrapper(d
->seg
, d
->bus
[0], 0, 1);
80 if (I10NM_GET_SCK_BAR(d
, reg
)) {
81 i10nm_printk(KERN_ERR
, "Failed to socket bar\n");
85 base
= I10NM_GET_SCK_MMIO_BASE(reg
);
86 edac_dbg(2, "socket%d mmio base 0x%llx (reg 0x%x)\n",
89 for (i
= 0; i
< I10NM_NUM_IMC
; i
++) {
90 mdev
= pci_get_dev_wrapper(d
->seg
, d
->bus
[0],
92 if (i
== 0 && !mdev
) {
93 i10nm_printk(KERN_ERR
, "No IMC found\n");
99 d
->imc
[i
].mdev
= mdev
;
101 if (I10NM_GET_IMC_BAR(d
, i
, reg
)) {
102 i10nm_printk(KERN_ERR
, "Failed to get mc bar\n");
106 off
= I10NM_GET_IMC_MMIO_OFFSET(reg
);
107 size
= I10NM_GET_IMC_MMIO_SIZE(reg
);
108 edac_dbg(2, "mc%d mmio base 0x%llx size 0x%lx (reg 0x%x)\n",
109 i
, base
+ off
, size
, reg
);
111 mbase
= ioremap(base
+ off
, size
);
113 i10nm_printk(KERN_ERR
, "Failed to ioremap 0x%llx\n",
118 d
->imc
[i
].mbase
= mbase
;
125 static const struct x86_cpu_id i10nm_cpuids
[] = {
126 X86_MATCH_INTEL_FAM6_MODEL(ATOM_TREMONT_D
, NULL
),
127 X86_MATCH_INTEL_FAM6_MODEL(ICELAKE_X
, NULL
),
128 X86_MATCH_INTEL_FAM6_MODEL(ICELAKE_D
, NULL
),
131 MODULE_DEVICE_TABLE(x86cpu
, i10nm_cpuids
);
133 static bool i10nm_check_ecc(struct skx_imc
*imc
, int chan
)
137 mcmtr
= *(u32
*)(imc
->mbase
+ 0x20ef8 + chan
* 0x4000);
138 edac_dbg(1, "ch%d mcmtr reg %x\n", chan
, mcmtr
);
140 return !!GET_BITFIELD(mcmtr
, 2, 2);
143 static int i10nm_get_dimm_config(struct mem_ctl_info
*mci
)
145 struct skx_pvt
*pvt
= mci
->pvt_info
;
146 struct skx_imc
*imc
= pvt
->imc
;
147 struct dimm_info
*dimm
;
151 for (i
= 0; i
< I10NM_NUM_CHANNELS
; i
++) {
156 for (j
= 0; j
< I10NM_NUM_DIMMS
; j
++) {
157 dimm
= edac_get_dimm(mci
, i
, j
, 0);
158 mtr
= I10NM_GET_DIMMMTR(imc
, i
, j
);
159 mcddrtcfg
= I10NM_GET_MCDDRTCFG(imc
, i
, j
);
160 edac_dbg(1, "dimmmtr 0x%x mcddrtcfg 0x%x (mc%d ch%d dimm%d)\n",
161 mtr
, mcddrtcfg
, imc
->mc
, i
, j
);
163 if (IS_DIMM_PRESENT(mtr
))
164 ndimms
+= skx_get_dimm_info(mtr
, 0, dimm
,
166 else if (IS_NVDIMM_PRESENT(mcddrtcfg
, j
))
167 ndimms
+= skx_get_nvdimm_info(dimm
, imc
, i
, j
,
170 if (ndimms
&& !i10nm_check_ecc(imc
, i
)) {
171 i10nm_printk(KERN_ERR
, "ECC is disabled on imc %d channel %d\n",
180 static struct notifier_block i10nm_mce_dec
= {
181 .notifier_call
= skx_mce_check_error
,
182 .priority
= MCE_PRIO_EDAC
,
185 #ifdef CONFIG_EDAC_DEBUG
188 * Exercise the address decode logic by writing an address to
189 * /sys/kernel/debug/edac/i10nm_test/addr.
191 static struct dentry
*i10nm_test
;
193 static int debugfs_u64_set(void *data
, u64 val
)
197 pr_warn_once("Fake error to 0x%llx injected via debugfs\n", val
);
199 memset(&m
, 0, sizeof(m
));
200 /* ADDRV + MemRd + Unknown channel */
201 m
.status
= MCI_STATUS_ADDRV
+ 0x90;
202 /* One corrected error */
203 m
.status
|= BIT_ULL(MCI_STATUS_CEC_SHIFT
);
205 skx_mce_check_error(NULL
, 0, &m
);
209 DEFINE_SIMPLE_ATTRIBUTE(fops_u64_wo
, NULL
, debugfs_u64_set
, "%llu\n");
211 static void setup_i10nm_debug(void)
213 i10nm_test
= edac_debugfs_create_dir("i10nm_test");
217 if (!edac_debugfs_create_file("addr", 0200, i10nm_test
,
218 NULL
, &fops_u64_wo
)) {
219 debugfs_remove(i10nm_test
);
224 static void teardown_i10nm_debug(void)
226 debugfs_remove_recursive(i10nm_test
);
229 static inline void setup_i10nm_debug(void) {}
230 static inline void teardown_i10nm_debug(void) {}
231 #endif /*CONFIG_EDAC_DEBUG*/
233 static int __init
i10nm_init(void)
235 u8 mc
= 0, src_id
= 0, node_id
= 0;
236 const struct x86_cpu_id
*id
;
239 int rc
, i
, off
[3] = {0xd0, 0xc8, 0xcc};
244 owner
= edac_get_owner();
245 if (owner
&& strncmp(owner
, EDAC_MOD_STR
, sizeof(EDAC_MOD_STR
)))
248 id
= x86_match_cpu(i10nm_cpuids
);
252 rc
= skx_get_hi_lo(0x09a2, off
, &tolm
, &tohm
);
256 rc
= skx_get_all_bus_mappings(0x3452, 0xcc, I10NM
, &i10nm_edac_list
);
260 i10nm_printk(KERN_ERR
, "No memory controllers found\n");
264 rc
= i10nm_get_all_munits();
268 list_for_each_entry(d
, i10nm_edac_list
, list
) {
269 rc
= skx_get_src_id(d
, 0xf8, &src_id
);
273 rc
= skx_get_node_id(d
, &node_id
);
277 edac_dbg(2, "src_id = %d node_id = %d\n", src_id
, node_id
);
278 for (i
= 0; i
< I10NM_NUM_IMC
; i
++) {
284 d
->imc
[i
].src_id
= src_id
;
285 d
->imc
[i
].node_id
= node_id
;
287 rc
= skx_register_mci(&d
->imc
[i
], d
->imc
[i
].mdev
,
288 "Intel_10nm Socket", EDAC_MOD_STR
,
289 i10nm_get_dimm_config
);
300 mce_register_decode_chain(&i10nm_mce_dec
);
303 i10nm_printk(KERN_INFO
, "%s\n", I10NM_REVISION
);
311 static void __exit
i10nm_exit(void)
314 teardown_i10nm_debug();
315 mce_unregister_decode_chain(&i10nm_mce_dec
);
320 module_init(i10nm_init
);
321 module_exit(i10nm_exit
);
323 MODULE_LICENSE("GPL v2");
324 MODULE_DESCRIPTION("MC Driver for Intel 10nm server processors");