blob: c1d4536ae466e7b3d683ecb0741baf7e7f653407 [file] [log] [blame]
Thomas Gleixner09c434b2019-05-19 13:08:20 +01001// SPDX-License-Identifier: GPL-2.0-only
Doug Thompson2bc65412009-05-04 20:11:14 +02002#include "amd64_edac.h"
Andreas Herrmann23ac4ae2010-09-17 18:03:43 +02003#include <asm/amd_nb.h>
Doug Thompson2bc65412009-05-04 20:11:14 +02004
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01005static struct edac_pci_ctl_info *pci_ctl;
Doug Thompson2bc65412009-05-04 20:11:14 +02006
7static int report_gart_errors;
8module_param(report_gart_errors, int, 0644);
9
10/*
11 * Set by command line parameter. If BIOS has enabled the ECC, this override is
12 * cleared to prevent re-enabling the hardware by this driver.
13 */
14static int ecc_enable_override;
15module_param(ecc_enable_override, int, 0644);
16
Tejun Heoa29d8b82010-02-02 14:39:15 +090017static struct msr __percpu *msrs;
Borislav Petkov50542252009-12-11 18:14:40 +010018
Borislav Petkov2ec591a2015-02-17 10:58:34 +010019/* Per-node stuff */
Borislav Petkovae7bb7c2010-10-14 16:01:30 +020020static struct ecc_settings **ecc_stngs;
Doug Thompson2bc65412009-05-04 20:11:14 +020021
Yazen Ghannambdcee772019-02-28 15:36:10 +000022/* Number of Unified Memory Controllers */
23static u8 num_umcs;
24
Doug Thompson2bc65412009-05-04 20:11:14 +020025/*
Borislav Petkovb70ef012009-06-25 19:32:38 +020026 * Valid scrub rates for the K8 hardware memory scrubber. We map the scrubbing
27 * bandwidth to a valid bit pattern. The 'set' operation finds the 'matching-
28 * or higher value'.
29 *
30 *FIXME: Produce a better mapping/linearisation.
31 */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +080032static const struct scrubrate {
Borislav Petkov39094442010-11-24 19:52:09 +010033 u32 scrubval; /* bit pattern for scrub rate */
34 u32 bandwidth; /* bandwidth consumed (bytes/sec) */
35} scrubrates[] = {
Borislav Petkovb70ef012009-06-25 19:32:38 +020036 { 0x01, 1600000000UL},
37 { 0x02, 800000000UL},
38 { 0x03, 400000000UL},
39 { 0x04, 200000000UL},
40 { 0x05, 100000000UL},
41 { 0x06, 50000000UL},
42 { 0x07, 25000000UL},
43 { 0x08, 12284069UL},
44 { 0x09, 6274509UL},
45 { 0x0A, 3121951UL},
46 { 0x0B, 1560975UL},
47 { 0x0C, 781440UL},
48 { 0x0D, 390720UL},
49 { 0x0E, 195300UL},
50 { 0x0F, 97650UL},
51 { 0x10, 48854UL},
52 { 0x11, 24427UL},
53 { 0x12, 12213UL},
54 { 0x13, 6101UL},
55 { 0x14, 3051UL},
56 { 0x15, 1523UL},
57 { 0x16, 761UL},
58 { 0x00, 0UL}, /* scrubbing off */
59};
60
Borislav Petkov66fed2d2012-08-09 18:41:07 +020061int __amd64_read_pci_cfg_dword(struct pci_dev *pdev, int offset,
62 u32 *val, const char *func)
Borislav Petkovb2b0c602010-10-08 18:32:29 +020063{
64 int err = 0;
65
66 err = pci_read_config_dword(pdev, offset, val);
67 if (err)
68 amd64_warn("%s: error reading F%dx%03x.\n",
69 func, PCI_FUNC(pdev->devfn), offset);
70
71 return err;
72}
73
74int __amd64_write_pci_cfg_dword(struct pci_dev *pdev, int offset,
75 u32 val, const char *func)
76{
77 int err = 0;
78
79 err = pci_write_config_dword(pdev, offset, val);
80 if (err)
81 amd64_warn("%s: error writing to F%dx%03x.\n",
82 func, PCI_FUNC(pdev->devfn), offset);
83
84 return err;
85}
86
87/*
Borislav Petkov73ba8592011-09-19 17:34:45 +020088 * Select DCT to which PCI cfg accesses are routed
89 */
90static void f15h_select_dct(struct amd64_pvt *pvt, u8 dct)
91{
92 u32 reg = 0;
93
94 amd64_read_pci_cfg(pvt->F1, DCT_CFG_SEL, &reg);
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -050095 reg &= (pvt->model == 0x30) ? ~3 : ~1;
Borislav Petkov73ba8592011-09-19 17:34:45 +020096 reg |= dct;
97 amd64_write_pci_cfg(pvt->F1, DCT_CFG_SEL, reg);
98}
99
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500100/*
101 *
102 * Depending on the family, F2 DCT reads need special handling:
103 *
104 * K8: has a single DCT only and no address offsets >= 0x100
105 *
106 * F10h: each DCT has its own set of regs
107 * DCT0 -> F2x040..
108 * DCT1 -> F2x140..
109 *
110 * F16h: has only 1 DCT
111 *
112 * F15h: we select which DCT we access using F1x10C[DctCfgSel]
113 */
114static inline int amd64_read_dct_pci_cfg(struct amd64_pvt *pvt, u8 dct,
115 int offset, u32 *val)
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200116{
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500117 switch (pvt->fam) {
118 case 0xf:
119 if (dct || offset >= 0x100)
120 return -EINVAL;
121 break;
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200122
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500123 case 0x10:
124 if (dct) {
125 /*
126 * Note: If ganging is enabled, barring the regs
127 * F2x[1,0]98 and F2x[1,0]9C; reads reads to F2x1xx
128 * return 0. (cf. Section 2.8.1 F10h BKDG)
129 */
130 if (dct_ganging_enabled(pvt))
131 return 0;
132
133 offset += 0x100;
134 }
135 break;
136
137 case 0x15:
138 /*
139 * F15h: F2x1xx addresses do not map explicitly to DCT1.
140 * We should select which DCT we access using F1x10C[DctCfgSel]
141 */
142 dct = (dct && pvt->model == 0x30) ? 3 : dct;
143 f15h_select_dct(pvt, dct);
144 break;
145
146 case 0x16:
147 if (dct)
148 return -EINVAL;
149 break;
150
151 default:
152 break;
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200153 }
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500154 return amd64_read_pci_cfg(pvt->F2, offset, val);
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200155}
156
Borislav Petkovb70ef012009-06-25 19:32:38 +0200157/*
Doug Thompson2bc65412009-05-04 20:11:14 +0200158 * Memory scrubber control interface. For K8, memory scrubbing is handled by
159 * hardware and can involve L2 cache, dcache as well as the main memory. With
160 * F10, this is extended to L3 cache scrubbing on CPU models sporting that
161 * functionality.
162 *
163 * This causes the "units" for the scrubbing speed to vary from 64 byte blocks
164 * (dram) over to cache lines. This is nasty, so we will use bandwidth in
165 * bytes/sec for the setting.
166 *
167 * Currently, we only do dram scrubbing. If the scrubbing is done in software on
168 * other archs, we might not have access to the caches directly.
169 */
170
Yazen Ghannam8051c0a2016-11-17 17:57:42 -0500171static inline void __f17h_set_scrubval(struct amd64_pvt *pvt, u32 scrubval)
172{
173 /*
174 * Fam17h supports scrub values between 0x5 and 0x14. Also, the values
175 * are shifted down by 0x5, so scrubval 0x5 is written to the register
176 * as 0x0, scrubval 0x6 as 0x1, etc.
177 */
178 if (scrubval >= 0x5 && scrubval <= 0x14) {
179 scrubval -= 0x5;
180 pci_write_bits32(pvt->F6, F17H_SCR_LIMIT_ADDR, scrubval, 0xF);
181 pci_write_bits32(pvt->F6, F17H_SCR_BASE_ADDR, 1, 0x1);
182 } else {
183 pci_write_bits32(pvt->F6, F17H_SCR_BASE_ADDR, 0, 0x1);
184 }
185}
Doug Thompson2bc65412009-05-04 20:11:14 +0200186/*
Yazen Ghannam8051c0a2016-11-17 17:57:42 -0500187 * Scan the scrub rate mapping table for a close or matching bandwidth value to
Doug Thompson2bc65412009-05-04 20:11:14 +0200188 * issue. If requested is too big, then use last maximum value found.
189 */
Aravind Gopalakrishnanda921102015-09-28 06:43:12 -0500190static int __set_scrub_rate(struct amd64_pvt *pvt, u32 new_bw, u32 min_rate)
Doug Thompson2bc65412009-05-04 20:11:14 +0200191{
192 u32 scrubval;
193 int i;
194
195 /*
196 * map the configured rate (new_bw) to a value specific to the AMD64
197 * memory controller and apply to register. Search for the first
198 * bandwidth entry that is greater or equal than the setting requested
199 * and program that. If at last entry, turn off DRAM scrubbing.
Andrew Morton168bfee2012-10-23 14:09:39 -0700200 *
201 * If no suitable bandwidth is found, turn off DRAM scrubbing entirely
202 * by falling back to the last element in scrubrates[].
Doug Thompson2bc65412009-05-04 20:11:14 +0200203 */
Andrew Morton168bfee2012-10-23 14:09:39 -0700204 for (i = 0; i < ARRAY_SIZE(scrubrates) - 1; i++) {
Doug Thompson2bc65412009-05-04 20:11:14 +0200205 /*
206 * skip scrub rates which aren't recommended
207 * (see F10 BKDG, F3x58)
208 */
Borislav Petkov395ae782010-10-01 18:38:19 +0200209 if (scrubrates[i].scrubval < min_rate)
Doug Thompson2bc65412009-05-04 20:11:14 +0200210 continue;
211
212 if (scrubrates[i].bandwidth <= new_bw)
213 break;
Doug Thompson2bc65412009-05-04 20:11:14 +0200214 }
215
216 scrubval = scrubrates[i].scrubval;
Doug Thompson2bc65412009-05-04 20:11:14 +0200217
Pu Wenc4a3e942018-09-27 16:31:28 +0200218 if (pvt->fam == 0x17 || pvt->fam == 0x18) {
Yazen Ghannam8051c0a2016-11-17 17:57:42 -0500219 __f17h_set_scrubval(pvt, scrubval);
220 } else if (pvt->fam == 0x15 && pvt->model == 0x60) {
Aravind Gopalakrishnanda921102015-09-28 06:43:12 -0500221 f15h_select_dct(pvt, 0);
222 pci_write_bits32(pvt->F2, F15H_M60H_SCRCTRL, scrubval, 0x001F);
223 f15h_select_dct(pvt, 1);
224 pci_write_bits32(pvt->F2, F15H_M60H_SCRCTRL, scrubval, 0x001F);
225 } else {
226 pci_write_bits32(pvt->F3, SCRCTRL, scrubval, 0x001F);
227 }
Doug Thompson2bc65412009-05-04 20:11:14 +0200228
Borislav Petkov39094442010-11-24 19:52:09 +0100229 if (scrubval)
230 return scrubrates[i].bandwidth;
231
Doug Thompson2bc65412009-05-04 20:11:14 +0200232 return 0;
233}
234
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100235static int set_scrub_rate(struct mem_ctl_info *mci, u32 bw)
Doug Thompson2bc65412009-05-04 20:11:14 +0200236{
237 struct amd64_pvt *pvt = mci->pvt_info;
Borislav Petkov87b3e0e2011-01-19 20:02:38 +0100238 u32 min_scrubrate = 0x5;
Doug Thompson2bc65412009-05-04 20:11:14 +0200239
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200240 if (pvt->fam == 0xf)
Borislav Petkov87b3e0e2011-01-19 20:02:38 +0100241 min_scrubrate = 0x0;
242
Aravind Gopalakrishnanda921102015-09-28 06:43:12 -0500243 if (pvt->fam == 0x15) {
244 /* Erratum #505 */
245 if (pvt->model < 0x10)
246 f15h_select_dct(pvt, 0);
Borislav Petkov73ba8592011-09-19 17:34:45 +0200247
Aravind Gopalakrishnanda921102015-09-28 06:43:12 -0500248 if (pvt->model == 0x60)
249 min_scrubrate = 0x6;
250 }
251 return __set_scrub_rate(pvt, bw, min_scrubrate);
Doug Thompson2bc65412009-05-04 20:11:14 +0200252}
253
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100254static int get_scrub_rate(struct mem_ctl_info *mci)
Doug Thompson2bc65412009-05-04 20:11:14 +0200255{
256 struct amd64_pvt *pvt = mci->pvt_info;
Borislav Petkov39094442010-11-24 19:52:09 +0100257 int i, retval = -EINVAL;
Yazen Ghannam8051c0a2016-11-17 17:57:42 -0500258 u32 scrubval = 0;
Doug Thompson2bc65412009-05-04 20:11:14 +0200259
Yazen Ghannam8051c0a2016-11-17 17:57:42 -0500260 switch (pvt->fam) {
261 case 0x15:
Aravind Gopalakrishnanda921102015-09-28 06:43:12 -0500262 /* Erratum #505 */
263 if (pvt->model < 0x10)
264 f15h_select_dct(pvt, 0);
Borislav Petkov73ba8592011-09-19 17:34:45 +0200265
Aravind Gopalakrishnanda921102015-09-28 06:43:12 -0500266 if (pvt->model == 0x60)
267 amd64_read_pci_cfg(pvt->F2, F15H_M60H_SCRCTRL, &scrubval);
Yazen Ghannam8051c0a2016-11-17 17:57:42 -0500268 break;
269
270 case 0x17:
Pu Wenc4a3e942018-09-27 16:31:28 +0200271 case 0x18:
Yazen Ghannam8051c0a2016-11-17 17:57:42 -0500272 amd64_read_pci_cfg(pvt->F6, F17H_SCR_BASE_ADDR, &scrubval);
273 if (scrubval & BIT(0)) {
274 amd64_read_pci_cfg(pvt->F6, F17H_SCR_LIMIT_ADDR, &scrubval);
275 scrubval &= 0xF;
276 scrubval += 0x5;
277 } else {
278 scrubval = 0;
279 }
280 break;
281
282 default:
Aravind Gopalakrishnanda921102015-09-28 06:43:12 -0500283 amd64_read_pci_cfg(pvt->F3, SCRCTRL, &scrubval);
Yazen Ghannam8051c0a2016-11-17 17:57:42 -0500284 break;
285 }
Doug Thompson2bc65412009-05-04 20:11:14 +0200286
287 scrubval = scrubval & 0x001F;
288
Roel Kluin926311f2010-01-11 20:58:21 +0100289 for (i = 0; i < ARRAY_SIZE(scrubrates); i++) {
Doug Thompson2bc65412009-05-04 20:11:14 +0200290 if (scrubrates[i].scrubval == scrubval) {
Borislav Petkov39094442010-11-24 19:52:09 +0100291 retval = scrubrates[i].bandwidth;
Doug Thompson2bc65412009-05-04 20:11:14 +0200292 break;
293 }
294 }
Borislav Petkov39094442010-11-24 19:52:09 +0100295 return retval;
Doug Thompson2bc65412009-05-04 20:11:14 +0200296}
297
Doug Thompson67757632009-04-27 15:53:22 +0200298/*
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200299 * returns true if the SysAddr given by sys_addr matches the
300 * DRAM base/limit associated with node_id
Doug Thompson67757632009-04-27 15:53:22 +0200301 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100302static bool base_limit_match(struct amd64_pvt *pvt, u64 sys_addr, u8 nid)
Doug Thompson67757632009-04-27 15:53:22 +0200303{
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200304 u64 addr;
Doug Thompson67757632009-04-27 15:53:22 +0200305
306 /* The K8 treats this as a 40-bit value. However, bits 63-40 will be
307 * all ones if the most significant implemented address bit is 1.
308 * Here we discard bits 63-40. See section 3.4.2 of AMD publication
309 * 24592: AMD x86-64 Architecture Programmer's Manual Volume 1
310 * Application Programming.
311 */
312 addr = sys_addr & 0x000000ffffffffffull;
313
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200314 return ((addr >= get_dram_base(pvt, nid)) &&
315 (addr <= get_dram_limit(pvt, nid)));
Doug Thompson67757632009-04-27 15:53:22 +0200316}
317
318/*
319 * Attempt to map a SysAddr to a node. On success, return a pointer to the
320 * mem_ctl_info structure for the node that the SysAddr maps to.
321 *
322 * On failure, return NULL.
323 */
324static struct mem_ctl_info *find_mc_by_sys_addr(struct mem_ctl_info *mci,
325 u64 sys_addr)
326{
327 struct amd64_pvt *pvt;
Daniel J Bluemanc7e53012012-11-30 16:44:20 +0800328 u8 node_id;
Doug Thompson67757632009-04-27 15:53:22 +0200329 u32 intlv_en, bits;
330
331 /*
332 * Here we use the DRAM Base (section 3.4.4.1) and DRAM Limit (section
333 * 3.4.4.2) registers to map the SysAddr to a node ID.
334 */
335 pvt = mci->pvt_info;
336
337 /*
338 * The value of this field should be the same for all DRAM Base
339 * registers. Therefore we arbitrarily choose to read it from the
340 * register for node 0.
341 */
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200342 intlv_en = dram_intlv_en(pvt, 0);
Doug Thompson67757632009-04-27 15:53:22 +0200343
344 if (intlv_en == 0) {
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200345 for (node_id = 0; node_id < DRAM_RANGES; node_id++) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100346 if (base_limit_match(pvt, sys_addr, node_id))
Borislav Petkov8edc5442009-09-18 12:39:19 +0200347 goto found;
Doug Thompson67757632009-04-27 15:53:22 +0200348 }
Borislav Petkov8edc5442009-09-18 12:39:19 +0200349 goto err_no_match;
Doug Thompson67757632009-04-27 15:53:22 +0200350 }
351
Borislav Petkov72f158f2009-09-18 12:27:27 +0200352 if (unlikely((intlv_en != 0x01) &&
353 (intlv_en != 0x03) &&
354 (intlv_en != 0x07))) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +0200355 amd64_warn("DRAM Base[IntlvEn] junk value: 0x%x, BIOS bug?\n", intlv_en);
Doug Thompson67757632009-04-27 15:53:22 +0200356 return NULL;
357 }
358
359 bits = (((u32) sys_addr) >> 12) & intlv_en;
360
361 for (node_id = 0; ; ) {
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200362 if ((dram_intlv_sel(pvt, node_id) & intlv_en) == bits)
Doug Thompson67757632009-04-27 15:53:22 +0200363 break; /* intlv_sel field matches */
364
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200365 if (++node_id >= DRAM_RANGES)
Doug Thompson67757632009-04-27 15:53:22 +0200366 goto err_no_match;
367 }
368
369 /* sanity test for sys_addr */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100370 if (unlikely(!base_limit_match(pvt, sys_addr, node_id))) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +0200371 amd64_warn("%s: sys_addr 0x%llx falls outside base/limit address"
372 "range for node %d with node interleaving enabled.\n",
373 __func__, sys_addr, node_id);
Doug Thompson67757632009-04-27 15:53:22 +0200374 return NULL;
375 }
376
377found:
Borislav Petkovb487c332011-02-21 18:55:00 +0100378 return edac_mc_find((int)node_id);
Doug Thompson67757632009-04-27 15:53:22 +0200379
380err_no_match:
Joe Perches956b9ba2012-04-29 17:08:39 -0300381 edac_dbg(2, "sys_addr 0x%lx doesn't match any node\n",
382 (unsigned long)sys_addr);
Doug Thompson67757632009-04-27 15:53:22 +0200383
384 return NULL;
385}
Doug Thompsone2ce7252009-04-27 15:57:12 +0200386
387/*
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100388 * compute the CS base address of the @csrow on the DRAM controller @dct.
389 * For details see F2x[5C:40] in the processor's BKDG
Doug Thompsone2ce7252009-04-27 15:57:12 +0200390 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100391static void get_cs_base_and_mask(struct amd64_pvt *pvt, int csrow, u8 dct,
392 u64 *base, u64 *mask)
Doug Thompsone2ce7252009-04-27 15:57:12 +0200393{
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100394 u64 csbase, csmask, base_bits, mask_bits;
395 u8 addr_shift;
396
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500397 if (pvt->fam == 0xf && pvt->ext_model < K8_REV_F) {
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100398 csbase = pvt->csels[dct].csbases[csrow];
399 csmask = pvt->csels[dct].csmasks[csrow];
Chen, Gong10ef6b02013-10-18 14:29:07 -0700400 base_bits = GENMASK_ULL(31, 21) | GENMASK_ULL(15, 9);
401 mask_bits = GENMASK_ULL(29, 21) | GENMASK_ULL(15, 9);
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100402 addr_shift = 4;
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500403
404 /*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500405 * F16h and F15h, models 30h and later need two addr_shift values:
406 * 8 for high and 6 for low (cf. F16h BKDG).
407 */
408 } else if (pvt->fam == 0x16 ||
409 (pvt->fam == 0x15 && pvt->model >= 0x30)) {
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500410 csbase = pvt->csels[dct].csbases[csrow];
411 csmask = pvt->csels[dct].csmasks[csrow >> 1];
412
Chen, Gong10ef6b02013-10-18 14:29:07 -0700413 *base = (csbase & GENMASK_ULL(15, 5)) << 6;
414 *base |= (csbase & GENMASK_ULL(30, 19)) << 8;
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500415
416 *mask = ~0ULL;
417 /* poke holes for the csmask */
Chen, Gong10ef6b02013-10-18 14:29:07 -0700418 *mask &= ~((GENMASK_ULL(15, 5) << 6) |
419 (GENMASK_ULL(30, 19) << 8));
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500420
Chen, Gong10ef6b02013-10-18 14:29:07 -0700421 *mask |= (csmask & GENMASK_ULL(15, 5)) << 6;
422 *mask |= (csmask & GENMASK_ULL(30, 19)) << 8;
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500423
424 return;
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100425 } else {
426 csbase = pvt->csels[dct].csbases[csrow];
427 csmask = pvt->csels[dct].csmasks[csrow >> 1];
428 addr_shift = 8;
429
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200430 if (pvt->fam == 0x15)
Chen, Gong10ef6b02013-10-18 14:29:07 -0700431 base_bits = mask_bits =
432 GENMASK_ULL(30,19) | GENMASK_ULL(13,5);
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100433 else
Chen, Gong10ef6b02013-10-18 14:29:07 -0700434 base_bits = mask_bits =
435 GENMASK_ULL(28,19) | GENMASK_ULL(13,5);
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100436 }
437
438 *base = (csbase & base_bits) << addr_shift;
439
440 *mask = ~0ULL;
441 /* poke holes for the csmask */
442 *mask &= ~(mask_bits << addr_shift);
443 /* OR them in */
444 *mask |= (csmask & mask_bits) << addr_shift;
Doug Thompsone2ce7252009-04-27 15:57:12 +0200445}
446
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100447#define for_each_chip_select(i, dct, pvt) \
448 for (i = 0; i < pvt->csels[dct].b_cnt; i++)
Doug Thompsone2ce7252009-04-27 15:57:12 +0200449
Borislav Petkov614ec9d2011-01-13 18:02:22 +0100450#define chip_select_base(i, dct, pvt) \
451 pvt->csels[dct].csbases[i]
452
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100453#define for_each_chip_select_mask(i, dct, pvt) \
454 for (i = 0; i < pvt->csels[dct].m_cnt; i++)
Doug Thompsone2ce7252009-04-27 15:57:12 +0200455
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +0000456#define for_each_umc(i) \
Yazen Ghannambdcee772019-02-28 15:36:10 +0000457 for (i = 0; i < num_umcs; i++)
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +0000458
Doug Thompsone2ce7252009-04-27 15:57:12 +0200459/*
460 * @input_addr is an InputAddr associated with the node given by mci. Return the
461 * csrow that input_addr maps to, or -1 on failure (no csrow claims input_addr).
462 */
463static int input_addr_to_csrow(struct mem_ctl_info *mci, u64 input_addr)
464{
465 struct amd64_pvt *pvt;
466 int csrow;
467 u64 base, mask;
468
469 pvt = mci->pvt_info;
470
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100471 for_each_chip_select(csrow, 0, pvt) {
472 if (!csrow_enabled(csrow, 0, pvt))
Doug Thompsone2ce7252009-04-27 15:57:12 +0200473 continue;
474
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100475 get_cs_base_and_mask(pvt, csrow, 0, &base, &mask);
476
477 mask = ~mask;
Doug Thompsone2ce7252009-04-27 15:57:12 +0200478
479 if ((input_addr & mask) == (base & mask)) {
Joe Perches956b9ba2012-04-29 17:08:39 -0300480 edac_dbg(2, "InputAddr 0x%lx matches csrow %d (node %d)\n",
481 (unsigned long)input_addr, csrow,
482 pvt->mc_node_id);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200483
484 return csrow;
485 }
486 }
Joe Perches956b9ba2012-04-29 17:08:39 -0300487 edac_dbg(2, "no matching csrow for InputAddr 0x%lx (MC node %d)\n",
488 (unsigned long)input_addr, pvt->mc_node_id);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200489
490 return -1;
491}
492
493/*
Doug Thompsone2ce7252009-04-27 15:57:12 +0200494 * Obtain info from the DRAM Hole Address Register (section 3.4.8, pub #26094)
495 * for the node represented by mci. Info is passed back in *hole_base,
496 * *hole_offset, and *hole_size. Function returns 0 if info is valid or 1 if
497 * info is invalid. Info may be invalid for either of the following reasons:
498 *
499 * - The revision of the node is not E or greater. In this case, the DRAM Hole
500 * Address Register does not exist.
501 *
502 * - The DramHoleValid bit is cleared in the DRAM Hole Address Register,
503 * indicating that its contents are not valid.
504 *
505 * The values passed back in *hole_base, *hole_offset, and *hole_size are
506 * complete 32-bit values despite the fact that the bitfields in the DHAR
507 * only represent bits 31-24 of the base and offset values.
508 */
509int amd64_get_dram_hole_info(struct mem_ctl_info *mci, u64 *hole_base,
510 u64 *hole_offset, u64 *hole_size)
511{
512 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompsone2ce7252009-04-27 15:57:12 +0200513
514 /* only revE and later have the DRAM Hole Address Register */
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200515 if (pvt->fam == 0xf && pvt->ext_model < K8_REV_E) {
Joe Perches956b9ba2012-04-29 17:08:39 -0300516 edac_dbg(1, " revision %d for node %d does not support DHAR\n",
517 pvt->ext_model, pvt->mc_node_id);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200518 return 1;
519 }
520
Borislav Petkovbc21fa52010-11-11 17:29:13 +0100521 /* valid for Fam10h and above */
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200522 if (pvt->fam >= 0x10 && !dhar_mem_hoist_valid(pvt)) {
Joe Perches956b9ba2012-04-29 17:08:39 -0300523 edac_dbg(1, " Dram Memory Hoisting is DISABLED on this system\n");
Doug Thompsone2ce7252009-04-27 15:57:12 +0200524 return 1;
525 }
526
Borislav Petkovc8e518d2010-12-10 19:49:19 +0100527 if (!dhar_valid(pvt)) {
Joe Perches956b9ba2012-04-29 17:08:39 -0300528 edac_dbg(1, " Dram Memory Hoisting is DISABLED on this node %d\n",
529 pvt->mc_node_id);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200530 return 1;
531 }
532
533 /* This node has Memory Hoisting */
534
535 /* +------------------+--------------------+--------------------+-----
536 * | memory | DRAM hole | relocated |
537 * | [0, (x - 1)] | [x, 0xffffffff] | addresses from |
538 * | | | DRAM hole |
539 * | | | [0x100000000, |
540 * | | | (0x100000000+ |
541 * | | | (0xffffffff-x))] |
542 * +------------------+--------------------+--------------------+-----
543 *
544 * Above is a diagram of physical memory showing the DRAM hole and the
545 * relocated addresses from the DRAM hole. As shown, the DRAM hole
546 * starts at address x (the base address) and extends through address
547 * 0xffffffff. The DRAM Hole Address Register (DHAR) relocates the
548 * addresses in the hole so that they start at 0x100000000.
549 */
550
Borislav Petkov1f316772012-08-10 12:50:50 +0200551 *hole_base = dhar_base(pvt);
552 *hole_size = (1ULL << 32) - *hole_base;
Doug Thompsone2ce7252009-04-27 15:57:12 +0200553
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200554 *hole_offset = (pvt->fam > 0xf) ? f10_dhar_offset(pvt)
555 : k8_dhar_offset(pvt);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200556
Joe Perches956b9ba2012-04-29 17:08:39 -0300557 edac_dbg(1, " DHAR info for node %d base 0x%lx offset 0x%lx size 0x%lx\n",
558 pvt->mc_node_id, (unsigned long)*hole_base,
559 (unsigned long)*hole_offset, (unsigned long)*hole_size);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200560
561 return 0;
562}
563EXPORT_SYMBOL_GPL(amd64_get_dram_hole_info);
564
Doug Thompson93c2df52009-05-04 20:46:50 +0200565/*
566 * Return the DramAddr that the SysAddr given by @sys_addr maps to. It is
567 * assumed that sys_addr maps to the node given by mci.
568 *
569 * The first part of section 3.4.4 (p. 70) shows how the DRAM Base (section
570 * 3.4.4.1) and DRAM Limit (section 3.4.4.2) registers are used to translate a
571 * SysAddr to a DramAddr. If the DRAM Hole Address Register (DHAR) is enabled,
572 * then it is also involved in translating a SysAddr to a DramAddr. Sections
573 * 3.4.8 and 3.5.8.2 describe the DHAR and how it is used for memory hoisting.
574 * These parts of the documentation are unclear. I interpret them as follows:
575 *
576 * When node n receives a SysAddr, it processes the SysAddr as follows:
577 *
578 * 1. It extracts the DRAMBase and DRAMLimit values from the DRAM Base and DRAM
579 * Limit registers for node n. If the SysAddr is not within the range
580 * specified by the base and limit values, then node n ignores the Sysaddr
581 * (since it does not map to node n). Otherwise continue to step 2 below.
582 *
583 * 2. If the DramHoleValid bit of the DHAR for node n is clear, the DHAR is
584 * disabled so skip to step 3 below. Otherwise see if the SysAddr is within
585 * the range of relocated addresses (starting at 0x100000000) from the DRAM
586 * hole. If not, skip to step 3 below. Else get the value of the
587 * DramHoleOffset field from the DHAR. To obtain the DramAddr, subtract the
588 * offset defined by this value from the SysAddr.
589 *
590 * 3. Obtain the base address for node n from the DRAMBase field of the DRAM
591 * Base register for node n. To obtain the DramAddr, subtract the base
592 * address from the SysAddr, as shown near the start of section 3.4.4 (p.70).
593 */
594static u64 sys_addr_to_dram_addr(struct mem_ctl_info *mci, u64 sys_addr)
595{
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200596 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompson93c2df52009-05-04 20:46:50 +0200597 u64 dram_base, hole_base, hole_offset, hole_size, dram_addr;
Borislav Petkov1f316772012-08-10 12:50:50 +0200598 int ret;
Doug Thompson93c2df52009-05-04 20:46:50 +0200599
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200600 dram_base = get_dram_base(pvt, pvt->mc_node_id);
Doug Thompson93c2df52009-05-04 20:46:50 +0200601
602 ret = amd64_get_dram_hole_info(mci, &hole_base, &hole_offset,
603 &hole_size);
604 if (!ret) {
Borislav Petkov1f316772012-08-10 12:50:50 +0200605 if ((sys_addr >= (1ULL << 32)) &&
606 (sys_addr < ((1ULL << 32) + hole_size))) {
Doug Thompson93c2df52009-05-04 20:46:50 +0200607 /* use DHAR to translate SysAddr to DramAddr */
608 dram_addr = sys_addr - hole_offset;
609
Joe Perches956b9ba2012-04-29 17:08:39 -0300610 edac_dbg(2, "using DHAR to translate SysAddr 0x%lx to DramAddr 0x%lx\n",
611 (unsigned long)sys_addr,
612 (unsigned long)dram_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200613
614 return dram_addr;
615 }
616 }
617
618 /*
619 * Translate the SysAddr to a DramAddr as shown near the start of
620 * section 3.4.4 (p. 70). Although sys_addr is a 64-bit value, the k8
621 * only deals with 40-bit values. Therefore we discard bits 63-40 of
622 * sys_addr below. If bit 39 of sys_addr is 1 then the bits we
623 * discard are all 1s. Otherwise the bits we discard are all 0s. See
624 * section 3.4.2 of AMD publication 24592: AMD x86-64 Architecture
625 * Programmer's Manual Volume 1 Application Programming.
626 */
Chen, Gong10ef6b02013-10-18 14:29:07 -0700627 dram_addr = (sys_addr & GENMASK_ULL(39, 0)) - dram_base;
Doug Thompson93c2df52009-05-04 20:46:50 +0200628
Joe Perches956b9ba2012-04-29 17:08:39 -0300629 edac_dbg(2, "using DRAM Base register to translate SysAddr 0x%lx to DramAddr 0x%lx\n",
630 (unsigned long)sys_addr, (unsigned long)dram_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200631 return dram_addr;
632}
633
634/*
635 * @intlv_en is the value of the IntlvEn field from a DRAM Base register
636 * (section 3.4.4.1). Return the number of bits from a SysAddr that are used
637 * for node interleaving.
638 */
639static int num_node_interleave_bits(unsigned intlv_en)
640{
641 static const int intlv_shift_table[] = { 0, 1, 0, 2, 0, 0, 0, 3 };
642 int n;
643
644 BUG_ON(intlv_en > 7);
645 n = intlv_shift_table[intlv_en];
646 return n;
647}
648
649/* Translate the DramAddr given by @dram_addr to an InputAddr. */
650static u64 dram_addr_to_input_addr(struct mem_ctl_info *mci, u64 dram_addr)
651{
652 struct amd64_pvt *pvt;
653 int intlv_shift;
654 u64 input_addr;
655
656 pvt = mci->pvt_info;
657
658 /*
659 * See the start of section 3.4.4 (p. 70, BKDG #26094, K8, revA-E)
660 * concerning translating a DramAddr to an InputAddr.
661 */
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200662 intlv_shift = num_node_interleave_bits(dram_intlv_en(pvt, 0));
Chen, Gong10ef6b02013-10-18 14:29:07 -0700663 input_addr = ((dram_addr >> intlv_shift) & GENMASK_ULL(35, 12)) +
Borislav Petkovf678b8c2010-12-13 19:21:07 +0100664 (dram_addr & 0xfff);
Doug Thompson93c2df52009-05-04 20:46:50 +0200665
Joe Perches956b9ba2012-04-29 17:08:39 -0300666 edac_dbg(2, " Intlv Shift=%d DramAddr=0x%lx maps to InputAddr=0x%lx\n",
667 intlv_shift, (unsigned long)dram_addr,
668 (unsigned long)input_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200669
670 return input_addr;
671}
672
673/*
674 * Translate the SysAddr represented by @sys_addr to an InputAddr. It is
675 * assumed that @sys_addr maps to the node given by mci.
676 */
677static u64 sys_addr_to_input_addr(struct mem_ctl_info *mci, u64 sys_addr)
678{
679 u64 input_addr;
680
681 input_addr =
682 dram_addr_to_input_addr(mci, sys_addr_to_dram_addr(mci, sys_addr));
683
Masanari Iidac19ca6c2016-02-08 20:53:12 +0900684 edac_dbg(2, "SysAddr 0x%lx translates to InputAddr 0x%lx\n",
Joe Perches956b9ba2012-04-29 17:08:39 -0300685 (unsigned long)sys_addr, (unsigned long)input_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200686
687 return input_addr;
688}
689
Doug Thompson93c2df52009-05-04 20:46:50 +0200690/* Map the Error address to a PAGE and PAGE OFFSET. */
691static inline void error_address_to_page_and_offset(u64 error_address,
Borislav Petkov33ca0642012-08-30 18:01:36 +0200692 struct err_info *err)
Doug Thompson93c2df52009-05-04 20:46:50 +0200693{
Borislav Petkov33ca0642012-08-30 18:01:36 +0200694 err->page = (u32) (error_address >> PAGE_SHIFT);
695 err->offset = ((u32) error_address) & ~PAGE_MASK;
Doug Thompson93c2df52009-05-04 20:46:50 +0200696}
697
698/*
699 * @sys_addr is an error address (a SysAddr) extracted from the MCA NB Address
700 * Low (section 3.6.4.5) and MCA NB Address High (section 3.6.4.6) registers
701 * of a node that detected an ECC memory error. mci represents the node that
702 * the error address maps to (possibly different from the node that detected
703 * the error). Return the number of the csrow that sys_addr maps to, or -1 on
704 * error.
705 */
706static int sys_addr_to_csrow(struct mem_ctl_info *mci, u64 sys_addr)
707{
708 int csrow;
709
710 csrow = input_addr_to_csrow(mci, sys_addr_to_input_addr(mci, sys_addr));
711
712 if (csrow == -1)
Borislav Petkov24f9a7f2010-10-07 18:29:15 +0200713 amd64_mc_err(mci, "Failed to translate InputAddr to csrow for "
714 "address 0x%lx\n", (unsigned long)sys_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200715 return csrow;
716}
Doug Thompsone2ce7252009-04-27 15:57:12 +0200717
Borislav Petkovbfc04ae2009-11-12 19:05:07 +0100718static int get_channel_from_ecc_syndrome(struct mem_ctl_info *, u16);
Doug Thompson2da11652009-04-27 16:09:09 +0200719
Doug Thompson2da11652009-04-27 16:09:09 +0200720/*
721 * Determine if the DIMMs have ECC enabled. ECC is enabled ONLY if all the DIMMs
722 * are ECC capable.
723 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100724static unsigned long determine_edac_cap(struct amd64_pvt *pvt)
Doug Thompson2da11652009-04-27 16:09:09 +0200725{
Dan Carpenter1f6189e2011-10-06 02:30:25 -0400726 unsigned long edac_cap = EDAC_FLAG_NONE;
Yazen Ghannamd27f3a32016-11-17 17:57:40 -0500727 u8 bit;
Doug Thompson2da11652009-04-27 16:09:09 +0200728
Yazen Ghannamd27f3a32016-11-17 17:57:40 -0500729 if (pvt->umc) {
730 u8 i, umc_en_mask = 0, dimm_ecc_en_mask = 0;
Doug Thompson2da11652009-04-27 16:09:09 +0200731
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +0000732 for_each_umc(i) {
Yazen Ghannamd27f3a32016-11-17 17:57:40 -0500733 if (!(pvt->umc[i].sdp_ctrl & UMC_SDP_INIT))
734 continue;
735
736 umc_en_mask |= BIT(i);
737
738 /* UMC Configuration bit 12 (DimmEccEn) */
739 if (pvt->umc[i].umc_cfg & BIT(12))
740 dimm_ecc_en_mask |= BIT(i);
741 }
742
743 if (umc_en_mask == dimm_ecc_en_mask)
744 edac_cap = EDAC_FLAG_SECDED;
745 } else {
746 bit = (pvt->fam > 0xf || pvt->ext_model >= K8_REV_F)
747 ? 19
748 : 17;
749
750 if (pvt->dclr0 & BIT(bit))
751 edac_cap = EDAC_FLAG_SECDED;
752 }
Doug Thompson2da11652009-04-27 16:09:09 +0200753
754 return edac_cap;
755}
756
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100757static void debug_display_dimm_sizes(struct amd64_pvt *, u8);
Doug Thompson2da11652009-04-27 16:09:09 +0200758
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100759static void debug_dump_dramcfg_low(struct amd64_pvt *pvt, u32 dclr, int chan)
Borislav Petkov68798e12009-11-03 16:18:33 +0100760{
Joe Perches956b9ba2012-04-29 17:08:39 -0300761 edac_dbg(1, "F2x%d90 (DRAM Cfg Low): 0x%08x\n", chan, dclr);
Borislav Petkov68798e12009-11-03 16:18:33 +0100762
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100763 if (pvt->dram_type == MEM_LRDDR3) {
764 u32 dcsm = pvt->csels[chan].csmasks[0];
765 /*
766 * It's assumed all LRDIMMs in a DCT are going to be of
767 * same 'type' until proven otherwise. So, use a cs
768 * value of '0' here to get dcsm value.
769 */
770 edac_dbg(1, " LRDIMM %dx rank multiply\n", (dcsm & 0x3));
771 }
772
773 edac_dbg(1, "All DIMMs support ECC:%s\n",
774 (dclr & BIT(19)) ? "yes" : "no");
775
Borislav Petkov68798e12009-11-03 16:18:33 +0100776
Joe Perches956b9ba2012-04-29 17:08:39 -0300777 edac_dbg(1, " PAR/ERR parity: %s\n",
778 (dclr & BIT(8)) ? "enabled" : "disabled");
Borislav Petkov68798e12009-11-03 16:18:33 +0100779
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200780 if (pvt->fam == 0x10)
Joe Perches956b9ba2012-04-29 17:08:39 -0300781 edac_dbg(1, " DCT 128bit mode width: %s\n",
782 (dclr & BIT(11)) ? "128b" : "64b");
Borislav Petkov68798e12009-11-03 16:18:33 +0100783
Joe Perches956b9ba2012-04-29 17:08:39 -0300784 edac_dbg(1, " x4 logical DIMMs present: L0: %s L1: %s L2: %s L3: %s\n",
785 (dclr & BIT(12)) ? "yes" : "no",
786 (dclr & BIT(13)) ? "yes" : "no",
787 (dclr & BIT(14)) ? "yes" : "no",
788 (dclr & BIT(15)) ? "yes" : "no");
Borislav Petkov68798e12009-11-03 16:18:33 +0100789}
790
Yazen Ghanname53a3b22019-08-21 23:59:59 +0000791#define CS_EVEN_PRIMARY BIT(0)
792#define CS_ODD_PRIMARY BIT(1)
Yazen Ghannam81f50902019-08-22 00:00:02 +0000793#define CS_EVEN_SECONDARY BIT(2)
794#define CS_ODD_SECONDARY BIT(3)
Yazen Ghanname53a3b22019-08-21 23:59:59 +0000795
Yazen Ghannam81f50902019-08-22 00:00:02 +0000796#define CS_EVEN (CS_EVEN_PRIMARY | CS_EVEN_SECONDARY)
797#define CS_ODD (CS_ODD_PRIMARY | CS_ODD_SECONDARY)
Yazen Ghanname53a3b22019-08-21 23:59:59 +0000798
799static int f17_get_cs_mode(int dimm, u8 ctrl, struct amd64_pvt *pvt)
Yazen Ghannamfc00c6a2019-02-28 15:36:12 +0000800{
Yazen Ghanname53a3b22019-08-21 23:59:59 +0000801 int cs_mode = 0;
Yazen Ghannamfc00c6a2019-02-28 15:36:12 +0000802
Yazen Ghanname53a3b22019-08-21 23:59:59 +0000803 if (csrow_enabled(2 * dimm, ctrl, pvt))
804 cs_mode |= CS_EVEN_PRIMARY;
Yazen Ghannamfc00c6a2019-02-28 15:36:12 +0000805
Yazen Ghanname53a3b22019-08-21 23:59:59 +0000806 if (csrow_enabled(2 * dimm + 1, ctrl, pvt))
807 cs_mode |= CS_ODD_PRIMARY;
808
Yazen Ghannam81f50902019-08-22 00:00:02 +0000809 /* Asymmetric dual-rank DIMM support. */
810 if (csrow_sec_enabled(2 * dimm + 1, ctrl, pvt))
811 cs_mode |= CS_ODD_SECONDARY;
812
Yazen Ghanname53a3b22019-08-21 23:59:59 +0000813 return cs_mode;
Yazen Ghannamfc00c6a2019-02-28 15:36:12 +0000814}
815
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600816static void debug_display_dimm_sizes_df(struct amd64_pvt *pvt, u8 ctrl)
817{
Yazen Ghanname53a3b22019-08-21 23:59:59 +0000818 int dimm, size0, size1, cs0, cs1, cs_mode;
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600819
820 edac_printk(KERN_DEBUG, EDAC_MC, "UMC%d chip selects:\n", ctrl);
821
Yazen Ghannamd971e282019-08-21 23:59:55 +0000822 for (dimm = 0; dimm < 2; dimm++) {
Yazen Ghannameb77e6b2017-04-27 12:11:54 -0500823 cs0 = dimm * 2;
Yazen Ghannameb77e6b2017-04-27 12:11:54 -0500824 cs1 = dimm * 2 + 1;
825
Yazen Ghanname53a3b22019-08-21 23:59:59 +0000826 cs_mode = f17_get_cs_mode(dimm, ctrl, pvt);
827
828 size0 = pvt->ops->dbam_to_cs(pvt, ctrl, cs_mode, cs0);
829 size1 = pvt->ops->dbam_to_cs(pvt, ctrl, cs_mode, cs1);
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600830
831 amd64_info(EDAC_MC ": %d: %5dMB %d: %5dMB\n",
Yazen Ghannameb77e6b2017-04-27 12:11:54 -0500832 cs0, size0,
833 cs1, size1);
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600834 }
835}
836
837static void __dump_misc_regs_df(struct amd64_pvt *pvt)
838{
839 struct amd64_umc *umc;
840 u32 i, tmp, umc_base;
841
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +0000842 for_each_umc(i) {
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600843 umc_base = get_umc_base(i);
844 umc = &pvt->umc[i];
845
846 edac_dbg(1, "UMC%d DIMM cfg: 0x%x\n", i, umc->dimm_cfg);
847 edac_dbg(1, "UMC%d UMC cfg: 0x%x\n", i, umc->umc_cfg);
848 edac_dbg(1, "UMC%d SDP ctrl: 0x%x\n", i, umc->sdp_ctrl);
849 edac_dbg(1, "UMC%d ECC ctrl: 0x%x\n", i, umc->ecc_ctrl);
850
851 amd_smn_read(pvt->mc_node_id, umc_base + UMCCH_ECC_BAD_SYMBOL, &tmp);
852 edac_dbg(1, "UMC%d ECC bad symbol: 0x%x\n", i, tmp);
853
854 amd_smn_read(pvt->mc_node_id, umc_base + UMCCH_UMC_CAP, &tmp);
855 edac_dbg(1, "UMC%d UMC cap: 0x%x\n", i, tmp);
856 edac_dbg(1, "UMC%d UMC cap high: 0x%x\n", i, umc->umc_cap_hi);
857
858 edac_dbg(1, "UMC%d ECC capable: %s, ChipKill ECC capable: %s\n",
859 i, (umc->umc_cap_hi & BIT(30)) ? "yes" : "no",
860 (umc->umc_cap_hi & BIT(31)) ? "yes" : "no");
861 edac_dbg(1, "UMC%d All DIMMs support ECC: %s\n",
862 i, (umc->umc_cfg & BIT(12)) ? "yes" : "no");
863 edac_dbg(1, "UMC%d x4 DIMMs present: %s\n",
864 i, (umc->dimm_cfg & BIT(6)) ? "yes" : "no");
865 edac_dbg(1, "UMC%d x16 DIMMs present: %s\n",
866 i, (umc->dimm_cfg & BIT(7)) ? "yes" : "no");
867
868 if (pvt->dram_type == MEM_LRDDR4) {
869 amd_smn_read(pvt->mc_node_id, umc_base + UMCCH_ADDR_CFG, &tmp);
870 edac_dbg(1, "UMC%d LRDIMM %dx rank multiply\n",
871 i, 1 << ((tmp >> 4) & 0x3));
872 }
873
874 debug_display_dimm_sizes_df(pvt, i);
875 }
876
877 edac_dbg(1, "F0x104 (DRAM Hole Address): 0x%08x, base: 0x%08x\n",
878 pvt->dhar, dhar_base(pvt));
879}
880
Doug Thompson2da11652009-04-27 16:09:09 +0200881/* Display and decode various NB registers for debug purposes. */
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600882static void __dump_misc_regs(struct amd64_pvt *pvt)
Doug Thompson2da11652009-04-27 16:09:09 +0200883{
Joe Perches956b9ba2012-04-29 17:08:39 -0300884 edac_dbg(1, "F3xE8 (NB Cap): 0x%08x\n", pvt->nbcap);
Doug Thompson2da11652009-04-27 16:09:09 +0200885
Joe Perches956b9ba2012-04-29 17:08:39 -0300886 edac_dbg(1, " NB two channel DRAM capable: %s\n",
887 (pvt->nbcap & NBCAP_DCT_DUAL) ? "yes" : "no");
Borislav Petkov68798e12009-11-03 16:18:33 +0100888
Joe Perches956b9ba2012-04-29 17:08:39 -0300889 edac_dbg(1, " ECC capable: %s, ChipKill ECC capable: %s\n",
890 (pvt->nbcap & NBCAP_SECDED) ? "yes" : "no",
891 (pvt->nbcap & NBCAP_CHIPKILL) ? "yes" : "no");
Borislav Petkov68798e12009-11-03 16:18:33 +0100892
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100893 debug_dump_dramcfg_low(pvt, pvt->dclr0, 0);
Doug Thompson2da11652009-04-27 16:09:09 +0200894
Joe Perches956b9ba2012-04-29 17:08:39 -0300895 edac_dbg(1, "F3xB0 (Online Spare): 0x%08x\n", pvt->online_spare);
Doug Thompson2da11652009-04-27 16:09:09 +0200896
Joe Perches956b9ba2012-04-29 17:08:39 -0300897 edac_dbg(1, "F1xF0 (DRAM Hole Address): 0x%08x, base: 0x%08x, offset: 0x%08x\n",
898 pvt->dhar, dhar_base(pvt),
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200899 (pvt->fam == 0xf) ? k8_dhar_offset(pvt)
900 : f10_dhar_offset(pvt));
Doug Thompson2da11652009-04-27 16:09:09 +0200901
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100902 debug_display_dimm_sizes(pvt, 0);
Borislav Petkov4d796362011-02-03 15:59:57 +0100903
Borislav Petkov8de1d912009-10-16 13:39:30 +0200904 /* everything below this point is Fam10h and above */
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200905 if (pvt->fam == 0xf)
Doug Thompson2da11652009-04-27 16:09:09 +0200906 return;
Borislav Petkov4d796362011-02-03 15:59:57 +0100907
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100908 debug_display_dimm_sizes(pvt, 1);
Doug Thompson2da11652009-04-27 16:09:09 +0200909
Borislav Petkov8de1d912009-10-16 13:39:30 +0200910 /* Only if NOT ganged does dclr1 have valid info */
Borislav Petkov68798e12009-11-03 16:18:33 +0100911 if (!dct_ganging_enabled(pvt))
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100912 debug_dump_dramcfg_low(pvt, pvt->dclr1, 1);
Doug Thompson2da11652009-04-27 16:09:09 +0200913}
914
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600915/* Display and decode various NB registers for debug purposes. */
916static void dump_misc_regs(struct amd64_pvt *pvt)
917{
918 if (pvt->umc)
919 __dump_misc_regs_df(pvt);
920 else
921 __dump_misc_regs(pvt);
922
923 edac_dbg(1, " DramHoleValid: %s\n", dhar_valid(pvt) ? "yes" : "no");
924
Yazen Ghannam78359612019-02-28 15:36:11 +0000925 amd64_info("using x%u syndromes.\n", pvt->ecc_sym_sz);
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600926}
927
Doug Thompson94be4bf2009-04-27 16:12:00 +0200928/*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500929 * See BKDG, F2x[1,0][5C:40], F2[1,0][6C:60]
Doug Thompson94be4bf2009-04-27 16:12:00 +0200930 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100931static void prep_chip_selects(struct amd64_pvt *pvt)
Doug Thompson94be4bf2009-04-27 16:12:00 +0200932{
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500933 if (pvt->fam == 0xf && pvt->ext_model < K8_REV_F) {
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100934 pvt->csels[0].b_cnt = pvt->csels[1].b_cnt = 8;
935 pvt->csels[0].m_cnt = pvt->csels[1].m_cnt = 8;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100936 } else if (pvt->fam == 0x15 && pvt->model == 0x30) {
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500937 pvt->csels[0].b_cnt = pvt->csels[1].b_cnt = 4;
938 pvt->csels[0].m_cnt = pvt->csels[1].m_cnt = 2;
Yazen Ghannamd971e282019-08-21 23:59:55 +0000939 } else if (pvt->fam >= 0x17) {
940 int umc;
941
942 for_each_umc(umc) {
943 pvt->csels[umc].b_cnt = 4;
944 pvt->csels[umc].m_cnt = 2;
945 }
946
Borislav Petkov9d858bb2009-09-21 14:35:51 +0200947 } else {
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100948 pvt->csels[0].b_cnt = pvt->csels[1].b_cnt = 8;
949 pvt->csels[0].m_cnt = pvt->csels[1].m_cnt = 4;
Doug Thompson94be4bf2009-04-27 16:12:00 +0200950 }
951}
952
Yazen Ghannamd971e282019-08-21 23:59:55 +0000953static void read_umc_base_mask(struct amd64_pvt *pvt)
954{
Yazen Ghannam75747292019-08-22 00:00:01 +0000955 u32 umc_base_reg, umc_base_reg_sec;
956 u32 umc_mask_reg, umc_mask_reg_sec;
957 u32 base_reg, base_reg_sec;
958 u32 mask_reg, mask_reg_sec;
959 u32 *base, *base_sec;
960 u32 *mask, *mask_sec;
Yazen Ghannamd971e282019-08-21 23:59:55 +0000961 int cs, umc;
962
963 for_each_umc(umc) {
964 umc_base_reg = get_umc_base(umc) + UMCCH_BASE_ADDR;
Yazen Ghannam75747292019-08-22 00:00:01 +0000965 umc_base_reg_sec = get_umc_base(umc) + UMCCH_BASE_ADDR_SEC;
Yazen Ghannamd971e282019-08-21 23:59:55 +0000966
967 for_each_chip_select(cs, umc, pvt) {
968 base = &pvt->csels[umc].csbases[cs];
Yazen Ghannam75747292019-08-22 00:00:01 +0000969 base_sec = &pvt->csels[umc].csbases_sec[cs];
Yazen Ghannamd971e282019-08-21 23:59:55 +0000970
971 base_reg = umc_base_reg + (cs * 4);
Yazen Ghannam75747292019-08-22 00:00:01 +0000972 base_reg_sec = umc_base_reg_sec + (cs * 4);
Yazen Ghannamd971e282019-08-21 23:59:55 +0000973
974 if (!amd_smn_read(pvt->mc_node_id, base_reg, base))
975 edac_dbg(0, " DCSB%d[%d]=0x%08x reg: 0x%x\n",
976 umc, cs, *base, base_reg);
Yazen Ghannam75747292019-08-22 00:00:01 +0000977
978 if (!amd_smn_read(pvt->mc_node_id, base_reg_sec, base_sec))
979 edac_dbg(0, " DCSB_SEC%d[%d]=0x%08x reg: 0x%x\n",
980 umc, cs, *base_sec, base_reg_sec);
Yazen Ghannamd971e282019-08-21 23:59:55 +0000981 }
982
983 umc_mask_reg = get_umc_base(umc) + UMCCH_ADDR_MASK;
Yazen Ghannam75747292019-08-22 00:00:01 +0000984 umc_mask_reg_sec = get_umc_base(umc) + UMCCH_ADDR_MASK_SEC;
Yazen Ghannamd971e282019-08-21 23:59:55 +0000985
986 for_each_chip_select_mask(cs, umc, pvt) {
987 mask = &pvt->csels[umc].csmasks[cs];
Yazen Ghannam75747292019-08-22 00:00:01 +0000988 mask_sec = &pvt->csels[umc].csmasks_sec[cs];
Yazen Ghannamd971e282019-08-21 23:59:55 +0000989
990 mask_reg = umc_mask_reg + (cs * 4);
Yazen Ghannam75747292019-08-22 00:00:01 +0000991 mask_reg_sec = umc_mask_reg_sec + (cs * 4);
Yazen Ghannamd971e282019-08-21 23:59:55 +0000992
993 if (!amd_smn_read(pvt->mc_node_id, mask_reg, mask))
994 edac_dbg(0, " DCSM%d[%d]=0x%08x reg: 0x%x\n",
995 umc, cs, *mask, mask_reg);
Yazen Ghannam75747292019-08-22 00:00:01 +0000996
997 if (!amd_smn_read(pvt->mc_node_id, mask_reg_sec, mask_sec))
998 edac_dbg(0, " DCSM_SEC%d[%d]=0x%08x reg: 0x%x\n",
999 umc, cs, *mask_sec, mask_reg_sec);
Yazen Ghannamd971e282019-08-21 23:59:55 +00001000 }
1001 }
1002}
1003
Doug Thompson94be4bf2009-04-27 16:12:00 +02001004/*
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001005 * Function 2 Offset F10_DCSB0; read in the DCS Base and DCS Mask registers
Doug Thompson94be4bf2009-04-27 16:12:00 +02001006 */
Borislav Petkovb2b0c602010-10-08 18:32:29 +02001007static void read_dct_base_mask(struct amd64_pvt *pvt)
Doug Thompson94be4bf2009-04-27 16:12:00 +02001008{
Yazen Ghannamd971e282019-08-21 23:59:55 +00001009 int cs;
Doug Thompson94be4bf2009-04-27 16:12:00 +02001010
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001011 prep_chip_selects(pvt);
Doug Thompson94be4bf2009-04-27 16:12:00 +02001012
Yazen Ghannamd971e282019-08-21 23:59:55 +00001013 if (pvt->umc)
1014 return read_umc_base_mask(pvt);
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05001015
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001016 for_each_chip_select(cs, 0, pvt) {
Yazen Ghannamd971e282019-08-21 23:59:55 +00001017 int reg0 = DCSB0 + (cs * 4);
1018 int reg1 = DCSB1 + (cs * 4);
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001019 u32 *base0 = &pvt->csels[0].csbases[cs];
1020 u32 *base1 = &pvt->csels[1].csbases[cs];
Borislav Petkovb2b0c602010-10-08 18:32:29 +02001021
Yazen Ghannamd971e282019-08-21 23:59:55 +00001022 if (!amd64_read_dct_pci_cfg(pvt, 0, reg0, base0))
1023 edac_dbg(0, " DCSB0[%d]=0x%08x reg: F2x%x\n",
1024 cs, *base0, reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +02001025
Yazen Ghannamd971e282019-08-21 23:59:55 +00001026 if (pvt->fam == 0xf)
1027 continue;
Borislav Petkovb2b0c602010-10-08 18:32:29 +02001028
Yazen Ghannamd971e282019-08-21 23:59:55 +00001029 if (!amd64_read_dct_pci_cfg(pvt, 1, reg0, base1))
1030 edac_dbg(0, " DCSB1[%d]=0x%08x reg: F2x%x\n",
1031 cs, *base1, (pvt->fam == 0x10) ? reg1
1032 : reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +02001033 }
1034
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001035 for_each_chip_select_mask(cs, 0, pvt) {
Yazen Ghannamd971e282019-08-21 23:59:55 +00001036 int reg0 = DCSM0 + (cs * 4);
1037 int reg1 = DCSM1 + (cs * 4);
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001038 u32 *mask0 = &pvt->csels[0].csmasks[cs];
1039 u32 *mask1 = &pvt->csels[1].csmasks[cs];
Borislav Petkovb2b0c602010-10-08 18:32:29 +02001040
Yazen Ghannamd971e282019-08-21 23:59:55 +00001041 if (!amd64_read_dct_pci_cfg(pvt, 0, reg0, mask0))
1042 edac_dbg(0, " DCSM0[%d]=0x%08x reg: F2x%x\n",
1043 cs, *mask0, reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +02001044
Yazen Ghannamd971e282019-08-21 23:59:55 +00001045 if (pvt->fam == 0xf)
1046 continue;
Borislav Petkovb2b0c602010-10-08 18:32:29 +02001047
Yazen Ghannamd971e282019-08-21 23:59:55 +00001048 if (!amd64_read_dct_pci_cfg(pvt, 1, reg0, mask1))
1049 edac_dbg(0, " DCSM1[%d]=0x%08x reg: F2x%x\n",
1050 cs, *mask1, (pvt->fam == 0x10) ? reg1
1051 : reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +02001052 }
1053}
1054
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001055static void determine_memory_type(struct amd64_pvt *pvt)
Doug Thompson94be4bf2009-04-27 16:12:00 +02001056{
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001057 u32 dram_ctrl, dcsm;
Doug Thompson94be4bf2009-04-27 16:12:00 +02001058
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001059 switch (pvt->fam) {
1060 case 0xf:
1061 if (pvt->ext_model >= K8_REV_F)
1062 goto ddr3;
1063
1064 pvt->dram_type = (pvt->dclr0 & BIT(18)) ? MEM_DDR : MEM_RDDR;
1065 return;
1066
1067 case 0x10:
Borislav Petkov6b4c0bd2009-11-12 15:37:57 +01001068 if (pvt->dchr0 & DDR3_MODE)
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001069 goto ddr3;
1070
1071 pvt->dram_type = (pvt->dclr0 & BIT(16)) ? MEM_DDR2 : MEM_RDDR2;
1072 return;
1073
1074 case 0x15:
1075 if (pvt->model < 0x60)
1076 goto ddr3;
1077
1078 /*
1079 * Model 0x60h needs special handling:
1080 *
1081 * We use a Chip Select value of '0' to obtain dcsm.
1082 * Theoretically, it is possible to populate LRDIMMs of different
1083 * 'Rank' value on a DCT. But this is not the common case. So,
1084 * it's reasonable to assume all DIMMs are going to be of same
1085 * 'type' until proven otherwise.
1086 */
1087 amd64_read_dct_pci_cfg(pvt, 0, DRAM_CONTROL, &dram_ctrl);
1088 dcsm = pvt->csels[0].csmasks[0];
1089
1090 if (((dram_ctrl >> 8) & 0x7) == 0x2)
1091 pvt->dram_type = MEM_DDR4;
1092 else if (pvt->dclr0 & BIT(16))
1093 pvt->dram_type = MEM_DDR3;
1094 else if (dcsm & 0x3)
1095 pvt->dram_type = MEM_LRDDR3;
Borislav Petkov6b4c0bd2009-11-12 15:37:57 +01001096 else
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001097 pvt->dram_type = MEM_RDDR3;
1098
1099 return;
1100
1101 case 0x16:
1102 goto ddr3;
1103
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05001104 case 0x17:
Pu Wenc4a3e942018-09-27 16:31:28 +02001105 case 0x18:
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05001106 if ((pvt->umc[0].dimm_cfg | pvt->umc[1].dimm_cfg) & BIT(5))
1107 pvt->dram_type = MEM_LRDDR4;
1108 else if ((pvt->umc[0].dimm_cfg | pvt->umc[1].dimm_cfg) & BIT(4))
1109 pvt->dram_type = MEM_RDDR4;
1110 else
1111 pvt->dram_type = MEM_DDR4;
1112 return;
1113
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001114 default:
1115 WARN(1, KERN_ERR "%s: Family??? 0x%x\n", __func__, pvt->fam);
1116 pvt->dram_type = MEM_EMPTY;
Doug Thompson94be4bf2009-04-27 16:12:00 +02001117 }
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001118 return;
Doug Thompson94be4bf2009-04-27 16:12:00 +02001119
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001120ddr3:
1121 pvt->dram_type = (pvt->dclr0 & BIT(16)) ? MEM_DDR3 : MEM_RDDR3;
Doug Thompson94be4bf2009-04-27 16:12:00 +02001122}
1123
Borislav Petkovcb328502010-12-22 14:28:24 +01001124/* Get the number of DCT channels the memory controller is using. */
Doug Thompsonddff8762009-04-27 16:14:52 +02001125static int k8_early_channel_count(struct amd64_pvt *pvt)
1126{
Borislav Petkovcb328502010-12-22 14:28:24 +01001127 int flag;
Doug Thompsonddff8762009-04-27 16:14:52 +02001128
Borislav Petkov9f56da02010-10-01 19:44:53 +02001129 if (pvt->ext_model >= K8_REV_F)
Doug Thompsonddff8762009-04-27 16:14:52 +02001130 /* RevF (NPT) and later */
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001131 flag = pvt->dclr0 & WIDTH_128;
Borislav Petkov9f56da02010-10-01 19:44:53 +02001132 else
Doug Thompsonddff8762009-04-27 16:14:52 +02001133 /* RevE and earlier */
1134 flag = pvt->dclr0 & REVE_WIDTH_128;
Doug Thompsonddff8762009-04-27 16:14:52 +02001135
1136 /* not used */
1137 pvt->dclr1 = 0;
1138
1139 return (flag) ? 2 : 1;
1140}
1141
Borislav Petkov70046622011-01-10 14:37:27 +01001142/* On F10h and later ErrAddr is MC4_ADDR[47:1] */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001143static u64 get_error_address(struct amd64_pvt *pvt, struct mce *m)
Doug Thompsonddff8762009-04-27 16:14:52 +02001144{
Borislav Petkov2ec591a2015-02-17 10:58:34 +01001145 u16 mce_nid = amd_get_nb_id(m->extcpu);
1146 struct mem_ctl_info *mci;
Borislav Petkov70046622011-01-10 14:37:27 +01001147 u8 start_bit = 1;
1148 u8 end_bit = 47;
Borislav Petkov2ec591a2015-02-17 10:58:34 +01001149 u64 addr;
1150
1151 mci = edac_mc_find(mce_nid);
1152 if (!mci)
1153 return 0;
1154
1155 pvt = mci->pvt_info;
Borislav Petkov70046622011-01-10 14:37:27 +01001156
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001157 if (pvt->fam == 0xf) {
Borislav Petkov70046622011-01-10 14:37:27 +01001158 start_bit = 3;
1159 end_bit = 39;
1160 }
1161
Chen, Gong10ef6b02013-10-18 14:29:07 -07001162 addr = m->addr & GENMASK_ULL(end_bit, start_bit);
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001163
1164 /*
1165 * Erratum 637 workaround
1166 */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001167 if (pvt->fam == 0x15) {
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001168 u64 cc6_base, tmp_addr;
1169 u32 tmp;
Daniel J Blueman8b84c8d2012-11-27 14:32:10 +08001170 u8 intlv_en;
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001171
Chen, Gong10ef6b02013-10-18 14:29:07 -07001172 if ((addr & GENMASK_ULL(47, 24)) >> 24 != 0x00fdf7)
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001173 return addr;
1174
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001175
1176 amd64_read_pci_cfg(pvt->F1, DRAM_LOCAL_NODE_LIM, &tmp);
1177 intlv_en = tmp >> 21 & 0x7;
1178
1179 /* add [47:27] + 3 trailing bits */
Chen, Gong10ef6b02013-10-18 14:29:07 -07001180 cc6_base = (tmp & GENMASK_ULL(20, 0)) << 3;
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001181
1182 /* reverse and add DramIntlvEn */
1183 cc6_base |= intlv_en ^ 0x7;
1184
1185 /* pin at [47:24] */
1186 cc6_base <<= 24;
1187
1188 if (!intlv_en)
Chen, Gong10ef6b02013-10-18 14:29:07 -07001189 return cc6_base | (addr & GENMASK_ULL(23, 0));
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001190
1191 amd64_read_pci_cfg(pvt->F1, DRAM_LOCAL_NODE_BASE, &tmp);
1192
1193 /* faster log2 */
Chen, Gong10ef6b02013-10-18 14:29:07 -07001194 tmp_addr = (addr & GENMASK_ULL(23, 12)) << __fls(intlv_en + 1);
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001195
1196 /* OR DramIntlvSel into bits [14:12] */
Chen, Gong10ef6b02013-10-18 14:29:07 -07001197 tmp_addr |= (tmp & GENMASK_ULL(23, 21)) >> 9;
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001198
1199 /* add remaining [11:0] bits from original MC4_ADDR */
Chen, Gong10ef6b02013-10-18 14:29:07 -07001200 tmp_addr |= addr & GENMASK_ULL(11, 0);
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001201
1202 return cc6_base | tmp_addr;
1203 }
1204
1205 return addr;
Doug Thompsonddff8762009-04-27 16:14:52 +02001206}
1207
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001208static struct pci_dev *pci_get_related_function(unsigned int vendor,
1209 unsigned int device,
1210 struct pci_dev *related)
1211{
1212 struct pci_dev *dev = NULL;
1213
1214 while ((dev = pci_get_device(vendor, device, dev))) {
1215 if (pci_domain_nr(dev->bus) == pci_domain_nr(related->bus) &&
1216 (dev->bus->number == related->bus->number) &&
1217 (PCI_SLOT(dev->devfn) == PCI_SLOT(related->devfn)))
1218 break;
1219 }
1220
1221 return dev;
1222}
1223
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001224static void read_dram_base_limit_regs(struct amd64_pvt *pvt, unsigned range)
Doug Thompsonddff8762009-04-27 16:14:52 +02001225{
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001226 struct amd_northbridge *nb;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001227 struct pci_dev *f1 = NULL;
1228 unsigned int pci_func;
Borislav Petkov71d2a322011-02-21 19:37:24 +01001229 int off = range << 3;
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001230 u32 llim;
Doug Thompsonddff8762009-04-27 16:14:52 +02001231
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001232 amd64_read_pci_cfg(pvt->F1, DRAM_BASE_LO + off, &pvt->ranges[range].base.lo);
1233 amd64_read_pci_cfg(pvt->F1, DRAM_LIMIT_LO + off, &pvt->ranges[range].lim.lo);
Doug Thompsonddff8762009-04-27 16:14:52 +02001234
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001235 if (pvt->fam == 0xf)
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001236 return;
Doug Thompsonddff8762009-04-27 16:14:52 +02001237
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001238 if (!dram_rw(pvt, range))
1239 return;
Doug Thompsonddff8762009-04-27 16:14:52 +02001240
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001241 amd64_read_pci_cfg(pvt->F1, DRAM_BASE_HI + off, &pvt->ranges[range].base.hi);
1242 amd64_read_pci_cfg(pvt->F1, DRAM_LIMIT_HI + off, &pvt->ranges[range].lim.hi);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001243
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001244 /* F15h: factor in CC6 save area by reading dst node's limit reg */
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001245 if (pvt->fam != 0x15)
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001246 return;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001247
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001248 nb = node_to_amd_nb(dram_dst_node(pvt, range));
1249 if (WARN_ON(!nb))
1250 return;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001251
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001252 if (pvt->model == 0x60)
1253 pci_func = PCI_DEVICE_ID_AMD_15H_M60H_NB_F1;
1254 else if (pvt->model == 0x30)
1255 pci_func = PCI_DEVICE_ID_AMD_15H_M30H_NB_F1;
1256 else
1257 pci_func = PCI_DEVICE_ID_AMD_15H_NB_F1;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001258
1259 f1 = pci_get_related_function(nb->misc->vendor, pci_func, nb->misc);
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001260 if (WARN_ON(!f1))
1261 return;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001262
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001263 amd64_read_pci_cfg(f1, DRAM_LOCAL_NODE_LIM, &llim);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001264
Chen, Gong10ef6b02013-10-18 14:29:07 -07001265 pvt->ranges[range].lim.lo &= GENMASK_ULL(15, 0);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001266
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001267 /* {[39:27],111b} */
1268 pvt->ranges[range].lim.lo |= ((llim & 0x1fff) << 3 | 0x7) << 16;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001269
Chen, Gong10ef6b02013-10-18 14:29:07 -07001270 pvt->ranges[range].lim.hi &= GENMASK_ULL(7, 0);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001271
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001272 /* [47:40] */
1273 pvt->ranges[range].lim.hi |= llim >> 13;
1274
1275 pci_dev_put(f1);
Doug Thompsonddff8762009-04-27 16:14:52 +02001276}
1277
Borislav Petkovf192c7b2011-01-10 14:24:32 +01001278static void k8_map_sysaddr_to_csrow(struct mem_ctl_info *mci, u64 sys_addr,
Borislav Petkov33ca0642012-08-30 18:01:36 +02001279 struct err_info *err)
Doug Thompsonddff8762009-04-27 16:14:52 +02001280{
Borislav Petkovf192c7b2011-01-10 14:24:32 +01001281 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompsonddff8762009-04-27 16:14:52 +02001282
Borislav Petkov33ca0642012-08-30 18:01:36 +02001283 error_address_to_page_and_offset(sys_addr, err);
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001284
1285 /*
1286 * Find out which node the error address belongs to. This may be
1287 * different from the node that detected the error.
1288 */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001289 err->src_mci = find_mc_by_sys_addr(mci, sys_addr);
1290 if (!err->src_mci) {
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001291 amd64_mc_err(mci, "failed to map error addr 0x%lx to a node\n",
1292 (unsigned long)sys_addr);
Borislav Petkov33ca0642012-08-30 18:01:36 +02001293 err->err_code = ERR_NODE;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001294 return;
1295 }
1296
1297 /* Now map the sys_addr to a CSROW */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001298 err->csrow = sys_addr_to_csrow(err->src_mci, sys_addr);
1299 if (err->csrow < 0) {
1300 err->err_code = ERR_CSROW;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001301 return;
1302 }
1303
Doug Thompsonddff8762009-04-27 16:14:52 +02001304 /* CHIPKILL enabled */
Borislav Petkovf192c7b2011-01-10 14:24:32 +01001305 if (pvt->nbcfg & NBCFG_CHIPKILL) {
Borislav Petkov33ca0642012-08-30 18:01:36 +02001306 err->channel = get_channel_from_ecc_syndrome(mci, err->syndrome);
1307 if (err->channel < 0) {
Doug Thompsonddff8762009-04-27 16:14:52 +02001308 /*
1309 * Syndrome didn't map, so we don't know which of the
1310 * 2 DIMMs is in error. So we need to ID 'both' of them
1311 * as suspect.
1312 */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001313 amd64_mc_warn(err->src_mci, "unknown syndrome 0x%04x - "
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001314 "possible error reporting race\n",
Borislav Petkov33ca0642012-08-30 18:01:36 +02001315 err->syndrome);
1316 err->err_code = ERR_CHANNEL;
Doug Thompsonddff8762009-04-27 16:14:52 +02001317 return;
1318 }
1319 } else {
1320 /*
1321 * non-chipkill ecc mode
1322 *
1323 * The k8 documentation is unclear about how to determine the
1324 * channel number when using non-chipkill memory. This method
1325 * was obtained from email communication with someone at AMD.
1326 * (Wish the email was placed in this comment - norsk)
1327 */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001328 err->channel = ((sys_addr & BIT(3)) != 0);
Doug Thompsonddff8762009-04-27 16:14:52 +02001329 }
Doug Thompsonddff8762009-04-27 16:14:52 +02001330}
1331
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001332static int ddr2_cs_size(unsigned i, bool dct_width)
Doug Thompsonddff8762009-04-27 16:14:52 +02001333{
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001334 unsigned shift = 0;
Doug Thompsonddff8762009-04-27 16:14:52 +02001335
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001336 if (i <= 2)
1337 shift = i;
1338 else if (!(i & 0x1))
1339 shift = i >> 1;
Borislav Petkov1433eb92009-10-21 13:44:36 +02001340 else
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001341 shift = (i + 1) >> 1;
Doug Thompsonddff8762009-04-27 16:14:52 +02001342
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001343 return 128 << (shift + !!dct_width);
1344}
1345
1346static int k8_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001347 unsigned cs_mode, int cs_mask_nr)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001348{
1349 u32 dclr = dct ? pvt->dclr1 : pvt->dclr0;
1350
1351 if (pvt->ext_model >= K8_REV_F) {
1352 WARN_ON(cs_mode > 11);
1353 return ddr2_cs_size(cs_mode, dclr & WIDTH_128);
1354 }
1355 else if (pvt->ext_model >= K8_REV_D) {
Borislav Petkov11b0a312011-11-09 21:28:43 +01001356 unsigned diff;
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001357 WARN_ON(cs_mode > 10);
1358
Borislav Petkov11b0a312011-11-09 21:28:43 +01001359 /*
1360 * the below calculation, besides trying to win an obfuscated C
1361 * contest, maps cs_mode values to DIMM chip select sizes. The
1362 * mappings are:
1363 *
1364 * cs_mode CS size (mb)
1365 * ======= ============
1366 * 0 32
1367 * 1 64
1368 * 2 128
1369 * 3 128
1370 * 4 256
1371 * 5 512
1372 * 6 256
1373 * 7 512
1374 * 8 1024
1375 * 9 1024
1376 * 10 2048
1377 *
1378 * Basically, it calculates a value with which to shift the
1379 * smallest CS size of 32MB.
1380 *
1381 * ddr[23]_cs_size have a similar purpose.
1382 */
1383 diff = cs_mode/3 + (unsigned)(cs_mode > 5);
1384
1385 return 32 << (cs_mode - diff);
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001386 }
1387 else {
1388 WARN_ON(cs_mode > 6);
1389 return 32 << cs_mode;
1390 }
Doug Thompsonddff8762009-04-27 16:14:52 +02001391}
1392
Doug Thompson1afd3c92009-04-27 16:16:50 +02001393/*
1394 * Get the number of DCT channels in use.
1395 *
1396 * Return:
1397 * number of Memory Channels in operation
1398 * Pass back:
1399 * contents of the DCL0_LOW register
1400 */
Borislav Petkov7d20d142011-01-07 17:58:04 +01001401static int f1x_early_channel_count(struct amd64_pvt *pvt)
Doug Thompson1afd3c92009-04-27 16:16:50 +02001402{
Borislav Petkov6ba5dcd2009-10-13 19:26:55 +02001403 int i, j, channels = 0;
Doug Thompsonddff8762009-04-27 16:14:52 +02001404
Borislav Petkov7d20d142011-01-07 17:58:04 +01001405 /* On F10h, if we are in 128 bit mode, then we are using 2 channels */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001406 if (pvt->fam == 0x10 && (pvt->dclr0 & WIDTH_128))
Borislav Petkov7d20d142011-01-07 17:58:04 +01001407 return 2;
Doug Thompson1afd3c92009-04-27 16:16:50 +02001408
1409 /*
Borislav Petkovd16149e2009-10-16 19:55:49 +02001410 * Need to check if in unganged mode: In such, there are 2 channels,
1411 * but they are not in 128 bit mode and thus the above 'dclr0' status
1412 * bit will be OFF.
Doug Thompson1afd3c92009-04-27 16:16:50 +02001413 *
1414 * Need to check DCT0[0] and DCT1[0] to see if only one of them has
1415 * their CSEnable bit on. If so, then SINGLE DIMM case.
1416 */
Joe Perches956b9ba2012-04-29 17:08:39 -03001417 edac_dbg(0, "Data width is not 128 bits - need more decoding\n");
Doug Thompson1afd3c92009-04-27 16:16:50 +02001418
1419 /*
1420 * Check DRAM Bank Address Mapping values for each DIMM to see if there
1421 * is more than just one DIMM present in unganged mode. Need to check
1422 * both controllers since DIMMs can be placed in either one.
1423 */
Borislav Petkov525a1b22010-12-21 15:53:27 +01001424 for (i = 0; i < 2; i++) {
1425 u32 dbam = (i ? pvt->dbam1 : pvt->dbam0);
Doug Thompson1afd3c92009-04-27 16:16:50 +02001426
Wan Wei57a30852009-08-07 17:04:49 +02001427 for (j = 0; j < 4; j++) {
1428 if (DBAM_DIMM(j, dbam) > 0) {
1429 channels++;
1430 break;
1431 }
1432 }
Doug Thompson1afd3c92009-04-27 16:16:50 +02001433 }
1434
Borislav Petkovd16149e2009-10-16 19:55:49 +02001435 if (channels > 2)
1436 channels = 2;
1437
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02001438 amd64_info("MCT channel count: %d\n", channels);
Doug Thompson1afd3c92009-04-27 16:16:50 +02001439
1440 return channels;
Doug Thompson1afd3c92009-04-27 16:16:50 +02001441}
1442
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001443static int f17_early_channel_count(struct amd64_pvt *pvt)
1444{
1445 int i, channels = 0;
1446
1447 /* SDP Control bit 31 (SdpInit) is clear for unused UMC channels */
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00001448 for_each_umc(i)
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001449 channels += !!(pvt->umc[i].sdp_ctrl & UMC_SDP_INIT);
1450
1451 amd64_info("MCT channel count: %d\n", channels);
1452
1453 return channels;
1454}
1455
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001456static int ddr3_cs_size(unsigned i, bool dct_width)
Doug Thompson1afd3c92009-04-27 16:16:50 +02001457{
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001458 unsigned shift = 0;
1459 int cs_size = 0;
1460
1461 if (i == 0 || i == 3 || i == 4)
1462 cs_size = -1;
1463 else if (i <= 2)
1464 shift = i;
1465 else if (i == 12)
1466 shift = 7;
1467 else if (!(i & 0x1))
1468 shift = i >> 1;
1469 else
1470 shift = (i + 1) >> 1;
1471
1472 if (cs_size != -1)
1473 cs_size = (128 * (1 << !!dct_width)) << shift;
1474
1475 return cs_size;
1476}
1477
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001478static int ddr3_lrdimm_cs_size(unsigned i, unsigned rank_multiply)
1479{
1480 unsigned shift = 0;
1481 int cs_size = 0;
1482
1483 if (i < 4 || i == 6)
1484 cs_size = -1;
1485 else if (i == 12)
1486 shift = 7;
1487 else if (!(i & 0x1))
1488 shift = i >> 1;
1489 else
1490 shift = (i + 1) >> 1;
1491
1492 if (cs_size != -1)
1493 cs_size = rank_multiply * (128 << shift);
1494
1495 return cs_size;
1496}
1497
1498static int ddr4_cs_size(unsigned i)
1499{
1500 int cs_size = 0;
1501
1502 if (i == 0)
1503 cs_size = -1;
1504 else if (i == 1)
1505 cs_size = 1024;
1506 else
1507 /* Min cs_size = 1G */
1508 cs_size = 1024 * (1 << (i >> 1));
1509
1510 return cs_size;
1511}
1512
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001513static int f10_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001514 unsigned cs_mode, int cs_mask_nr)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001515{
1516 u32 dclr = dct ? pvt->dclr1 : pvt->dclr0;
1517
1518 WARN_ON(cs_mode > 11);
Borislav Petkov1433eb92009-10-21 13:44:36 +02001519
1520 if (pvt->dchr0 & DDR3_MODE || pvt->dchr1 & DDR3_MODE)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001521 return ddr3_cs_size(cs_mode, dclr & WIDTH_128);
Borislav Petkov1433eb92009-10-21 13:44:36 +02001522 else
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001523 return ddr2_cs_size(cs_mode, dclr & WIDTH_128);
1524}
Borislav Petkov1433eb92009-10-21 13:44:36 +02001525
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001526/*
1527 * F15h supports only 64bit DCT interfaces
1528 */
1529static int f15_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001530 unsigned cs_mode, int cs_mask_nr)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001531{
1532 WARN_ON(cs_mode > 12);
1533
1534 return ddr3_cs_size(cs_mode, false);
Doug Thompson1afd3c92009-04-27 16:16:50 +02001535}
1536
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001537/* F15h M60h supports DDR4 mapping as well.. */
1538static int f15_m60h_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
1539 unsigned cs_mode, int cs_mask_nr)
1540{
1541 int cs_size;
1542 u32 dcsm = pvt->csels[dct].csmasks[cs_mask_nr];
1543
1544 WARN_ON(cs_mode > 12);
1545
1546 if (pvt->dram_type == MEM_DDR4) {
1547 if (cs_mode > 9)
1548 return -1;
1549
1550 cs_size = ddr4_cs_size(cs_mode);
1551 } else if (pvt->dram_type == MEM_LRDDR3) {
1552 unsigned rank_multiply = dcsm & 0xf;
1553
1554 if (rank_multiply == 3)
1555 rank_multiply = 4;
1556 cs_size = ddr3_lrdimm_cs_size(cs_mode, rank_multiply);
1557 } else {
1558 /* Minimum cs size is 512mb for F15hM60h*/
1559 if (cs_mode == 0x1)
1560 return -1;
1561
1562 cs_size = ddr3_cs_size(cs_mode, false);
1563 }
1564
1565 return cs_size;
1566}
1567
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001568/*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001569 * F16h and F15h model 30h have only limited cs_modes.
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001570 */
1571static int f16_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001572 unsigned cs_mode, int cs_mask_nr)
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001573{
1574 WARN_ON(cs_mode > 12);
1575
1576 if (cs_mode == 6 || cs_mode == 8 ||
1577 cs_mode == 9 || cs_mode == 12)
1578 return -1;
1579 else
1580 return ddr3_cs_size(cs_mode, false);
1581}
1582
Yazen Ghanname53a3b22019-08-21 23:59:59 +00001583static int f17_addr_mask_to_cs_size(struct amd64_pvt *pvt, u8 umc,
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001584 unsigned int cs_mode, int csrow_nr)
1585{
Yazen Ghanname53a3b22019-08-21 23:59:59 +00001586 u32 addr_mask_orig, addr_mask_deinterleaved;
1587 u32 msb, weight, num_zero_bits;
1588 int dimm, size = 0;
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001589
Yazen Ghanname53a3b22019-08-21 23:59:59 +00001590 /* No Chip Selects are enabled. */
1591 if (!cs_mode)
1592 return size;
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001593
Yazen Ghanname53a3b22019-08-21 23:59:59 +00001594 /* Requested size of an even CS but none are enabled. */
1595 if (!(cs_mode & CS_EVEN) && !(csrow_nr & 1))
1596 return size;
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001597
Yazen Ghanname53a3b22019-08-21 23:59:59 +00001598 /* Requested size of an odd CS but none are enabled. */
1599 if (!(cs_mode & CS_ODD) && (csrow_nr & 1))
1600 return size;
1601
1602 /*
1603 * There is one mask per DIMM, and two Chip Selects per DIMM.
1604 * CS0 and CS1 -> DIMM0
1605 * CS2 and CS3 -> DIMM1
1606 */
1607 dimm = csrow_nr >> 1;
1608
Yazen Ghannam81f50902019-08-22 00:00:02 +00001609 /* Asymmetric dual-rank DIMM support. */
1610 if ((csrow_nr & 1) && (cs_mode & CS_ODD_SECONDARY))
1611 addr_mask_orig = pvt->csels[umc].csmasks_sec[dimm];
1612 else
1613 addr_mask_orig = pvt->csels[umc].csmasks[dimm];
Yazen Ghanname53a3b22019-08-21 23:59:59 +00001614
1615 /*
1616 * The number of zero bits in the mask is equal to the number of bits
1617 * in a full mask minus the number of bits in the current mask.
1618 *
1619 * The MSB is the number of bits in the full mask because BIT[0] is
1620 * always 0.
1621 */
1622 msb = fls(addr_mask_orig) - 1;
1623 weight = hweight_long(addr_mask_orig);
1624 num_zero_bits = msb - weight;
1625
1626 /* Take the number of zero bits off from the top of the mask. */
1627 addr_mask_deinterleaved = GENMASK_ULL(msb - num_zero_bits, 1);
1628
1629 edac_dbg(1, "CS%d DIMM%d AddrMasks:\n", csrow_nr, dimm);
1630 edac_dbg(1, " Original AddrMask: 0x%x\n", addr_mask_orig);
1631 edac_dbg(1, " Deinterleaved AddrMask: 0x%x\n", addr_mask_deinterleaved);
1632
1633 /* Register [31:1] = Address [39:9]. Size is in kBs here. */
1634 size = (addr_mask_deinterleaved >> 2) + 1;
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001635
1636 /* Return size in MBs. */
1637 return size >> 10;
1638}
1639
Borislav Petkov5a5d2372011-01-17 17:52:57 +01001640static void read_dram_ctl_register(struct amd64_pvt *pvt)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001641{
Doug Thompson6163b5d2009-04-27 16:20:17 +02001642
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001643 if (pvt->fam == 0xf)
Borislav Petkov5a5d2372011-01-17 17:52:57 +01001644 return;
1645
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001646 if (!amd64_read_pci_cfg(pvt->F2, DCT_SEL_LO, &pvt->dct_sel_lo)) {
Joe Perches956b9ba2012-04-29 17:08:39 -03001647 edac_dbg(0, "F2x110 (DCTSelLow): 0x%08x, High range addrs at: 0x%x\n",
1648 pvt->dct_sel_lo, dct_sel_baseaddr(pvt));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001649
Joe Perches956b9ba2012-04-29 17:08:39 -03001650 edac_dbg(0, " DCTs operate in %s mode\n",
1651 (dct_ganging_enabled(pvt) ? "ganged" : "unganged"));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001652
Borislav Petkov72381bd2009-10-09 19:14:43 +02001653 if (!dct_ganging_enabled(pvt))
Joe Perches956b9ba2012-04-29 17:08:39 -03001654 edac_dbg(0, " Address range split per DCT: %s\n",
1655 (dct_high_range_enabled(pvt) ? "yes" : "no"));
Borislav Petkov72381bd2009-10-09 19:14:43 +02001656
Joe Perches956b9ba2012-04-29 17:08:39 -03001657 edac_dbg(0, " data interleave for ECC: %s, DRAM cleared since last warm reset: %s\n",
1658 (dct_data_intlv_enabled(pvt) ? "enabled" : "disabled"),
1659 (dct_memory_cleared(pvt) ? "yes" : "no"));
Borislav Petkov72381bd2009-10-09 19:14:43 +02001660
Joe Perches956b9ba2012-04-29 17:08:39 -03001661 edac_dbg(0, " channel interleave: %s, "
1662 "interleave bits selector: 0x%x\n",
1663 (dct_interleave_enabled(pvt) ? "enabled" : "disabled"),
1664 dct_sel_interleave_addr(pvt));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001665 }
1666
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001667 amd64_read_pci_cfg(pvt->F2, DCT_SEL_HI, &pvt->dct_sel_hi);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001668}
1669
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001670/*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001671 * Determine channel (DCT) based on the interleaving mode (see F15h M30h BKDG,
1672 * 2.10.12 Memory Interleaving Modes).
1673 */
1674static u8 f15_m30h_determine_channel(struct amd64_pvt *pvt, u64 sys_addr,
1675 u8 intlv_en, int num_dcts_intlv,
1676 u32 dct_sel)
1677{
1678 u8 channel = 0;
1679 u8 select;
1680
1681 if (!(intlv_en))
1682 return (u8)(dct_sel);
1683
1684 if (num_dcts_intlv == 2) {
1685 select = (sys_addr >> 8) & 0x3;
1686 channel = select ? 0x3 : 0;
Aravind Gopalakrishnan9d0e8d82014-01-21 15:03:36 -06001687 } else if (num_dcts_intlv == 4) {
1688 u8 intlv_addr = dct_sel_interleave_addr(pvt);
1689 switch (intlv_addr) {
1690 case 0x4:
1691 channel = (sys_addr >> 8) & 0x3;
1692 break;
1693 case 0x5:
1694 channel = (sys_addr >> 9) & 0x3;
1695 break;
1696 }
1697 }
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001698 return channel;
1699}
1700
1701/*
Borislav Petkov229a7a12010-12-09 18:57:54 +01001702 * Determine channel (DCT) based on the interleaving mode: F10h BKDG, 2.8.9 Memory
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001703 * Interleaving Modes.
1704 */
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001705static u8 f1x_determine_channel(struct amd64_pvt *pvt, u64 sys_addr,
Borislav Petkov229a7a12010-12-09 18:57:54 +01001706 bool hi_range_sel, u8 intlv_en)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001707{
Borislav Petkov151fa712011-02-21 19:33:10 +01001708 u8 dct_sel_high = (pvt->dct_sel_lo >> 1) & 1;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001709
1710 if (dct_ganging_enabled(pvt))
Borislav Petkov229a7a12010-12-09 18:57:54 +01001711 return 0;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001712
Borislav Petkov229a7a12010-12-09 18:57:54 +01001713 if (hi_range_sel)
1714 return dct_sel_high;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001715
Borislav Petkov229a7a12010-12-09 18:57:54 +01001716 /*
1717 * see F2x110[DctSelIntLvAddr] - channel interleave mode
1718 */
1719 if (dct_interleave_enabled(pvt)) {
1720 u8 intlv_addr = dct_sel_interleave_addr(pvt);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001721
Borislav Petkov229a7a12010-12-09 18:57:54 +01001722 /* return DCT select function: 0=DCT0, 1=DCT1 */
1723 if (!intlv_addr)
1724 return sys_addr >> 6 & 1;
1725
1726 if (intlv_addr & 0x2) {
1727 u8 shift = intlv_addr & 0x1 ? 9 : 6;
Yazen Ghannamdc0a50a82016-08-03 10:59:15 -04001728 u32 temp = hweight_long((u32) ((sys_addr >> 16) & 0x1F)) & 1;
Borislav Petkov229a7a12010-12-09 18:57:54 +01001729
1730 return ((sys_addr >> shift) & 1) ^ temp;
1731 }
1732
Yazen Ghannamdc0a50a82016-08-03 10:59:15 -04001733 if (intlv_addr & 0x4) {
1734 u8 shift = intlv_addr & 0x1 ? 9 : 8;
1735
1736 return (sys_addr >> shift) & 1;
1737 }
1738
Borislav Petkov229a7a12010-12-09 18:57:54 +01001739 return (sys_addr >> (12 + hweight8(intlv_en))) & 1;
1740 }
1741
1742 if (dct_high_range_enabled(pvt))
1743 return ~dct_sel_high & 1;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001744
1745 return 0;
1746}
1747
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001748/* Convert the sys_addr to the normalized DCT address */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08001749static u64 f1x_get_norm_dct_addr(struct amd64_pvt *pvt, u8 range,
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001750 u64 sys_addr, bool hi_rng,
1751 u32 dct_sel_base_addr)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001752{
1753 u64 chan_off;
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001754 u64 dram_base = get_dram_base(pvt, range);
1755 u64 hole_off = f10_dhar_offset(pvt);
Dan Carpenter6f3508f2016-01-20 12:54:51 +03001756 u64 dct_sel_base_off = (u64)(pvt->dct_sel_hi & 0xFFFFFC00) << 16;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001757
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001758 if (hi_rng) {
1759 /*
1760 * if
1761 * base address of high range is below 4Gb
1762 * (bits [47:27] at [31:11])
1763 * DRAM address space on this DCT is hoisted above 4Gb &&
1764 * sys_addr > 4Gb
1765 *
1766 * remove hole offset from sys_addr
1767 * else
1768 * remove high range offset from sys_addr
1769 */
1770 if ((!(dct_sel_base_addr >> 16) ||
1771 dct_sel_base_addr < dhar_base(pvt)) &&
Borislav Petkov972ea172011-02-21 19:43:02 +01001772 dhar_valid(pvt) &&
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001773 (sys_addr >= BIT_64(32)))
Borislav Petkovbc21fa52010-11-11 17:29:13 +01001774 chan_off = hole_off;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001775 else
1776 chan_off = dct_sel_base_off;
1777 } else {
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001778 /*
1779 * if
1780 * we have a valid hole &&
1781 * sys_addr > 4Gb
1782 *
1783 * remove hole
1784 * else
1785 * remove dram base to normalize to DCT address
1786 */
Borislav Petkov972ea172011-02-21 19:43:02 +01001787 if (dhar_valid(pvt) && (sys_addr >= BIT_64(32)))
Borislav Petkovbc21fa52010-11-11 17:29:13 +01001788 chan_off = hole_off;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001789 else
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001790 chan_off = dram_base;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001791 }
1792
Chen, Gong10ef6b02013-10-18 14:29:07 -07001793 return (sys_addr & GENMASK_ULL(47,6)) - (chan_off & GENMASK_ULL(47,23));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001794}
1795
Doug Thompson6163b5d2009-04-27 16:20:17 +02001796/*
1797 * checks if the csrow passed in is marked as SPARED, if so returns the new
1798 * spare row
1799 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001800static int f10_process_possible_spare(struct amd64_pvt *pvt, u8 dct, int csrow)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001801{
Borislav Petkov614ec9d2011-01-13 18:02:22 +01001802 int tmp_cs;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001803
Borislav Petkov614ec9d2011-01-13 18:02:22 +01001804 if (online_spare_swap_done(pvt, dct) &&
1805 csrow == online_spare_bad_dramcs(pvt, dct)) {
1806
1807 for_each_chip_select(tmp_cs, dct, pvt) {
1808 if (chip_select_base(tmp_cs, dct, pvt) & 0x2) {
1809 csrow = tmp_cs;
1810 break;
1811 }
1812 }
Doug Thompson6163b5d2009-04-27 16:20:17 +02001813 }
1814 return csrow;
1815}
1816
1817/*
1818 * Iterate over the DRAM DCT "base" and "mask" registers looking for a
1819 * SystemAddr match on the specified 'ChannelSelect' and 'NodeID'
1820 *
1821 * Return:
1822 * -EINVAL: NOT FOUND
1823 * 0..csrow = Chip-Select Row
1824 */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08001825static int f1x_lookup_addr_in_dct(u64 in_addr, u8 nid, u8 dct)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001826{
1827 struct mem_ctl_info *mci;
1828 struct amd64_pvt *pvt;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001829 u64 cs_base, cs_mask;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001830 int cs_found = -EINVAL;
1831 int csrow;
1832
Borislav Petkov2ec591a2015-02-17 10:58:34 +01001833 mci = edac_mc_find(nid);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001834 if (!mci)
1835 return cs_found;
1836
1837 pvt = mci->pvt_info;
1838
Joe Perches956b9ba2012-04-29 17:08:39 -03001839 edac_dbg(1, "input addr: 0x%llx, DCT: %d\n", in_addr, dct);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001840
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001841 for_each_chip_select(csrow, dct, pvt) {
1842 if (!csrow_enabled(csrow, dct, pvt))
Doug Thompson6163b5d2009-04-27 16:20:17 +02001843 continue;
1844
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001845 get_cs_base_and_mask(pvt, csrow, dct, &cs_base, &cs_mask);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001846
Joe Perches956b9ba2012-04-29 17:08:39 -03001847 edac_dbg(1, " CSROW=%d CSBase=0x%llx CSMask=0x%llx\n",
1848 csrow, cs_base, cs_mask);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001849
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001850 cs_mask = ~cs_mask;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001851
Joe Perches956b9ba2012-04-29 17:08:39 -03001852 edac_dbg(1, " (InputAddr & ~CSMask)=0x%llx (CSBase & ~CSMask)=0x%llx\n",
1853 (in_addr & cs_mask), (cs_base & cs_mask));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001854
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001855 if ((in_addr & cs_mask) == (cs_base & cs_mask)) {
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001856 if (pvt->fam == 0x15 && pvt->model >= 0x30) {
1857 cs_found = csrow;
1858 break;
1859 }
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001860 cs_found = f10_process_possible_spare(pvt, dct, csrow);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001861
Joe Perches956b9ba2012-04-29 17:08:39 -03001862 edac_dbg(1, " MATCH csrow=%d\n", cs_found);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001863 break;
1864 }
1865 }
1866 return cs_found;
1867}
1868
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001869/*
1870 * See F2x10C. Non-interleaved graphics framebuffer memory under the 16G is
1871 * swapped with a region located at the bottom of memory so that the GPU can use
1872 * the interleaved region and thus two channels.
1873 */
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001874static u64 f1x_swap_interleaved_region(struct amd64_pvt *pvt, u64 sys_addr)
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001875{
1876 u32 swap_reg, swap_base, swap_limit, rgn_size, tmp_addr;
1877
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001878 if (pvt->fam == 0x10) {
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001879 /* only revC3 and revE have that feature */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001880 if (pvt->model < 4 || (pvt->model < 0xa && pvt->stepping < 3))
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001881 return sys_addr;
1882 }
1883
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001884 amd64_read_pci_cfg(pvt->F2, SWAP_INTLV_REG, &swap_reg);
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001885
1886 if (!(swap_reg & 0x1))
1887 return sys_addr;
1888
1889 swap_base = (swap_reg >> 3) & 0x7f;
1890 swap_limit = (swap_reg >> 11) & 0x7f;
1891 rgn_size = (swap_reg >> 20) & 0x7f;
1892 tmp_addr = sys_addr >> 27;
1893
1894 if (!(sys_addr >> 34) &&
1895 (((tmp_addr >= swap_base) &&
1896 (tmp_addr <= swap_limit)) ||
1897 (tmp_addr < rgn_size)))
1898 return sys_addr ^ (u64)swap_base << 27;
1899
1900 return sys_addr;
1901}
1902
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001903/* For a given @dram_range, check if @sys_addr falls within it. */
Borislav Petkove761359a2011-02-21 19:49:01 +01001904static int f1x_match_to_this_node(struct amd64_pvt *pvt, unsigned range,
Borislav Petkov33ca0642012-08-30 18:01:36 +02001905 u64 sys_addr, int *chan_sel)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001906{
Borislav Petkov229a7a12010-12-09 18:57:54 +01001907 int cs_found = -EINVAL;
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001908 u64 chan_addr;
Borislav Petkov5d4b58e2011-01-13 16:01:13 +01001909 u32 dct_sel_base;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001910 u8 channel;
Borislav Petkov229a7a12010-12-09 18:57:54 +01001911 bool high_range = false;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001912
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001913 u8 node_id = dram_dst_node(pvt, range);
Borislav Petkov229a7a12010-12-09 18:57:54 +01001914 u8 intlv_en = dram_intlv_en(pvt, range);
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001915 u32 intlv_sel = dram_intlv_sel(pvt, range);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001916
Joe Perches956b9ba2012-04-29 17:08:39 -03001917 edac_dbg(1, "(range %d) SystemAddr= 0x%llx Limit=0x%llx\n",
1918 range, sys_addr, get_dram_limit(pvt, range));
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001919
Borislav Petkov355fba62011-01-17 13:03:26 +01001920 if (dhar_valid(pvt) &&
1921 dhar_base(pvt) <= sys_addr &&
1922 sys_addr < BIT_64(32)) {
1923 amd64_warn("Huh? Address is in the MMIO hole: 0x%016llx\n",
1924 sys_addr);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001925 return -EINVAL;
Borislav Petkov355fba62011-01-17 13:03:26 +01001926 }
1927
Borislav Petkovf030ddf2011-04-08 15:05:21 +02001928 if (intlv_en && (intlv_sel != ((sys_addr >> 12) & intlv_en)))
Borislav Petkov355fba62011-01-17 13:03:26 +01001929 return -EINVAL;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001930
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001931 sys_addr = f1x_swap_interleaved_region(pvt, sys_addr);
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001932
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001933 dct_sel_base = dct_sel_baseaddr(pvt);
1934
1935 /*
1936 * check whether addresses >= DctSelBaseAddr[47:27] are to be used to
1937 * select between DCT0 and DCT1.
1938 */
1939 if (dct_high_range_enabled(pvt) &&
1940 !dct_ganging_enabled(pvt) &&
1941 ((sys_addr >> 27) >= (dct_sel_base >> 11)))
Borislav Petkov229a7a12010-12-09 18:57:54 +01001942 high_range = true;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001943
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001944 channel = f1x_determine_channel(pvt, sys_addr, high_range, intlv_en);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001945
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001946 chan_addr = f1x_get_norm_dct_addr(pvt, range, sys_addr,
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001947 high_range, dct_sel_base);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001948
Borislav Petkove2f79db2011-01-13 14:57:34 +01001949 /* Remove node interleaving, see F1x120 */
1950 if (intlv_en)
1951 chan_addr = ((chan_addr >> (12 + hweight8(intlv_en))) << 12) |
1952 (chan_addr & 0xfff);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001953
Borislav Petkov5d4b58e2011-01-13 16:01:13 +01001954 /* remove channel interleave */
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001955 if (dct_interleave_enabled(pvt) &&
1956 !dct_high_range_enabled(pvt) &&
1957 !dct_ganging_enabled(pvt)) {
Borislav Petkov5d4b58e2011-01-13 16:01:13 +01001958
1959 if (dct_sel_interleave_addr(pvt) != 1) {
1960 if (dct_sel_interleave_addr(pvt) == 0x3)
1961 /* hash 9 */
1962 chan_addr = ((chan_addr >> 10) << 9) |
1963 (chan_addr & 0x1ff);
1964 else
1965 /* A[6] or hash 6 */
1966 chan_addr = ((chan_addr >> 7) << 6) |
1967 (chan_addr & 0x3f);
1968 } else
1969 /* A[12] */
1970 chan_addr = ((chan_addr >> 13) << 12) |
1971 (chan_addr & 0xfff);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001972 }
1973
Joe Perches956b9ba2012-04-29 17:08:39 -03001974 edac_dbg(1, " Normalized DCT addr: 0x%llx\n", chan_addr);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001975
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001976 cs_found = f1x_lookup_addr_in_dct(chan_addr, node_id, channel);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001977
Borislav Petkov33ca0642012-08-30 18:01:36 +02001978 if (cs_found >= 0)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001979 *chan_sel = channel;
Borislav Petkov33ca0642012-08-30 18:01:36 +02001980
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001981 return cs_found;
1982}
1983
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001984static int f15_m30h_match_to_this_node(struct amd64_pvt *pvt, unsigned range,
1985 u64 sys_addr, int *chan_sel)
1986{
1987 int cs_found = -EINVAL;
1988 int num_dcts_intlv = 0;
1989 u64 chan_addr, chan_offset;
1990 u64 dct_base, dct_limit;
1991 u32 dct_cont_base_reg, dct_cont_limit_reg, tmp;
1992 u8 channel, alias_channel, leg_mmio_hole, dct_sel, dct_offset_en;
1993
1994 u64 dhar_offset = f10_dhar_offset(pvt);
1995 u8 intlv_addr = dct_sel_interleave_addr(pvt);
1996 u8 node_id = dram_dst_node(pvt, range);
1997 u8 intlv_en = dram_intlv_en(pvt, range);
1998
1999 amd64_read_pci_cfg(pvt->F1, DRAM_CONT_BASE, &dct_cont_base_reg);
2000 amd64_read_pci_cfg(pvt->F1, DRAM_CONT_LIMIT, &dct_cont_limit_reg);
2001
2002 dct_offset_en = (u8) ((dct_cont_base_reg >> 3) & BIT(0));
2003 dct_sel = (u8) ((dct_cont_base_reg >> 4) & 0x7);
2004
2005 edac_dbg(1, "(range %d) SystemAddr= 0x%llx Limit=0x%llx\n",
2006 range, sys_addr, get_dram_limit(pvt, range));
2007
2008 if (!(get_dram_base(pvt, range) <= sys_addr) &&
2009 !(get_dram_limit(pvt, range) >= sys_addr))
2010 return -EINVAL;
2011
2012 if (dhar_valid(pvt) &&
2013 dhar_base(pvt) <= sys_addr &&
2014 sys_addr < BIT_64(32)) {
2015 amd64_warn("Huh? Address is in the MMIO hole: 0x%016llx\n",
2016 sys_addr);
2017 return -EINVAL;
2018 }
2019
2020 /* Verify sys_addr is within DCT Range. */
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05002021 dct_base = (u64) dct_sel_baseaddr(pvt);
2022 dct_limit = (dct_cont_limit_reg >> 11) & 0x1FFF;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002023
2024 if (!(dct_cont_base_reg & BIT(0)) &&
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05002025 !(dct_base <= (sys_addr >> 27) &&
2026 dct_limit >= (sys_addr >> 27)))
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002027 return -EINVAL;
2028
2029 /* Verify number of dct's that participate in channel interleaving. */
2030 num_dcts_intlv = (int) hweight8(intlv_en);
2031
2032 if (!(num_dcts_intlv % 2 == 0) || (num_dcts_intlv > 4))
2033 return -EINVAL;
2034
Yazen Ghannamdc0a50a82016-08-03 10:59:15 -04002035 if (pvt->model >= 0x60)
2036 channel = f1x_determine_channel(pvt, sys_addr, false, intlv_en);
2037 else
2038 channel = f15_m30h_determine_channel(pvt, sys_addr, intlv_en,
2039 num_dcts_intlv, dct_sel);
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002040
2041 /* Verify we stay within the MAX number of channels allowed */
Aravind Gopalakrishnan7f3f5242013-12-04 11:40:11 -06002042 if (channel > 3)
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002043 return -EINVAL;
2044
2045 leg_mmio_hole = (u8) (dct_cont_base_reg >> 1 & BIT(0));
2046
2047 /* Get normalized DCT addr */
2048 if (leg_mmio_hole && (sys_addr >= BIT_64(32)))
2049 chan_offset = dhar_offset;
2050 else
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05002051 chan_offset = dct_base << 27;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002052
2053 chan_addr = sys_addr - chan_offset;
2054
2055 /* remove channel interleave */
2056 if (num_dcts_intlv == 2) {
2057 if (intlv_addr == 0x4)
2058 chan_addr = ((chan_addr >> 9) << 8) |
2059 (chan_addr & 0xff);
2060 else if (intlv_addr == 0x5)
2061 chan_addr = ((chan_addr >> 10) << 9) |
2062 (chan_addr & 0x1ff);
2063 else
2064 return -EINVAL;
2065
2066 } else if (num_dcts_intlv == 4) {
2067 if (intlv_addr == 0x4)
2068 chan_addr = ((chan_addr >> 10) << 8) |
2069 (chan_addr & 0xff);
2070 else if (intlv_addr == 0x5)
2071 chan_addr = ((chan_addr >> 11) << 9) |
2072 (chan_addr & 0x1ff);
2073 else
2074 return -EINVAL;
2075 }
2076
2077 if (dct_offset_en) {
2078 amd64_read_pci_cfg(pvt->F1,
2079 DRAM_CONT_HIGH_OFF + (int) channel * 4,
2080 &tmp);
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05002081 chan_addr += (u64) ((tmp >> 11) & 0xfff) << 27;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002082 }
2083
2084 f15h_select_dct(pvt, channel);
2085
2086 edac_dbg(1, " Normalized DCT addr: 0x%llx\n", chan_addr);
2087
2088 /*
2089 * Find Chip select:
2090 * if channel = 3, then alias it to 1. This is because, in F15 M30h,
2091 * there is support for 4 DCT's, but only 2 are currently functional.
2092 * They are DCT0 and DCT3. But we have read all registers of DCT3 into
2093 * pvt->csels[1]. So we need to use '1' here to get correct info.
2094 * Refer F15 M30h BKDG Section 2.10 and 2.10.3 for clarifications.
2095 */
2096 alias_channel = (channel == 3) ? 1 : channel;
2097
2098 cs_found = f1x_lookup_addr_in_dct(chan_addr, node_id, alias_channel);
2099
2100 if (cs_found >= 0)
2101 *chan_sel = alias_channel;
2102
2103 return cs_found;
2104}
2105
2106static int f1x_translate_sysaddr_to_cs(struct amd64_pvt *pvt,
2107 u64 sys_addr,
2108 int *chan_sel)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002109{
Borislav Petkove761359a2011-02-21 19:49:01 +01002110 int cs_found = -EINVAL;
2111 unsigned range;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002112
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002113 for (range = 0; range < DRAM_RANGES; range++) {
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002114 if (!dram_rw(pvt, range))
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002115 continue;
2116
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002117 if (pvt->fam == 0x15 && pvt->model >= 0x30)
2118 cs_found = f15_m30h_match_to_this_node(pvt, range,
2119 sys_addr,
2120 chan_sel);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002121
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002122 else if ((get_dram_base(pvt, range) <= sys_addr) &&
2123 (get_dram_limit(pvt, range) >= sys_addr)) {
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01002124 cs_found = f1x_match_to_this_node(pvt, range,
Borislav Petkov33ca0642012-08-30 18:01:36 +02002125 sys_addr, chan_sel);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002126 if (cs_found >= 0)
2127 break;
2128 }
2129 }
2130 return cs_found;
2131}
2132
2133/*
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002134 * For reference see "2.8.5 Routing DRAM Requests" in F10 BKDG. This code maps
2135 * a @sys_addr to NodeID, DCT (channel) and chip select (CSROW).
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002136 *
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002137 * The @sys_addr is usually an error address received from the hardware
2138 * (MCX_ADDR).
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002139 */
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01002140static void f1x_map_sysaddr_to_csrow(struct mem_ctl_info *mci, u64 sys_addr,
Borislav Petkov33ca0642012-08-30 18:01:36 +02002141 struct err_info *err)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002142{
2143 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002144
Borislav Petkov33ca0642012-08-30 18:01:36 +02002145 error_address_to_page_and_offset(sys_addr, err);
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03002146
Borislav Petkov33ca0642012-08-30 18:01:36 +02002147 err->csrow = f1x_translate_sysaddr_to_cs(pvt, sys_addr, &err->channel);
2148 if (err->csrow < 0) {
2149 err->err_code = ERR_CSROW;
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002150 return;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002151 }
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002152
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002153 /*
2154 * We need the syndromes for channel detection only when we're
2155 * ganged. Otherwise @chan should already contain the channel at
2156 * this point.
2157 */
Borislav Petkova97fa682010-12-23 14:07:18 +01002158 if (dct_ganging_enabled(pvt))
Borislav Petkov33ca0642012-08-30 18:01:36 +02002159 err->channel = get_channel_from_ecc_syndrome(mci, err->syndrome);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002160}
2161
2162/*
Borislav Petkov8566c4d2009-10-16 13:48:28 +02002163 * debug routine to display the memory sizes of all logical DIMMs and its
Borislav Petkovcb328502010-12-22 14:28:24 +01002164 * CSROWs
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002165 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002166static void debug_display_dimm_sizes(struct amd64_pvt *pvt, u8 ctrl)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002167{
Borislav Petkovbb89f5a2012-09-12 18:06:00 +02002168 int dimm, size0, size1;
Borislav Petkov525a1b22010-12-21 15:53:27 +01002169 u32 *dcsb = ctrl ? pvt->csels[1].csbases : pvt->csels[0].csbases;
2170 u32 dbam = ctrl ? pvt->dbam1 : pvt->dbam0;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002171
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002172 if (pvt->fam == 0xf) {
Borislav Petkov8566c4d2009-10-16 13:48:28 +02002173 /* K8 families < revF not supported yet */
Borislav Petkov1433eb92009-10-21 13:44:36 +02002174 if (pvt->ext_model < K8_REV_F)
Borislav Petkov8566c4d2009-10-16 13:48:28 +02002175 return;
2176 else
2177 WARN_ON(ctrl != 0);
2178 }
2179
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002180 if (pvt->fam == 0x10) {
2181 dbam = (ctrl && !dct_ganging_enabled(pvt)) ? pvt->dbam1
2182 : pvt->dbam0;
2183 dcsb = (ctrl && !dct_ganging_enabled(pvt)) ?
2184 pvt->csels[1].csbases :
2185 pvt->csels[0].csbases;
2186 } else if (ctrl) {
2187 dbam = pvt->dbam0;
2188 dcsb = pvt->csels[1].csbases;
2189 }
Joe Perches956b9ba2012-04-29 17:08:39 -03002190 edac_dbg(1, "F2x%d80 (DRAM Bank Address Mapping): 0x%08x\n",
2191 ctrl, dbam);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002192
Borislav Petkov8566c4d2009-10-16 13:48:28 +02002193 edac_printk(KERN_DEBUG, EDAC_MC, "DCT%d chip selects:\n", ctrl);
2194
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002195 /* Dump memory sizes for DIMM and its CSROWs */
2196 for (dimm = 0; dimm < 4; dimm++) {
2197
2198 size0 = 0;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002199 if (dcsb[dimm*2] & DCSB_CS_ENABLE)
Yazen Ghannam07ed82e2016-11-28 08:50:21 -06002200 /*
2201 * For F15m60h, we need multiplier for LRDIMM cs_size
2202 * calculation. We pass dimm value to the dbam_to_cs
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002203 * mapper so we can find the multiplier from the
2204 * corresponding DCSM.
2205 */
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01002206 size0 = pvt->ops->dbam_to_cs(pvt, ctrl,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002207 DBAM_DIMM(dimm, dbam),
2208 dimm);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002209
2210 size1 = 0;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002211 if (dcsb[dimm*2 + 1] & DCSB_CS_ENABLE)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01002212 size1 = pvt->ops->dbam_to_cs(pvt, ctrl,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002213 DBAM_DIMM(dimm, dbam),
2214 dimm);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002215
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002216 amd64_info(EDAC_MC ": %d: %5dMB %d: %5dMB\n",
Borislav Petkovbb89f5a2012-09-12 18:06:00 +02002217 dimm * 2, size0,
2218 dimm * 2 + 1, size1);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002219 }
2220}
2221
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002222static struct amd64_family_type family_types[] = {
Doug Thompson4d376072009-04-27 16:25:05 +02002223 [K8_CPUS] = {
Borislav Petkov0092b202010-10-01 19:20:05 +02002224 .ctl_name = "K8",
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002225 .f1_id = PCI_DEVICE_ID_AMD_K8_NB_ADDRMAP,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002226 .f2_id = PCI_DEVICE_ID_AMD_K8_NB_MEMCTL,
Doug Thompson4d376072009-04-27 16:25:05 +02002227 .ops = {
Borislav Petkov1433eb92009-10-21 13:44:36 +02002228 .early_channel_count = k8_early_channel_count,
Borislav Petkov1433eb92009-10-21 13:44:36 +02002229 .map_sysaddr_to_csrow = k8_map_sysaddr_to_csrow,
2230 .dbam_to_cs = k8_dbam_to_chip_select,
Doug Thompson4d376072009-04-27 16:25:05 +02002231 }
2232 },
2233 [F10_CPUS] = {
Borislav Petkov0092b202010-10-01 19:20:05 +02002234 .ctl_name = "F10h",
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002235 .f1_id = PCI_DEVICE_ID_AMD_10H_NB_MAP,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002236 .f2_id = PCI_DEVICE_ID_AMD_10H_NB_DRAM,
Doug Thompson4d376072009-04-27 16:25:05 +02002237 .ops = {
Borislav Petkov7d20d142011-01-07 17:58:04 +01002238 .early_channel_count = f1x_early_channel_count,
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01002239 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
Borislav Petkov1433eb92009-10-21 13:44:36 +02002240 .dbam_to_cs = f10_dbam_to_chip_select,
Borislav Petkovb2b0c602010-10-08 18:32:29 +02002241 }
2242 },
2243 [F15_CPUS] = {
2244 .ctl_name = "F15h",
Borislav Petkovdf71a052011-01-19 18:15:10 +01002245 .f1_id = PCI_DEVICE_ID_AMD_15H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002246 .f2_id = PCI_DEVICE_ID_AMD_15H_NB_F2,
Borislav Petkovb2b0c602010-10-08 18:32:29 +02002247 .ops = {
Borislav Petkov7d20d142011-01-07 17:58:04 +01002248 .early_channel_count = f1x_early_channel_count,
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01002249 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01002250 .dbam_to_cs = f15_dbam_to_chip_select,
Doug Thompson4d376072009-04-27 16:25:05 +02002251 }
2252 },
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002253 [F15_M30H_CPUS] = {
2254 .ctl_name = "F15h_M30h",
2255 .f1_id = PCI_DEVICE_ID_AMD_15H_M30H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002256 .f2_id = PCI_DEVICE_ID_AMD_15H_M30H_NB_F2,
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002257 .ops = {
2258 .early_channel_count = f1x_early_channel_count,
2259 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
2260 .dbam_to_cs = f16_dbam_to_chip_select,
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002261 }
2262 },
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002263 [F15_M60H_CPUS] = {
2264 .ctl_name = "F15h_M60h",
2265 .f1_id = PCI_DEVICE_ID_AMD_15H_M60H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002266 .f2_id = PCI_DEVICE_ID_AMD_15H_M60H_NB_F2,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002267 .ops = {
2268 .early_channel_count = f1x_early_channel_count,
2269 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
2270 .dbam_to_cs = f15_m60h_dbam_to_chip_select,
2271 }
2272 },
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05002273 [F16_CPUS] = {
2274 .ctl_name = "F16h",
2275 .f1_id = PCI_DEVICE_ID_AMD_16H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002276 .f2_id = PCI_DEVICE_ID_AMD_16H_NB_F2,
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05002277 .ops = {
2278 .early_channel_count = f1x_early_channel_count,
2279 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
2280 .dbam_to_cs = f16_dbam_to_chip_select,
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05002281 }
2282 },
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06002283 [F16_M30H_CPUS] = {
2284 .ctl_name = "F16h_M30h",
2285 .f1_id = PCI_DEVICE_ID_AMD_16H_M30H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002286 .f2_id = PCI_DEVICE_ID_AMD_16H_M30H_NB_F2,
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06002287 .ops = {
2288 .early_channel_count = f1x_early_channel_count,
2289 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
2290 .dbam_to_cs = f16_dbam_to_chip_select,
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06002291 }
2292 },
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05002293 [F17_CPUS] = {
2294 .ctl_name = "F17h",
2295 .f0_id = PCI_DEVICE_ID_AMD_17H_DF_F0,
2296 .f6_id = PCI_DEVICE_ID_AMD_17H_DF_F6,
2297 .ops = {
2298 .early_channel_count = f17_early_channel_count,
Yazen Ghanname53a3b22019-08-21 23:59:59 +00002299 .dbam_to_cs = f17_addr_mask_to_cs_size,
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05002300 }
2301 },
Michael Jin8960de42018-08-16 15:28:40 -04002302 [F17_M10H_CPUS] = {
2303 .ctl_name = "F17h_M10h",
2304 .f0_id = PCI_DEVICE_ID_AMD_17H_M10H_DF_F0,
2305 .f6_id = PCI_DEVICE_ID_AMD_17H_M10H_DF_F6,
2306 .ops = {
2307 .early_channel_count = f17_early_channel_count,
Yazen Ghanname53a3b22019-08-21 23:59:59 +00002308 .dbam_to_cs = f17_addr_mask_to_cs_size,
Michael Jin8960de42018-08-16 15:28:40 -04002309 }
2310 },
Yazen Ghannam6e8462392019-02-28 15:36:09 +00002311 [F17_M30H_CPUS] = {
2312 .ctl_name = "F17h_M30h",
2313 .f0_id = PCI_DEVICE_ID_AMD_17H_M30H_DF_F0,
2314 .f6_id = PCI_DEVICE_ID_AMD_17H_M30H_DF_F6,
2315 .ops = {
2316 .early_channel_count = f17_early_channel_count,
Yazen Ghanname53a3b22019-08-21 23:59:59 +00002317 .dbam_to_cs = f17_addr_mask_to_cs_size,
Yazen Ghannam6e8462392019-02-28 15:36:09 +00002318 }
2319 },
Isaac Vaughn3e443eb2019-09-06 23:21:38 +00002320 [F17_M70H_CPUS] = {
2321 .ctl_name = "F17h_M70h",
2322 .f0_id = PCI_DEVICE_ID_AMD_17H_M70H_DF_F0,
2323 .f6_id = PCI_DEVICE_ID_AMD_17H_M70H_DF_F6,
2324 .ops = {
2325 .early_channel_count = f17_early_channel_count,
2326 .dbam_to_cs = f17_addr_mask_to_cs_size,
2327 }
2328 },
Doug Thompson4d376072009-04-27 16:25:05 +02002329};
2330
Doug Thompsonb1289d62009-04-27 16:37:05 +02002331/*
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002332 * These are tables of eigenvectors (one per line) which can be used for the
2333 * construction of the syndrome tables. The modified syndrome search algorithm
2334 * uses those to find the symbol in error and thus the DIMM.
Doug Thompsonb1289d62009-04-27 16:37:05 +02002335 *
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002336 * Algorithm courtesy of Ross LaFetra from AMD.
Doug Thompsonb1289d62009-04-27 16:37:05 +02002337 */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002338static const u16 x4_vectors[] = {
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002339 0x2f57, 0x1afe, 0x66cc, 0xdd88,
2340 0x11eb, 0x3396, 0x7f4c, 0xeac8,
2341 0x0001, 0x0002, 0x0004, 0x0008,
2342 0x1013, 0x3032, 0x4044, 0x8088,
2343 0x106b, 0x30d6, 0x70fc, 0xe0a8,
2344 0x4857, 0xc4fe, 0x13cc, 0x3288,
2345 0x1ac5, 0x2f4a, 0x5394, 0xa1e8,
2346 0x1f39, 0x251e, 0xbd6c, 0x6bd8,
2347 0x15c1, 0x2a42, 0x89ac, 0x4758,
2348 0x2b03, 0x1602, 0x4f0c, 0xca08,
2349 0x1f07, 0x3a0e, 0x6b04, 0xbd08,
2350 0x8ba7, 0x465e, 0x244c, 0x1cc8,
2351 0x2b87, 0x164e, 0x642c, 0xdc18,
2352 0x40b9, 0x80de, 0x1094, 0x20e8,
2353 0x27db, 0x1eb6, 0x9dac, 0x7b58,
2354 0x11c1, 0x2242, 0x84ac, 0x4c58,
2355 0x1be5, 0x2d7a, 0x5e34, 0xa718,
2356 0x4b39, 0x8d1e, 0x14b4, 0x28d8,
2357 0x4c97, 0xc87e, 0x11fc, 0x33a8,
2358 0x8e97, 0x497e, 0x2ffc, 0x1aa8,
2359 0x16b3, 0x3d62, 0x4f34, 0x8518,
2360 0x1e2f, 0x391a, 0x5cac, 0xf858,
2361 0x1d9f, 0x3b7a, 0x572c, 0xfe18,
2362 0x15f5, 0x2a5a, 0x5264, 0xa3b8,
2363 0x1dbb, 0x3b66, 0x715c, 0xe3f8,
2364 0x4397, 0xc27e, 0x17fc, 0x3ea8,
2365 0x1617, 0x3d3e, 0x6464, 0xb8b8,
2366 0x23ff, 0x12aa, 0xab6c, 0x56d8,
2367 0x2dfb, 0x1ba6, 0x913c, 0x7328,
2368 0x185d, 0x2ca6, 0x7914, 0x9e28,
2369 0x171b, 0x3e36, 0x7d7c, 0xebe8,
2370 0x4199, 0x82ee, 0x19f4, 0x2e58,
2371 0x4807, 0xc40e, 0x130c, 0x3208,
2372 0x1905, 0x2e0a, 0x5804, 0xac08,
2373 0x213f, 0x132a, 0xadfc, 0x5ba8,
2374 0x19a9, 0x2efe, 0xb5cc, 0x6f88,
Doug Thompsonb1289d62009-04-27 16:37:05 +02002375};
2376
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002377static const u16 x8_vectors[] = {
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002378 0x0145, 0x028a, 0x2374, 0x43c8, 0xa1f0, 0x0520, 0x0a40, 0x1480,
2379 0x0211, 0x0422, 0x0844, 0x1088, 0x01b0, 0x44e0, 0x23c0, 0xed80,
2380 0x1011, 0x0116, 0x022c, 0x0458, 0x08b0, 0x8c60, 0x2740, 0x4e80,
2381 0x0411, 0x0822, 0x1044, 0x0158, 0x02b0, 0x2360, 0x46c0, 0xab80,
2382 0x0811, 0x1022, 0x012c, 0x0258, 0x04b0, 0x4660, 0x8cc0, 0x2780,
2383 0x2071, 0x40e2, 0xa0c4, 0x0108, 0x0210, 0x0420, 0x0840, 0x1080,
2384 0x4071, 0x80e2, 0x0104, 0x0208, 0x0410, 0x0820, 0x1040, 0x2080,
2385 0x8071, 0x0102, 0x0204, 0x0408, 0x0810, 0x1020, 0x2040, 0x4080,
2386 0x019d, 0x03d6, 0x136c, 0x2198, 0x50b0, 0xb2e0, 0x0740, 0x0e80,
2387 0x0189, 0x03ea, 0x072c, 0x0e58, 0x1cb0, 0x56e0, 0x37c0, 0xf580,
2388 0x01fd, 0x0376, 0x06ec, 0x0bb8, 0x1110, 0x2220, 0x4440, 0x8880,
2389 0x0163, 0x02c6, 0x1104, 0x0758, 0x0eb0, 0x2be0, 0x6140, 0xc280,
2390 0x02fd, 0x01c6, 0x0b5c, 0x1108, 0x07b0, 0x25a0, 0x8840, 0x6180,
2391 0x0801, 0x012e, 0x025c, 0x04b8, 0x1370, 0x26e0, 0x57c0, 0xb580,
2392 0x0401, 0x0802, 0x015c, 0x02b8, 0x22b0, 0x13e0, 0x7140, 0xe280,
2393 0x0201, 0x0402, 0x0804, 0x01b8, 0x11b0, 0x31a0, 0x8040, 0x7180,
2394 0x0101, 0x0202, 0x0404, 0x0808, 0x1010, 0x2020, 0x4040, 0x8080,
2395 0x0001, 0x0002, 0x0004, 0x0008, 0x0010, 0x0020, 0x0040, 0x0080,
2396 0x0100, 0x0200, 0x0400, 0x0800, 0x1000, 0x2000, 0x4000, 0x8000,
2397};
2398
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002399static int decode_syndrome(u16 syndrome, const u16 *vectors, unsigned num_vecs,
Borislav Petkovd34a6ec2011-02-23 17:41:50 +01002400 unsigned v_dim)
Doug Thompsonb1289d62009-04-27 16:37:05 +02002401{
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002402 unsigned int i, err_sym;
Doug Thompsonb1289d62009-04-27 16:37:05 +02002403
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002404 for (err_sym = 0; err_sym < num_vecs / v_dim; err_sym++) {
2405 u16 s = syndrome;
Borislav Petkovd34a6ec2011-02-23 17:41:50 +01002406 unsigned v_idx = err_sym * v_dim;
2407 unsigned v_end = (err_sym + 1) * v_dim;
Doug Thompsonb1289d62009-04-27 16:37:05 +02002408
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002409 /* walk over all 16 bits of the syndrome */
2410 for (i = 1; i < (1U << 16); i <<= 1) {
2411
2412 /* if bit is set in that eigenvector... */
2413 if (v_idx < v_end && vectors[v_idx] & i) {
2414 u16 ev_comp = vectors[v_idx++];
2415
2416 /* ... and bit set in the modified syndrome, */
2417 if (s & i) {
2418 /* remove it. */
2419 s ^= ev_comp;
2420
2421 if (!s)
2422 return err_sym;
2423 }
2424
2425 } else if (s & i)
2426 /* can't get to zero, move to next symbol */
2427 break;
2428 }
Doug Thompsonb1289d62009-04-27 16:37:05 +02002429 }
2430
Joe Perches956b9ba2012-04-29 17:08:39 -03002431 edac_dbg(0, "syndrome(%x) not found\n", syndrome);
Doug Thompsonb1289d62009-04-27 16:37:05 +02002432 return -1;
2433}
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002434
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002435static int map_err_sym_to_channel(int err_sym, int sym_size)
2436{
2437 if (sym_size == 4)
2438 switch (err_sym) {
2439 case 0x20:
2440 case 0x21:
2441 return 0;
2442 break;
2443 case 0x22:
2444 case 0x23:
2445 return 1;
2446 break;
2447 default:
2448 return err_sym >> 4;
2449 break;
2450 }
2451 /* x8 symbols */
2452 else
2453 switch (err_sym) {
2454 /* imaginary bits not in a DIMM */
2455 case 0x10:
2456 WARN(1, KERN_ERR "Invalid error symbol: 0x%x\n",
2457 err_sym);
2458 return -1;
2459 break;
2460
2461 case 0x11:
2462 return 0;
2463 break;
2464 case 0x12:
2465 return 1;
2466 break;
2467 default:
2468 return err_sym >> 3;
2469 break;
2470 }
2471 return -1;
2472}
2473
2474static int get_channel_from_ecc_syndrome(struct mem_ctl_info *mci, u16 syndrome)
2475{
2476 struct amd64_pvt *pvt = mci->pvt_info;
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002477 int err_sym = -1;
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002478
Borislav Petkova3b7db02011-01-19 20:35:12 +01002479 if (pvt->ecc_sym_sz == 8)
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002480 err_sym = decode_syndrome(syndrome, x8_vectors,
2481 ARRAY_SIZE(x8_vectors),
Borislav Petkova3b7db02011-01-19 20:35:12 +01002482 pvt->ecc_sym_sz);
2483 else if (pvt->ecc_sym_sz == 4)
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002484 err_sym = decode_syndrome(syndrome, x4_vectors,
2485 ARRAY_SIZE(x4_vectors),
Borislav Petkova3b7db02011-01-19 20:35:12 +01002486 pvt->ecc_sym_sz);
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002487 else {
Borislav Petkova3b7db02011-01-19 20:35:12 +01002488 amd64_warn("Illegal syndrome type: %u\n", pvt->ecc_sym_sz);
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002489 return err_sym;
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002490 }
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002491
Borislav Petkova3b7db02011-01-19 20:35:12 +01002492 return map_err_sym_to_channel(err_sym, pvt->ecc_sym_sz);
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002493}
2494
Yazen Ghanname70984d2016-11-17 17:57:31 -05002495static void __log_ecc_error(struct mem_ctl_info *mci, struct err_info *err,
Borislav Petkov33ca0642012-08-30 18:01:36 +02002496 u8 ecc_type)
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002497{
Borislav Petkov33ca0642012-08-30 18:01:36 +02002498 enum hw_event_mc_err_type err_type;
2499 const char *string;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002500
Borislav Petkov33ca0642012-08-30 18:01:36 +02002501 if (ecc_type == 2)
2502 err_type = HW_EVENT_ERR_CORRECTED;
2503 else if (ecc_type == 1)
2504 err_type = HW_EVENT_ERR_UNCORRECTED;
Yazen Ghannamd12a9692016-11-17 17:57:32 -05002505 else if (ecc_type == 3)
2506 err_type = HW_EVENT_ERR_DEFERRED;
Borislav Petkov33ca0642012-08-30 18:01:36 +02002507 else {
2508 WARN(1, "Something is rotten in the state of Denmark.\n");
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002509 return;
2510 }
2511
Borislav Petkov33ca0642012-08-30 18:01:36 +02002512 switch (err->err_code) {
2513 case DECODE_OK:
2514 string = "";
2515 break;
2516 case ERR_NODE:
2517 string = "Failed to map error addr to a node";
2518 break;
2519 case ERR_CSROW:
2520 string = "Failed to map error addr to a csrow";
2521 break;
2522 case ERR_CHANNEL:
Yazen Ghannam713ad542016-11-28 12:59:53 -06002523 string = "Unknown syndrome - possible error reporting race";
2524 break;
2525 case ERR_SYND:
2526 string = "MCA_SYND not valid - unknown syndrome and csrow";
2527 break;
2528 case ERR_NORM_ADDR:
2529 string = "Cannot decode normalized address";
Borislav Petkov33ca0642012-08-30 18:01:36 +02002530 break;
2531 default:
2532 string = "WTF error";
2533 break;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002534 }
Borislav Petkov33ca0642012-08-30 18:01:36 +02002535
2536 edac_mc_handle_error(err_type, mci, 1,
2537 err->page, err->offset, err->syndrome,
2538 err->csrow, err->channel, -1,
2539 string, "");
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002540}
2541
Borislav Petkovdf781d02013-12-15 17:29:44 +01002542static inline void decode_bus_error(int node_id, struct mce *m)
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002543{
Daniel J Blueman0c510cc2015-02-17 11:34:38 +08002544 struct mem_ctl_info *mci;
2545 struct amd64_pvt *pvt;
Borislav Petkovf192c7b2011-01-10 14:24:32 +01002546 u8 ecc_type = (m->status >> 45) & 0x3;
Borislav Petkov66fed2d2012-08-09 18:41:07 +02002547 u8 xec = XEC(m->status, 0x1f);
2548 u16 ec = EC(m->status);
Borislav Petkov33ca0642012-08-30 18:01:36 +02002549 u64 sys_addr;
2550 struct err_info err;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002551
Daniel J Blueman0c510cc2015-02-17 11:34:38 +08002552 mci = edac_mc_find(node_id);
2553 if (!mci)
2554 return;
2555
2556 pvt = mci->pvt_info;
2557
Borislav Petkov66fed2d2012-08-09 18:41:07 +02002558 /* Bail out early if this was an 'observed' error */
Borislav Petkov5980bb92011-01-07 16:26:49 +01002559 if (PP(ec) == NBSL_PP_OBS)
Borislav Petkovb70ef012009-06-25 19:32:38 +02002560 return;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002561
Borislav Petkovecaf5602009-07-23 16:32:01 +02002562 /* Do only ECC errors */
2563 if (xec && xec != F10_NBSL_EXT_ERR_ECC)
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002564 return;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002565
Borislav Petkov33ca0642012-08-30 18:01:36 +02002566 memset(&err, 0, sizeof(err));
2567
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002568 sys_addr = get_error_address(pvt, m);
Borislav Petkov33ca0642012-08-30 18:01:36 +02002569
Borislav Petkovecaf5602009-07-23 16:32:01 +02002570 if (ecc_type == 2)
Borislav Petkov33ca0642012-08-30 18:01:36 +02002571 err.syndrome = extract_syndrome(m->status);
2572
2573 pvt->ops->map_sysaddr_to_csrow(mci, sys_addr, &err);
2574
Yazen Ghanname70984d2016-11-17 17:57:31 -05002575 __log_ecc_error(mci, &err, ecc_type);
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002576}
2577
Doug Thompson0ec449e2009-04-27 19:41:25 +02002578/*
Yazen Ghannam713ad542016-11-28 12:59:53 -06002579 * To find the UMC channel represented by this bank we need to match on its
2580 * instance_id. The instance_id of a bank is held in the lower 32 bits of its
2581 * IPID.
Yazen Ghannambdcee772019-02-28 15:36:10 +00002582 *
2583 * Currently, we can derive the channel number by looking at the 6th nibble in
2584 * the instance_id. For example, instance_id=0xYXXXXX where Y is the channel
2585 * number.
Yazen Ghannam713ad542016-11-28 12:59:53 -06002586 */
Yazen Ghannambdcee772019-02-28 15:36:10 +00002587static int find_umc_channel(struct mce *m)
Yazen Ghannam713ad542016-11-28 12:59:53 -06002588{
Yazen Ghannambdcee772019-02-28 15:36:10 +00002589 return (m->ipid & GENMASK(31, 0)) >> 20;
Yazen Ghannam713ad542016-11-28 12:59:53 -06002590}
2591
2592static void decode_umc_error(int node_id, struct mce *m)
2593{
2594 u8 ecc_type = (m->status >> 45) & 0x3;
2595 struct mem_ctl_info *mci;
2596 struct amd64_pvt *pvt;
2597 struct err_info err;
2598 u64 sys_addr;
2599
2600 mci = edac_mc_find(node_id);
2601 if (!mci)
2602 return;
2603
2604 pvt = mci->pvt_info;
2605
2606 memset(&err, 0, sizeof(err));
2607
2608 if (m->status & MCI_STATUS_DEFERRED)
2609 ecc_type = 3;
2610
Yazen Ghannambdcee772019-02-28 15:36:10 +00002611 err.channel = find_umc_channel(m);
Yazen Ghannam713ad542016-11-28 12:59:53 -06002612
Yazen Ghannam713ad542016-11-28 12:59:53 -06002613 if (!(m->status & MCI_STATUS_SYNDV)) {
2614 err.err_code = ERR_SYND;
2615 goto log_error;
2616 }
2617
2618 if (ecc_type == 2) {
2619 u8 length = (m->synd >> 18) & 0x3f;
2620
2621 if (length)
2622 err.syndrome = (m->synd >> 32) & GENMASK(length - 1, 0);
2623 else
2624 err.err_code = ERR_CHANNEL;
2625 }
2626
2627 err.csrow = m->synd & 0x7;
2628
Yazen Ghannam8a2eaab2019-08-22 00:00:00 +00002629 if (umc_normaddr_to_sysaddr(m->addr, pvt->mc_node_id, err.channel, &sys_addr)) {
2630 err.err_code = ERR_NORM_ADDR;
2631 goto log_error;
2632 }
2633
2634 error_address_to_page_and_offset(sys_addr, &err);
2635
Yazen Ghannam713ad542016-11-28 12:59:53 -06002636log_error:
2637 __log_ecc_error(mci, &err, ecc_type);
2638}
2639
2640/*
Borislav Petkov3f37a362016-05-06 19:44:27 +02002641 * Use pvt->F3 which contains the F3 CPU PCI device to get the related
2642 * F1 (AddrMap) and F2 (Dct) devices. Return negative value on error.
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002643 * Reserve F0 and F6 on systems with a UMC.
Doug Thompson0ec449e2009-04-27 19:41:25 +02002644 */
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002645static int
2646reserve_mc_sibling_devs(struct amd64_pvt *pvt, u16 pci_id1, u16 pci_id2)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002647{
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002648 if (pvt->umc) {
2649 pvt->F0 = pci_get_related_function(pvt->F3->vendor, pci_id1, pvt->F3);
2650 if (!pvt->F0) {
Borislav Petkov5246c542016-12-01 11:35:07 +01002651 amd64_err("F0 not found, device 0x%x (broken BIOS?)\n", pci_id1);
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002652 return -ENODEV;
2653 }
2654
2655 pvt->F6 = pci_get_related_function(pvt->F3->vendor, pci_id2, pvt->F3);
2656 if (!pvt->F6) {
2657 pci_dev_put(pvt->F0);
2658 pvt->F0 = NULL;
2659
Borislav Petkov5246c542016-12-01 11:35:07 +01002660 amd64_err("F6 not found: device 0x%x (broken BIOS?)\n", pci_id2);
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002661 return -ENODEV;
2662 }
Borislav Petkov5246c542016-12-01 11:35:07 +01002663
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002664 edac_dbg(1, "F0: %s\n", pci_name(pvt->F0));
2665 edac_dbg(1, "F3: %s\n", pci_name(pvt->F3));
2666 edac_dbg(1, "F6: %s\n", pci_name(pvt->F6));
2667
2668 return 0;
2669 }
2670
Doug Thompson0ec449e2009-04-27 19:41:25 +02002671 /* Reserve the ADDRESS MAP Device */
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002672 pvt->F1 = pci_get_related_function(pvt->F3->vendor, pci_id1, pvt->F3);
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002673 if (!pvt->F1) {
Borislav Petkov5246c542016-12-01 11:35:07 +01002674 amd64_err("F1 not found: device 0x%x (broken BIOS?)\n", pci_id1);
Borislav Petkovbbd0c1f2010-10-01 19:27:58 +02002675 return -ENODEV;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002676 }
2677
Borislav Petkov3f37a362016-05-06 19:44:27 +02002678 /* Reserve the DCT Device */
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002679 pvt->F2 = pci_get_related_function(pvt->F3->vendor, pci_id2, pvt->F3);
Borislav Petkov3f37a362016-05-06 19:44:27 +02002680 if (!pvt->F2) {
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002681 pci_dev_put(pvt->F1);
2682 pvt->F1 = NULL;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002683
Borislav Petkov5246c542016-12-01 11:35:07 +01002684 amd64_err("F2 not found: device 0x%x (broken BIOS?)\n", pci_id2);
2685 return -ENODEV;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002686 }
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002687
Joe Perches956b9ba2012-04-29 17:08:39 -03002688 edac_dbg(1, "F1: %s\n", pci_name(pvt->F1));
2689 edac_dbg(1, "F2: %s\n", pci_name(pvt->F2));
2690 edac_dbg(1, "F3: %s\n", pci_name(pvt->F3));
Doug Thompson0ec449e2009-04-27 19:41:25 +02002691
2692 return 0;
2693}
2694
Borislav Petkov360b7f32010-10-15 19:25:38 +02002695static void free_mc_sibling_devs(struct amd64_pvt *pvt)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002696{
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002697 if (pvt->umc) {
2698 pci_dev_put(pvt->F0);
2699 pci_dev_put(pvt->F6);
2700 } else {
2701 pci_dev_put(pvt->F1);
2702 pci_dev_put(pvt->F2);
2703 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002704}
2705
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002706static void determine_ecc_sym_sz(struct amd64_pvt *pvt)
2707{
2708 pvt->ecc_sym_sz = 4;
2709
2710 if (pvt->umc) {
2711 u8 i;
2712
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00002713 for_each_umc(i) {
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002714 /* Check enabled channels only: */
Yazen Ghannam78359612019-02-28 15:36:11 +00002715 if (pvt->umc[i].sdp_ctrl & UMC_SDP_INIT) {
2716 if (pvt->umc[i].ecc_ctrl & BIT(9)) {
2717 pvt->ecc_sym_sz = 16;
2718 return;
2719 } else if (pvt->umc[i].ecc_ctrl & BIT(7)) {
2720 pvt->ecc_sym_sz = 8;
2721 return;
2722 }
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002723 }
2724 }
Yazen Ghannam78359612019-02-28 15:36:11 +00002725 } else if (pvt->fam >= 0x10) {
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002726 u32 tmp;
2727
2728 amd64_read_pci_cfg(pvt->F3, EXT_NB_MCA_CFG, &tmp);
2729 /* F16h has only DCT0, so no need to read dbam1. */
2730 if (pvt->fam != 0x16)
2731 amd64_read_dct_pci_cfg(pvt, 1, DBAM0, &pvt->dbam1);
2732
2733 /* F10h, revD and later can do x8 ECC too. */
2734 if ((pvt->fam > 0x10 || pvt->model > 7) && tmp & BIT(25))
2735 pvt->ecc_sym_sz = 8;
2736 }
2737}
2738
2739/*
2740 * Retrieve the hardware registers of the memory controller.
2741 */
2742static void __read_mc_regs_df(struct amd64_pvt *pvt)
2743{
2744 u8 nid = pvt->mc_node_id;
2745 struct amd64_umc *umc;
2746 u32 i, umc_base;
2747
2748 /* Read registers from each UMC */
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00002749 for_each_umc(i) {
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002750
2751 umc_base = get_umc_base(i);
2752 umc = &pvt->umc[i];
2753
Yazen Ghannam07ed82e2016-11-28 08:50:21 -06002754 amd_smn_read(nid, umc_base + UMCCH_DIMM_CFG, &umc->dimm_cfg);
2755 amd_smn_read(nid, umc_base + UMCCH_UMC_CFG, &umc->umc_cfg);
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002756 amd_smn_read(nid, umc_base + UMCCH_SDP_CTRL, &umc->sdp_ctrl);
2757 amd_smn_read(nid, umc_base + UMCCH_ECC_CTRL, &umc->ecc_ctrl);
Yazen Ghannam07ed82e2016-11-28 08:50:21 -06002758 amd_smn_read(nid, umc_base + UMCCH_UMC_CAP_HI, &umc->umc_cap_hi);
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002759 }
2760}
2761
Doug Thompson0ec449e2009-04-27 19:41:25 +02002762/*
2763 * Retrieve the hardware registers of the memory controller (this includes the
2764 * 'Address Map' and 'Misc' device regs)
2765 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02002766static void read_mc_regs(struct amd64_pvt *pvt)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002767{
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002768 unsigned int range;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002769 u64 msr_val;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002770
2771 /*
2772 * Retrieve TOP_MEM and TOP_MEM2; no masking off of reserved bits since
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002773 * those are Read-As-Zero.
Doug Thompson0ec449e2009-04-27 19:41:25 +02002774 */
Borislav Petkove97f8bb2009-10-12 15:27:45 +02002775 rdmsrl(MSR_K8_TOP_MEM1, pvt->top_mem);
Joe Perches956b9ba2012-04-29 17:08:39 -03002776 edac_dbg(0, " TOP_MEM: 0x%016llx\n", pvt->top_mem);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002777
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002778 /* Check first whether TOP_MEM2 is enabled: */
Doug Thompson0ec449e2009-04-27 19:41:25 +02002779 rdmsrl(MSR_K8_SYSCFG, msr_val);
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002780 if (msr_val & BIT(21)) {
Borislav Petkove97f8bb2009-10-12 15:27:45 +02002781 rdmsrl(MSR_K8_TOP_MEM2, pvt->top_mem2);
Joe Perches956b9ba2012-04-29 17:08:39 -03002782 edac_dbg(0, " TOP_MEM2: 0x%016llx\n", pvt->top_mem2);
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002783 } else {
Joe Perches956b9ba2012-04-29 17:08:39 -03002784 edac_dbg(0, " TOP_MEM2 disabled\n");
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002785 }
2786
2787 if (pvt->umc) {
2788 __read_mc_regs_df(pvt);
2789 amd64_read_pci_cfg(pvt->F0, DF_DHAR, &pvt->dhar);
2790
2791 goto skip;
2792 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002793
Borislav Petkov5980bb92011-01-07 16:26:49 +01002794 amd64_read_pci_cfg(pvt->F3, NBCAP, &pvt->nbcap);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002795
Borislav Petkov5a5d2372011-01-17 17:52:57 +01002796 read_dram_ctl_register(pvt);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002797
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002798 for (range = 0; range < DRAM_RANGES; range++) {
2799 u8 rw;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002800
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002801 /* read settings for this DRAM range */
2802 read_dram_base_limit_regs(pvt, range);
Borislav Petkove97f8bb2009-10-12 15:27:45 +02002803
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002804 rw = dram_rw(pvt, range);
2805 if (!rw)
2806 continue;
2807
Joe Perches956b9ba2012-04-29 17:08:39 -03002808 edac_dbg(1, " DRAM range[%d], base: 0x%016llx; limit: 0x%016llx\n",
2809 range,
2810 get_dram_base(pvt, range),
2811 get_dram_limit(pvt, range));
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002812
Joe Perches956b9ba2012-04-29 17:08:39 -03002813 edac_dbg(1, " IntlvEn=%s; Range access: %s%s IntlvSel=%d DstNode=%d\n",
2814 dram_intlv_en(pvt, range) ? "Enabled" : "Disabled",
2815 (rw & 0x1) ? "R" : "-",
2816 (rw & 0x2) ? "W" : "-",
2817 dram_intlv_sel(pvt, range),
2818 dram_dst_node(pvt, range));
Doug Thompson0ec449e2009-04-27 19:41:25 +02002819 }
2820
Borislav Petkovbc21fa52010-11-11 17:29:13 +01002821 amd64_read_pci_cfg(pvt->F1, DHAR, &pvt->dhar);
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002822 amd64_read_dct_pci_cfg(pvt, 0, DBAM0, &pvt->dbam0);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002823
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002824 amd64_read_pci_cfg(pvt->F3, F10_ONLINE_SPARE, &pvt->online_spare);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002825
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002826 amd64_read_dct_pci_cfg(pvt, 0, DCLR0, &pvt->dclr0);
2827 amd64_read_dct_pci_cfg(pvt, 0, DCHR0, &pvt->dchr0);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002828
Borislav Petkov78da1212010-12-22 19:31:45 +01002829 if (!dct_ganging_enabled(pvt)) {
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002830 amd64_read_dct_pci_cfg(pvt, 1, DCLR0, &pvt->dclr1);
2831 amd64_read_dct_pci_cfg(pvt, 1, DCHR0, &pvt->dchr1);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002832 }
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002833
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002834skip:
2835 read_dct_base_mask(pvt);
2836
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002837 determine_memory_type(pvt);
2838 edac_dbg(1, " DIMM type: %s\n", edac_mem_types[pvt->dram_type]);
Borislav Petkova3b7db02011-01-19 20:35:12 +01002839
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002840 determine_ecc_sym_sz(pvt);
Borislav Petkova3b7db02011-01-19 20:35:12 +01002841
Borislav Petkovb2b0c602010-10-08 18:32:29 +02002842 dump_misc_regs(pvt);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002843}
2844
2845/*
2846 * NOTE: CPU Revision Dependent code
2847 *
2848 * Input:
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002849 * @csrow_nr ChipSelect Row Number (0..NUM_CHIPSELECTS-1)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002850 * k8 private pointer to -->
2851 * DRAM Bank Address mapping register
2852 * node_id
2853 * DCL register where dual_channel_active is
2854 *
2855 * The DBAM register consists of 4 sets of 4 bits each definitions:
2856 *
2857 * Bits: CSROWs
2858 * 0-3 CSROWs 0 and 1
2859 * 4-7 CSROWs 2 and 3
2860 * 8-11 CSROWs 4 and 5
2861 * 12-15 CSROWs 6 and 7
2862 *
2863 * Values range from: 0 to 15
2864 * The meaning of the values depends on CPU revision and dual-channel state,
2865 * see relevant BKDG more info.
2866 *
2867 * The memory controller provides for total of only 8 CSROWs in its current
2868 * architecture. Each "pair" of CSROWs normally represents just one DIMM in
2869 * single channel or two (2) DIMMs in dual channel mode.
2870 *
2871 * The following code logic collapses the various tables for CSROW based on CPU
2872 * revision.
2873 *
2874 * Returns:
2875 * The number of PAGE_SIZE pages on the specified CSROW number it
2876 * encompasses
2877 *
2878 */
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002879static u32 get_csrow_nr_pages(struct amd64_pvt *pvt, u8 dct, int csrow_nr_orig)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002880{
Ashish Shenoyf92cae42012-02-22 17:20:38 -08002881 u32 dbam = dct ? pvt->dbam1 : pvt->dbam0;
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002882 int csrow_nr = csrow_nr_orig;
2883 u32 cs_mode, nr_pages;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002884
Yazen Ghanname53a3b22019-08-21 23:59:59 +00002885 if (!pvt->umc) {
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002886 csrow_nr >>= 1;
Yazen Ghanname53a3b22019-08-21 23:59:59 +00002887 cs_mode = DBAM_DIMM(csrow_nr, dbam);
2888 } else {
2889 cs_mode = f17_get_cs_mode(csrow_nr >> 1, dct, pvt);
2890 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002891
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002892 nr_pages = pvt->ops->dbam_to_cs(pvt, dct, cs_mode, csrow_nr);
2893 nr_pages <<= 20 - PAGE_SHIFT;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002894
Borislav Petkov10de6492012-09-12 19:00:38 +02002895 edac_dbg(0, "csrow: %d, channel: %d, DBAM idx: %d\n",
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002896 csrow_nr_orig, dct, cs_mode);
Borislav Petkov10de6492012-09-12 19:00:38 +02002897 edac_dbg(0, "nr_pages/channel: %u\n", nr_pages);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002898
2899 return nr_pages;
2900}
2901
Yazen Ghannam353a1fc2019-08-21 23:59:57 +00002902static int init_csrows_df(struct mem_ctl_info *mci)
2903{
2904 struct amd64_pvt *pvt = mci->pvt_info;
2905 enum edac_type edac_mode = EDAC_NONE;
2906 enum dev_type dev_type = DEV_UNKNOWN;
2907 struct dimm_info *dimm;
2908 int empty = 1;
2909 u8 umc, cs;
2910
2911 if (mci->edac_ctl_cap & EDAC_FLAG_S16ECD16ED) {
2912 edac_mode = EDAC_S16ECD16ED;
2913 dev_type = DEV_X16;
2914 } else if (mci->edac_ctl_cap & EDAC_FLAG_S8ECD8ED) {
2915 edac_mode = EDAC_S8ECD8ED;
2916 dev_type = DEV_X8;
2917 } else if (mci->edac_ctl_cap & EDAC_FLAG_S4ECD4ED) {
2918 edac_mode = EDAC_S4ECD4ED;
2919 dev_type = DEV_X4;
2920 } else if (mci->edac_ctl_cap & EDAC_FLAG_SECDED) {
2921 edac_mode = EDAC_SECDED;
2922 }
2923
2924 for_each_umc(umc) {
2925 for_each_chip_select(cs, umc, pvt) {
2926 if (!csrow_enabled(cs, umc, pvt))
2927 continue;
2928
2929 empty = 0;
2930 dimm = mci->csrows[cs]->channels[umc]->dimm;
2931
2932 edac_dbg(1, "MC node: %d, csrow: %d\n",
2933 pvt->mc_node_id, cs);
2934
2935 dimm->nr_pages = get_csrow_nr_pages(pvt, umc, cs);
2936 dimm->mtype = pvt->dram_type;
2937 dimm->edac_mode = edac_mode;
2938 dimm->dtype = dev_type;
2939 }
2940 }
2941
2942 return empty;
2943}
2944
Doug Thompson0ec449e2009-04-27 19:41:25 +02002945/*
2946 * Initialize the array of csrow attribute instances, based on the values
2947 * from pci config hardware registers.
2948 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02002949static int init_csrows(struct mem_ctl_info *mci)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002950{
Borislav Petkov10de6492012-09-12 19:00:38 +02002951 struct amd64_pvt *pvt = mci->pvt_info;
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06002952 enum edac_type edac_mode = EDAC_NONE;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002953 struct csrow_info *csrow;
Mauro Carvalho Chehabde3910eb2012-04-24 15:05:43 -03002954 struct dimm_info *dimm;
Borislav Petkov10de6492012-09-12 19:00:38 +02002955 int i, j, empty = 1;
Mauro Carvalho Chehaba895bf82012-01-28 09:09:38 -03002956 int nr_pages = 0;
Borislav Petkov10de6492012-09-12 19:00:38 +02002957 u32 val;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002958
Yazen Ghannam353a1fc2019-08-21 23:59:57 +00002959 if (pvt->umc)
2960 return init_csrows_df(mci);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002961
Yazen Ghannam353a1fc2019-08-21 23:59:57 +00002962 amd64_read_pci_cfg(pvt->F3, NBCFG, &val);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002963
Yazen Ghannam353a1fc2019-08-21 23:59:57 +00002964 pvt->nbcfg = val;
2965
2966 edac_dbg(0, "node %d, NBCFG=0x%08x[ChipKillEccCap: %d|DramEccEn: %d]\n",
2967 pvt->mc_node_id, val,
2968 !!(val & NBCFG_CHIPKILL), !!(val & NBCFG_ECC_ENABLE));
Doug Thompson0ec449e2009-04-27 19:41:25 +02002969
Borislav Petkov10de6492012-09-12 19:00:38 +02002970 /*
2971 * We iterate over DCT0 here but we look at DCT1 in parallel, if needed.
2972 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002973 for_each_chip_select(i, 0, pvt) {
Borislav Petkov10de6492012-09-12 19:00:38 +02002974 bool row_dct0 = !!csrow_enabled(i, 0, pvt);
2975 bool row_dct1 = false;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002976
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002977 if (pvt->fam != 0xf)
Borislav Petkov10de6492012-09-12 19:00:38 +02002978 row_dct1 = !!csrow_enabled(i, 1, pvt);
2979
2980 if (!row_dct0 && !row_dct1)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002981 continue;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002982
Borislav Petkov10de6492012-09-12 19:00:38 +02002983 csrow = mci->csrows[i];
Doug Thompson0ec449e2009-04-27 19:41:25 +02002984 empty = 0;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002985
Borislav Petkov10de6492012-09-12 19:00:38 +02002986 edac_dbg(1, "MC node: %d, csrow: %d\n",
2987 pvt->mc_node_id, i);
2988
Mauro Carvalho Chehab1eef1282013-03-11 09:07:46 -03002989 if (row_dct0) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002990 nr_pages = get_csrow_nr_pages(pvt, 0, i);
Mauro Carvalho Chehab1eef1282013-03-11 09:07:46 -03002991 csrow->channels[0]->dimm->nr_pages = nr_pages;
2992 }
Borislav Petkov10de6492012-09-12 19:00:38 +02002993
2994 /* K8 has only one DCT */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002995 if (pvt->fam != 0xf && row_dct1) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002996 int row_dct1_pages = get_csrow_nr_pages(pvt, 1, i);
Mauro Carvalho Chehab1eef1282013-03-11 09:07:46 -03002997
2998 csrow->channels[1]->dimm->nr_pages = row_dct1_pages;
2999 nr_pages += row_dct1_pages;
3000 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02003001
Borislav Petkov10de6492012-09-12 19:00:38 +02003002 edac_dbg(1, "Total csrow%d pages: %u\n", i, nr_pages);
Doug Thompson0ec449e2009-04-27 19:41:25 +02003003
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003004 /* Determine DIMM ECC mode: */
Yazen Ghannam353a1fc2019-08-21 23:59:57 +00003005 if (pvt->nbcfg & NBCFG_ECC_ENABLE) {
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003006 edac_mode = (pvt->nbcfg & NBCFG_CHIPKILL)
3007 ? EDAC_S4ECD4ED
3008 : EDAC_SECDED;
3009 }
Mauro Carvalho Chehab084a4fc2012-01-27 18:38:08 -03003010
3011 for (j = 0; j < pvt->channel_count; j++) {
Mauro Carvalho Chehabde3910eb2012-04-24 15:05:43 -03003012 dimm = csrow->channels[j]->dimm;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01003013 dimm->mtype = pvt->dram_type;
Mauro Carvalho Chehabde3910eb2012-04-24 15:05:43 -03003014 dimm->edac_mode = edac_mode;
Mauro Carvalho Chehab084a4fc2012-01-27 18:38:08 -03003015 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02003016 }
3017
3018 return empty;
3019}
Doug Thompsond27bf6f2009-05-06 17:55:27 +02003020
Borislav Petkov06724532009-09-16 13:05:46 +02003021/* get all cores on this DCT */
Daniel J Blueman8b84c8d2012-11-27 14:32:10 +08003022static void get_cpus_on_this_dct_cpumask(struct cpumask *mask, u16 nid)
Doug Thompsonf9431992009-04-27 19:46:08 +02003023{
Borislav Petkov06724532009-09-16 13:05:46 +02003024 int cpu;
Doug Thompsonf9431992009-04-27 19:46:08 +02003025
Borislav Petkov06724532009-09-16 13:05:46 +02003026 for_each_online_cpu(cpu)
3027 if (amd_get_nb_id(cpu) == nid)
3028 cpumask_set_cpu(cpu, mask);
Doug Thompsonf9431992009-04-27 19:46:08 +02003029}
3030
3031/* check MCG_CTL on all the cpus on this node */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003032static bool nb_mce_bank_enabled_on_node(u16 nid)
Doug Thompsonf9431992009-04-27 19:46:08 +02003033{
Rusty Russellba578cb2009-11-03 14:56:35 +10303034 cpumask_var_t mask;
Borislav Petkov50542252009-12-11 18:14:40 +01003035 int cpu, nbe;
Borislav Petkov06724532009-09-16 13:05:46 +02003036 bool ret = false;
Doug Thompsonf9431992009-04-27 19:46:08 +02003037
Rusty Russellba578cb2009-11-03 14:56:35 +10303038 if (!zalloc_cpumask_var(&mask, GFP_KERNEL)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003039 amd64_warn("%s: Error allocating mask\n", __func__);
Rusty Russellba578cb2009-11-03 14:56:35 +10303040 return false;
3041 }
Borislav Petkov06724532009-09-16 13:05:46 +02003042
Rusty Russellba578cb2009-11-03 14:56:35 +10303043 get_cpus_on_this_dct_cpumask(mask, nid);
Borislav Petkov06724532009-09-16 13:05:46 +02003044
Rusty Russellba578cb2009-11-03 14:56:35 +10303045 rdmsr_on_cpus(mask, MSR_IA32_MCG_CTL, msrs);
Borislav Petkov06724532009-09-16 13:05:46 +02003046
Rusty Russellba578cb2009-11-03 14:56:35 +10303047 for_each_cpu(cpu, mask) {
Borislav Petkov50542252009-12-11 18:14:40 +01003048 struct msr *reg = per_cpu_ptr(msrs, cpu);
Borislav Petkov5980bb92011-01-07 16:26:49 +01003049 nbe = reg->l & MSR_MCGCTL_NBE;
Borislav Petkov06724532009-09-16 13:05:46 +02003050
Joe Perches956b9ba2012-04-29 17:08:39 -03003051 edac_dbg(0, "core: %u, MCG_CTL: 0x%llx, NB MSR is %s\n",
3052 cpu, reg->q,
3053 (nbe ? "enabled" : "disabled"));
Borislav Petkov06724532009-09-16 13:05:46 +02003054
3055 if (!nbe)
3056 goto out;
Borislav Petkov06724532009-09-16 13:05:46 +02003057 }
3058 ret = true;
3059
3060out:
Rusty Russellba578cb2009-11-03 14:56:35 +10303061 free_cpumask_var(mask);
Doug Thompsonf9431992009-04-27 19:46:08 +02003062 return ret;
3063}
3064
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08003065static int toggle_ecc_err_reporting(struct ecc_settings *s, u16 nid, bool on)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003066{
3067 cpumask_var_t cmask;
Borislav Petkov50542252009-12-11 18:14:40 +01003068 int cpu;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003069
3070 if (!zalloc_cpumask_var(&cmask, GFP_KERNEL)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003071 amd64_warn("%s: error allocating mask\n", __func__);
Pan Bian0de27882016-12-04 14:07:18 +08003072 return -ENOMEM;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003073 }
3074
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003075 get_cpus_on_this_dct_cpumask(cmask, nid);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003076
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003077 rdmsr_on_cpus(cmask, MSR_IA32_MCG_CTL, msrs);
3078
3079 for_each_cpu(cpu, cmask) {
3080
Borislav Petkov50542252009-12-11 18:14:40 +01003081 struct msr *reg = per_cpu_ptr(msrs, cpu);
3082
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003083 if (on) {
Borislav Petkov5980bb92011-01-07 16:26:49 +01003084 if (reg->l & MSR_MCGCTL_NBE)
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003085 s->flags.nb_mce_enable = 1;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003086
Borislav Petkov5980bb92011-01-07 16:26:49 +01003087 reg->l |= MSR_MCGCTL_NBE;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003088 } else {
3089 /*
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01003090 * Turn off NB MCE reporting only when it was off before
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003091 */
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003092 if (!s->flags.nb_mce_enable)
Borislav Petkov5980bb92011-01-07 16:26:49 +01003093 reg->l &= ~MSR_MCGCTL_NBE;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003094 }
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003095 }
3096 wrmsr_on_cpus(cmask, MSR_IA32_MCG_CTL, msrs);
3097
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003098 free_cpumask_var(cmask);
3099
3100 return 0;
3101}
3102
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08003103static bool enable_ecc_error_reporting(struct ecc_settings *s, u16 nid,
Borislav Petkov2299ef72010-10-15 17:44:04 +02003104 struct pci_dev *F3)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003105{
Borislav Petkov2299ef72010-10-15 17:44:04 +02003106 bool ret = true;
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003107 u32 value, mask = 0x3; /* UECC/CECC enable */
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003108
Borislav Petkov2299ef72010-10-15 17:44:04 +02003109 if (toggle_ecc_err_reporting(s, nid, ON)) {
3110 amd64_warn("Error enabling ECC reporting over MCGCTL!\n");
3111 return false;
3112 }
3113
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003114 amd64_read_pci_cfg(F3, NBCTL, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003115
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003116 s->old_nbctl = value & mask;
3117 s->nbctl_valid = true;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003118
3119 value |= mask;
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003120 amd64_write_pci_cfg(F3, NBCTL, value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003121
Borislav Petkova97fa682010-12-23 14:07:18 +01003122 amd64_read_pci_cfg(F3, NBCFG, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003123
Joe Perches956b9ba2012-04-29 17:08:39 -03003124 edac_dbg(0, "1: node %d, NBCFG=0x%08x[DramEccEn: %d]\n",
3125 nid, value, !!(value & NBCFG_ECC_ENABLE));
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003126
Borislav Petkova97fa682010-12-23 14:07:18 +01003127 if (!(value & NBCFG_ECC_ENABLE)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003128 amd64_warn("DRAM ECC disabled on this node, enabling...\n");
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003129
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003130 s->flags.nb_ecc_prev = 0;
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01003131
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003132 /* Attempt to turn on DRAM ECC Enable */
Borislav Petkova97fa682010-12-23 14:07:18 +01003133 value |= NBCFG_ECC_ENABLE;
3134 amd64_write_pci_cfg(F3, NBCFG, value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003135
Borislav Petkova97fa682010-12-23 14:07:18 +01003136 amd64_read_pci_cfg(F3, NBCFG, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003137
Borislav Petkova97fa682010-12-23 14:07:18 +01003138 if (!(value & NBCFG_ECC_ENABLE)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003139 amd64_warn("Hardware rejected DRAM ECC enable,"
3140 "check memory DIMM configuration.\n");
Borislav Petkov2299ef72010-10-15 17:44:04 +02003141 ret = false;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003142 } else {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003143 amd64_info("Hardware accepted DRAM ECC Enable\n");
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003144 }
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01003145 } else {
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003146 s->flags.nb_ecc_prev = 1;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003147 }
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01003148
Joe Perches956b9ba2012-04-29 17:08:39 -03003149 edac_dbg(0, "2: node %d, NBCFG=0x%08x[DramEccEn: %d]\n",
3150 nid, value, !!(value & NBCFG_ECC_ENABLE));
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003151
Borislav Petkov2299ef72010-10-15 17:44:04 +02003152 return ret;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003153}
3154
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08003155static void restore_ecc_error_reporting(struct ecc_settings *s, u16 nid,
Borislav Petkov360b7f32010-10-15 19:25:38 +02003156 struct pci_dev *F3)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003157{
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003158 u32 value, mask = 0x3; /* UECC/CECC enable */
3159
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003160 if (!s->nbctl_valid)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003161 return;
3162
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003163 amd64_read_pci_cfg(F3, NBCTL, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003164 value &= ~mask;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003165 value |= s->old_nbctl;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003166
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003167 amd64_write_pci_cfg(F3, NBCTL, value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003168
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003169 /* restore previous BIOS DRAM ECC "off" setting we force-enabled */
3170 if (!s->flags.nb_ecc_prev) {
Borislav Petkova97fa682010-12-23 14:07:18 +01003171 amd64_read_pci_cfg(F3, NBCFG, &value);
3172 value &= ~NBCFG_ECC_ENABLE;
3173 amd64_write_pci_cfg(F3, NBCFG, value);
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01003174 }
3175
3176 /* restore the NB Enable MCGCTL bit */
Borislav Petkov2299ef72010-10-15 17:44:04 +02003177 if (toggle_ecc_err_reporting(s, nid, OFF))
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003178 amd64_warn("Error restoring NB MCGCTL settings!\n");
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003179}
3180
Doug Thompsonf9431992009-04-27 19:46:08 +02003181/*
Borislav Petkov2299ef72010-10-15 17:44:04 +02003182 * EDAC requires that the BIOS have ECC enabled before
3183 * taking over the processing of ECC errors. A command line
3184 * option allows to force-enable hardware ECC later in
3185 * enable_ecc_error_reporting().
Doug Thompsonf9431992009-04-27 19:46:08 +02003186 */
Borislav Petkovcab4d272010-02-11 17:15:57 +01003187static const char *ecc_msg =
3188 "ECC disabled in the BIOS or no ECC capability, module will not load.\n"
3189 " Either enable ECC checking or force module loading by setting "
3190 "'ecc_enable_override'.\n"
3191 " (Note that use of the override may cause unknown side effects.)\n";
Borislav Petkovbe3468e2009-08-05 15:47:22 +02003192
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08003193static bool ecc_enabled(struct pci_dev *F3, u16 nid)
Doug Thompsonf9431992009-04-27 19:46:08 +02003194{
Borislav Petkov06724532009-09-16 13:05:46 +02003195 bool nb_mce_en = false;
Yazen Ghannam196b79f2016-11-17 17:57:34 -05003196 u8 ecc_en = 0, i;
3197 u32 value;
Doug Thompsonf9431992009-04-27 19:46:08 +02003198
Yazen Ghannam196b79f2016-11-17 17:57:34 -05003199 if (boot_cpu_data.x86 >= 0x17) {
3200 u8 umc_en_mask = 0, ecc_en_mask = 0;
Doug Thompsonf9431992009-04-27 19:46:08 +02003201
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00003202 for_each_umc(i) {
Yazen Ghannam196b79f2016-11-17 17:57:34 -05003203 u32 base = get_umc_base(i);
3204
3205 /* Only check enabled UMCs. */
3206 if (amd_smn_read(nid, base + UMCCH_SDP_CTRL, &value))
3207 continue;
3208
3209 if (!(value & UMC_SDP_INIT))
3210 continue;
3211
3212 umc_en_mask |= BIT(i);
3213
3214 if (amd_smn_read(nid, base + UMCCH_UMC_CAP_HI, &value))
3215 continue;
3216
3217 if (value & UMC_ECC_ENABLED)
3218 ecc_en_mask |= BIT(i);
3219 }
3220
3221 /* Check whether at least one UMC is enabled: */
3222 if (umc_en_mask)
3223 ecc_en = umc_en_mask == ecc_en_mask;
Yazen Ghannam11ab1ca2017-01-27 11:24:19 -06003224 else
3225 edac_dbg(0, "Node %d: No enabled UMCs.\n", nid);
Yazen Ghannam196b79f2016-11-17 17:57:34 -05003226
3227 /* Assume UMC MCA banks are enabled. */
3228 nb_mce_en = true;
3229 } else {
3230 amd64_read_pci_cfg(F3, NBCFG, &value);
3231
3232 ecc_en = !!(value & NBCFG_ECC_ENABLE);
3233
3234 nb_mce_en = nb_mce_bank_enabled_on_node(nid);
3235 if (!nb_mce_en)
Yazen Ghannam11ab1ca2017-01-27 11:24:19 -06003236 edac_dbg(0, "NB MCE bank disabled, set MSR 0x%08x[4] on node %d to enable.\n",
Yazen Ghannam196b79f2016-11-17 17:57:34 -05003237 MSR_IA32_MCG_CTL, nid);
3238 }
3239
Yazen Ghannam11ab1ca2017-01-27 11:24:19 -06003240 amd64_info("Node %d: DRAM ECC %s.\n",
3241 nid, (ecc_en ? "enabled" : "disabled"));
Doug Thompsonf9431992009-04-27 19:46:08 +02003242
Borislav Petkov2299ef72010-10-15 17:44:04 +02003243 if (!ecc_en || !nb_mce_en) {
Yazen Ghannam11ab1ca2017-01-27 11:24:19 -06003244 amd64_info("%s", ecc_msg);
Borislav Petkov2299ef72010-10-15 17:44:04 +02003245 return false;
Borislav Petkov43f5e682009-12-21 18:55:18 +01003246 }
Borislav Petkov2299ef72010-10-15 17:44:04 +02003247 return true;
Doug Thompsonf9431992009-04-27 19:46:08 +02003248}
3249
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003250static inline void
3251f17h_determine_edac_ctl_cap(struct mem_ctl_info *mci, struct amd64_pvt *pvt)
3252{
Yazen Ghannamf8be8e52019-08-21 23:59:56 +00003253 u8 i, ecc_en = 1, cpk_en = 1, dev_x4 = 1, dev_x16 = 1;
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003254
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00003255 for_each_umc(i) {
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003256 if (pvt->umc[i].sdp_ctrl & UMC_SDP_INIT) {
3257 ecc_en &= !!(pvt->umc[i].umc_cap_hi & UMC_ECC_ENABLED);
3258 cpk_en &= !!(pvt->umc[i].umc_cap_hi & UMC_ECC_CHIPKILL_CAP);
Yazen Ghannamf8be8e52019-08-21 23:59:56 +00003259
3260 dev_x4 &= !!(pvt->umc[i].dimm_cfg & BIT(6));
3261 dev_x16 &= !!(pvt->umc[i].dimm_cfg & BIT(7));
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003262 }
3263 }
3264
3265 /* Set chipkill only if ECC is enabled: */
3266 if (ecc_en) {
3267 mci->edac_ctl_cap |= EDAC_FLAG_SECDED;
3268
Yazen Ghannamf8be8e52019-08-21 23:59:56 +00003269 if (!cpk_en)
3270 return;
3271
3272 if (dev_x4)
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003273 mci->edac_ctl_cap |= EDAC_FLAG_S4ECD4ED;
Yazen Ghannamf8be8e52019-08-21 23:59:56 +00003274 else if (dev_x16)
3275 mci->edac_ctl_cap |= EDAC_FLAG_S16ECD16ED;
3276 else
3277 mci->edac_ctl_cap |= EDAC_FLAG_S8ECD8ED;
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003278 }
3279}
3280
Borislav Petkovdf71a052011-01-19 18:15:10 +01003281static void setup_mci_misc_attrs(struct mem_ctl_info *mci,
3282 struct amd64_family_type *fam)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003283{
3284 struct amd64_pvt *pvt = mci->pvt_info;
3285
3286 mci->mtype_cap = MEM_FLAG_DDR2 | MEM_FLAG_RDDR2;
3287 mci->edac_ctl_cap = EDAC_FLAG_NONE;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003288
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003289 if (pvt->umc) {
3290 f17h_determine_edac_ctl_cap(mci, pvt);
3291 } else {
3292 if (pvt->nbcap & NBCAP_SECDED)
3293 mci->edac_ctl_cap |= EDAC_FLAG_SECDED;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003294
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003295 if (pvt->nbcap & NBCAP_CHIPKILL)
3296 mci->edac_ctl_cap |= EDAC_FLAG_S4ECD4ED;
3297 }
Doug Thompson7d6034d2009-04-27 20:01:01 +02003298
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003299 mci->edac_cap = determine_edac_cap(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003300 mci->mod_name = EDAC_MOD_STR;
Borislav Petkovdf71a052011-01-19 18:15:10 +01003301 mci->ctl_name = fam->ctl_name;
Yazen Ghanname7934b72016-11-17 17:57:30 -05003302 mci->dev_name = pci_name(pvt->F3);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003303 mci->ctl_page_to_phys = NULL;
3304
Doug Thompson7d6034d2009-04-27 20:01:01 +02003305 /* memory scrubber interface */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003306 mci->set_sdram_scrub_rate = set_scrub_rate;
3307 mci->get_sdram_scrub_rate = get_scrub_rate;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003308}
3309
Borislav Petkov0092b202010-10-01 19:20:05 +02003310/*
3311 * returns a pointer to the family descriptor on success, NULL otherwise.
3312 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003313static struct amd64_family_type *per_family_init(struct amd64_pvt *pvt)
Borislav Petkov395ae782010-10-01 18:38:19 +02003314{
Borislav Petkov0092b202010-10-01 19:20:05 +02003315 struct amd64_family_type *fam_type = NULL;
3316
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003317 pvt->ext_model = boot_cpu_data.x86_model >> 4;
Jia Zhangb3991512018-01-01 09:52:10 +08003318 pvt->stepping = boot_cpu_data.x86_stepping;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003319 pvt->model = boot_cpu_data.x86_model;
3320 pvt->fam = boot_cpu_data.x86;
3321
3322 switch (pvt->fam) {
Borislav Petkov395ae782010-10-01 18:38:19 +02003323 case 0xf:
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003324 fam_type = &family_types[K8_CPUS];
3325 pvt->ops = &family_types[K8_CPUS].ops;
Borislav Petkov395ae782010-10-01 18:38:19 +02003326 break;
Borislav Petkovdf71a052011-01-19 18:15:10 +01003327
Borislav Petkov395ae782010-10-01 18:38:19 +02003328 case 0x10:
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003329 fam_type = &family_types[F10_CPUS];
3330 pvt->ops = &family_types[F10_CPUS].ops;
Borislav Petkovdf71a052011-01-19 18:15:10 +01003331 break;
3332
3333 case 0x15:
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003334 if (pvt->model == 0x30) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003335 fam_type = &family_types[F15_M30H_CPUS];
3336 pvt->ops = &family_types[F15_M30H_CPUS].ops;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003337 break;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01003338 } else if (pvt->model == 0x60) {
3339 fam_type = &family_types[F15_M60H_CPUS];
3340 pvt->ops = &family_types[F15_M60H_CPUS].ops;
3341 break;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003342 }
3343
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003344 fam_type = &family_types[F15_CPUS];
3345 pvt->ops = &family_types[F15_CPUS].ops;
Borislav Petkov395ae782010-10-01 18:38:19 +02003346 break;
3347
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05003348 case 0x16:
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06003349 if (pvt->model == 0x30) {
3350 fam_type = &family_types[F16_M30H_CPUS];
3351 pvt->ops = &family_types[F16_M30H_CPUS].ops;
3352 break;
3353 }
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003354 fam_type = &family_types[F16_CPUS];
3355 pvt->ops = &family_types[F16_CPUS].ops;
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05003356 break;
3357
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05003358 case 0x17:
Michael Jin8960de42018-08-16 15:28:40 -04003359 if (pvt->model >= 0x10 && pvt->model <= 0x2f) {
3360 fam_type = &family_types[F17_M10H_CPUS];
3361 pvt->ops = &family_types[F17_M10H_CPUS].ops;
3362 break;
Yazen Ghannam6e8462392019-02-28 15:36:09 +00003363 } else if (pvt->model >= 0x30 && pvt->model <= 0x3f) {
3364 fam_type = &family_types[F17_M30H_CPUS];
3365 pvt->ops = &family_types[F17_M30H_CPUS].ops;
3366 break;
Isaac Vaughn3e443eb2019-09-06 23:21:38 +00003367 } else if (pvt->model >= 0x70 && pvt->model <= 0x7f) {
3368 fam_type = &family_types[F17_M70H_CPUS];
3369 pvt->ops = &family_types[F17_M70H_CPUS].ops;
3370 break;
Michael Jin8960de42018-08-16 15:28:40 -04003371 }
Pu Wenc4a3e942018-09-27 16:31:28 +02003372 /* fall through */
3373 case 0x18:
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05003374 fam_type = &family_types[F17_CPUS];
3375 pvt->ops = &family_types[F17_CPUS].ops;
Pu Wenc4a3e942018-09-27 16:31:28 +02003376
3377 if (pvt->fam == 0x18)
3378 family_types[F17_CPUS].ctl_name = "F18h";
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05003379 break;
3380
Borislav Petkov395ae782010-10-01 18:38:19 +02003381 default:
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003382 amd64_err("Unsupported family!\n");
Borislav Petkov0092b202010-10-01 19:20:05 +02003383 return NULL;
Borislav Petkov395ae782010-10-01 18:38:19 +02003384 }
Borislav Petkov0092b202010-10-01 19:20:05 +02003385
Borislav Petkovdf71a052011-01-19 18:15:10 +01003386 amd64_info("%s %sdetected (node %d).\n", fam_type->ctl_name,
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003387 (pvt->fam == 0xf ?
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003388 (pvt->ext_model >= K8_REV_F ? "revF or later "
3389 : "revE or earlier ")
3390 : ""), pvt->mc_node_id);
Borislav Petkov0092b202010-10-01 19:20:05 +02003391 return fam_type;
Borislav Petkov395ae782010-10-01 18:38:19 +02003392}
3393
Takashi Iwaie339f1e2015-02-04 11:48:53 +01003394static const struct attribute_group *amd64_edac_attr_groups[] = {
3395#ifdef CONFIG_EDAC_DEBUG
3396 &amd64_edac_dbg_group,
3397#endif
3398#ifdef CONFIG_EDAC_AMD64_ERROR_INJECTION
3399 &amd64_edac_inj_group,
3400#endif
3401 NULL
3402};
3403
Yazen Ghannambdcee772019-02-28 15:36:10 +00003404/* Set the number of Unified Memory Controllers in the system. */
3405static void compute_num_umcs(void)
3406{
3407 u8 model = boot_cpu_data.x86_model;
3408
3409 if (boot_cpu_data.x86 < 0x17)
3410 return;
3411
3412 if (model >= 0x30 && model <= 0x3f)
3413 num_umcs = 8;
3414 else
3415 num_umcs = 2;
3416
3417 edac_dbg(1, "Number of UMCs: %x", num_umcs);
3418}
3419
Borislav Petkov3f37a362016-05-06 19:44:27 +02003420static int init_one_instance(unsigned int nid)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003421{
Borislav Petkov3f37a362016-05-06 19:44:27 +02003422 struct pci_dev *F3 = node_to_amd_nb(nid)->misc;
Borislav Petkov0092b202010-10-01 19:20:05 +02003423 struct amd64_family_type *fam_type = NULL;
Borislav Petkov360b7f32010-10-15 19:25:38 +02003424 struct mem_ctl_info *mci = NULL;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03003425 struct edac_mc_layer layers[2];
Borislav Petkov3f37a362016-05-06 19:44:27 +02003426 struct amd64_pvt *pvt = NULL;
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003427 u16 pci_id1, pci_id2;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003428 int err = 0, ret;
3429
3430 ret = -ENOMEM;
3431 pvt = kzalloc(sizeof(struct amd64_pvt), GFP_KERNEL);
3432 if (!pvt)
Borislav Petkov360b7f32010-10-15 19:25:38 +02003433 goto err_ret;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003434
Borislav Petkov360b7f32010-10-15 19:25:38 +02003435 pvt->mc_node_id = nid;
Borislav Petkov3f37a362016-05-06 19:44:27 +02003436 pvt->F3 = F3;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003437
Borislav Petkov395ae782010-10-01 18:38:19 +02003438 ret = -EINVAL;
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003439 fam_type = per_family_init(pvt);
Borislav Petkov0092b202010-10-01 19:20:05 +02003440 if (!fam_type)
Borislav Petkov395ae782010-10-01 18:38:19 +02003441 goto err_free;
3442
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003443 if (pvt->fam >= 0x17) {
Yazen Ghannambdcee772019-02-28 15:36:10 +00003444 pvt->umc = kcalloc(num_umcs, sizeof(struct amd64_umc), GFP_KERNEL);
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003445 if (!pvt->umc) {
3446 ret = -ENOMEM;
3447 goto err_free;
3448 }
3449
3450 pci_id1 = fam_type->f0_id;
3451 pci_id2 = fam_type->f6_id;
3452 } else {
3453 pci_id1 = fam_type->f1_id;
3454 pci_id2 = fam_type->f2_id;
3455 }
3456
3457 err = reserve_mc_sibling_devs(pvt, pci_id1, pci_id2);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003458 if (err)
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003459 goto err_post_init;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003460
Borislav Petkov360b7f32010-10-15 19:25:38 +02003461 read_mc_regs(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003462
Doug Thompson7d6034d2009-04-27 20:01:01 +02003463 /*
3464 * We need to determine how many memory channels there are. Then use
3465 * that information for calculating the size of the dynamic instance
Borislav Petkov360b7f32010-10-15 19:25:38 +02003466 * tables in the 'mci' structure.
Doug Thompson7d6034d2009-04-27 20:01:01 +02003467 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02003468 ret = -EINVAL;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003469 pvt->channel_count = pvt->ops->early_channel_count(pvt);
3470 if (pvt->channel_count < 0)
Borislav Petkov360b7f32010-10-15 19:25:38 +02003471 goto err_siblings;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003472
3473 ret = -ENOMEM;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03003474 layers[0].type = EDAC_MC_LAYER_CHIP_SELECT;
3475 layers[0].size = pvt->csels[0].b_cnt;
3476 layers[0].is_virt_csrow = true;
3477 layers[1].type = EDAC_MC_LAYER_CHANNEL;
Borislav Petkovf0a56c42013-07-23 20:01:23 +02003478
3479 /*
3480 * Always allocate two channels since we can have setups with DIMMs on
3481 * only one channel. Also, this simplifies handling later for the price
3482 * of a couple of KBs tops.
Yazen Ghannam869adc42019-03-25 20:33:30 +00003483 *
3484 * On Fam17h+, the number of controllers may be greater than two. So set
3485 * the size equal to the maximum number of UMCs.
Borislav Petkovf0a56c42013-07-23 20:01:23 +02003486 */
Yazen Ghannam869adc42019-03-25 20:33:30 +00003487 if (pvt->fam >= 0x17)
3488 layers[1].size = num_umcs;
3489 else
3490 layers[1].size = 2;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03003491 layers[1].is_virt_csrow = false;
Borislav Petkovf0a56c42013-07-23 20:01:23 +02003492
Mauro Carvalho Chehabca0907b2012-05-02 14:37:00 -03003493 mci = edac_mc_alloc(nid, ARRAY_SIZE(layers), layers, 0);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003494 if (!mci)
Borislav Petkov360b7f32010-10-15 19:25:38 +02003495 goto err_siblings;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003496
3497 mci->pvt_info = pvt;
Borislav Petkov3f37a362016-05-06 19:44:27 +02003498 mci->pdev = &pvt->F3->dev;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003499
Borislav Petkovdf71a052011-01-19 18:15:10 +01003500 setup_mci_misc_attrs(mci, fam_type);
Borislav Petkov360b7f32010-10-15 19:25:38 +02003501
3502 if (init_csrows(mci))
Doug Thompson7d6034d2009-04-27 20:01:01 +02003503 mci->edac_cap = EDAC_FLAG_NONE;
3504
Doug Thompson7d6034d2009-04-27 20:01:01 +02003505 ret = -ENODEV;
Takashi Iwaie339f1e2015-02-04 11:48:53 +01003506 if (edac_mc_add_mc_with_groups(mci, amd64_edac_attr_groups)) {
Joe Perches956b9ba2012-04-29 17:08:39 -03003507 edac_dbg(1, "failed edac_mc_add_mc()\n");
Doug Thompson7d6034d2009-04-27 20:01:01 +02003508 goto err_add_mc;
3509 }
3510
Doug Thompson7d6034d2009-04-27 20:01:01 +02003511 return 0;
3512
3513err_add_mc:
3514 edac_mc_free(mci);
3515
Borislav Petkov360b7f32010-10-15 19:25:38 +02003516err_siblings:
3517 free_mc_sibling_devs(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003518
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003519err_post_init:
3520 if (pvt->fam >= 0x17)
3521 kfree(pvt->umc);
3522
Borislav Petkov360b7f32010-10-15 19:25:38 +02003523err_free:
3524 kfree(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003525
Borislav Petkov360b7f32010-10-15 19:25:38 +02003526err_ret:
Doug Thompson7d6034d2009-04-27 20:01:01 +02003527 return ret;
3528}
3529
Borislav Petkov3f37a362016-05-06 19:44:27 +02003530static int probe_one_instance(unsigned int nid)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003531{
Borislav Petkov2299ef72010-10-15 17:44:04 +02003532 struct pci_dev *F3 = node_to_amd_nb(nid)->misc;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003533 struct ecc_settings *s;
Borislav Petkov3f37a362016-05-06 19:44:27 +02003534 int ret;
Borislav Petkovb8cfa022010-10-01 19:35:38 +02003535
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003536 ret = -ENOMEM;
3537 s = kzalloc(sizeof(struct ecc_settings), GFP_KERNEL);
3538 if (!s)
Borislav Petkov2299ef72010-10-15 17:44:04 +02003539 goto err_out;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003540
3541 ecc_stngs[nid] = s;
3542
Borislav Petkov2299ef72010-10-15 17:44:04 +02003543 if (!ecc_enabled(F3, nid)) {
Yazen Ghannam4688c9b2017-01-27 11:24:22 -06003544 ret = 0;
Borislav Petkov2299ef72010-10-15 17:44:04 +02003545
3546 if (!ecc_enable_override)
3547 goto err_enable;
3548
Yazen Ghannam044e7a42016-11-22 15:40:16 -06003549 if (boot_cpu_data.x86 >= 0x17) {
3550 amd64_warn("Forcing ECC on is not recommended on newer systems. Please enable ECC in BIOS.");
3551 goto err_enable;
3552 } else
3553 amd64_warn("Forcing ECC on!\n");
Borislav Petkov2299ef72010-10-15 17:44:04 +02003554
3555 if (!enable_ecc_error_reporting(s, nid, F3))
3556 goto err_enable;
3557 }
3558
Borislav Petkov3f37a362016-05-06 19:44:27 +02003559 ret = init_one_instance(nid);
Borislav Petkov360b7f32010-10-15 19:25:38 +02003560 if (ret < 0) {
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003561 amd64_err("Error probing instance: %d\n", nid);
Yazen Ghannam044e7a42016-11-22 15:40:16 -06003562
3563 if (boot_cpu_data.x86 < 0x17)
3564 restore_ecc_error_reporting(s, nid, F3);
Yazen Ghannam2b9b2c42017-01-24 16:32:24 -06003565
3566 goto err_enable;
Borislav Petkov360b7f32010-10-15 19:25:38 +02003567 }
Doug Thompson7d6034d2009-04-27 20:01:01 +02003568
3569 return ret;
Borislav Petkov2299ef72010-10-15 17:44:04 +02003570
3571err_enable:
3572 kfree(s);
3573 ecc_stngs[nid] = NULL;
3574
3575err_out:
3576 return ret;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003577}
3578
Borislav Petkov3f37a362016-05-06 19:44:27 +02003579static void remove_one_instance(unsigned int nid)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003580{
Borislav Petkov360b7f32010-10-15 19:25:38 +02003581 struct pci_dev *F3 = node_to_amd_nb(nid)->misc;
3582 struct ecc_settings *s = ecc_stngs[nid];
Borislav Petkov3f37a362016-05-06 19:44:27 +02003583 struct mem_ctl_info *mci;
3584 struct amd64_pvt *pvt;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003585
Borislav Petkov3f37a362016-05-06 19:44:27 +02003586 mci = find_mci_by_dev(&F3->dev);
Borislav Petkova4b4bed2013-08-10 13:54:48 +02003587 WARN_ON(!mci);
3588
Doug Thompson7d6034d2009-04-27 20:01:01 +02003589 /* Remove from EDAC CORE tracking list */
Borislav Petkov3f37a362016-05-06 19:44:27 +02003590 mci = edac_mc_del_mc(&F3->dev);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003591 if (!mci)
3592 return;
3593
3594 pvt = mci->pvt_info;
3595
Borislav Petkov360b7f32010-10-15 19:25:38 +02003596 restore_ecc_error_reporting(s, nid, F3);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003597
Borislav Petkov360b7f32010-10-15 19:25:38 +02003598 free_mc_sibling_devs(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003599
Borislav Petkov360b7f32010-10-15 19:25:38 +02003600 kfree(ecc_stngs[nid]);
3601 ecc_stngs[nid] = NULL;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003602
Doug Thompson7d6034d2009-04-27 20:01:01 +02003603 /* Free the EDAC CORE resources */
Borislav Petkov8f68ed92009-12-21 15:15:59 +01003604 mci->pvt_info = NULL;
Borislav Petkov8f68ed92009-12-21 15:15:59 +01003605
3606 kfree(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003607 edac_mc_free(mci);
3608}
3609
Borislav Petkov360b7f32010-10-15 19:25:38 +02003610static void setup_pci_device(void)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003611{
3612 struct mem_ctl_info *mci;
3613 struct amd64_pvt *pvt;
3614
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003615 if (pci_ctl)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003616 return;
3617
Borislav Petkov2ec591a2015-02-17 10:58:34 +01003618 mci = edac_mc_find(0);
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003619 if (!mci)
3620 return;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003621
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003622 pvt = mci->pvt_info;
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003623 if (pvt->umc)
3624 pci_ctl = edac_pci_create_generic_ctl(&pvt->F0->dev, EDAC_MOD_STR);
3625 else
3626 pci_ctl = edac_pci_create_generic_ctl(&pvt->F2->dev, EDAC_MOD_STR);
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003627 if (!pci_ctl) {
3628 pr_warn("%s(): Unable to create PCI control\n", __func__);
3629 pr_warn("%s(): PCI error report via EDAC not set\n", __func__);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003630 }
3631}
3632
Yazen Ghannamd6efab72016-09-15 19:07:17 -05003633static const struct x86_cpu_id amd64_cpuids[] = {
3634 { X86_VENDOR_AMD, 0xF, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
3635 { X86_VENDOR_AMD, 0x10, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
3636 { X86_VENDOR_AMD, 0x15, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
3637 { X86_VENDOR_AMD, 0x16, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
Yazen Ghannam95d3af62016-11-17 17:57:43 -05003638 { X86_VENDOR_AMD, 0x17, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
Pu Wenc4a3e942018-09-27 16:31:28 +02003639 { X86_VENDOR_HYGON, 0x18, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
Yazen Ghannamd6efab72016-09-15 19:07:17 -05003640 { }
3641};
3642MODULE_DEVICE_TABLE(x86cpu, amd64_cpuids);
3643
Doug Thompson7d6034d2009-04-27 20:01:01 +02003644static int __init amd64_edac_init(void)
3645{
Toshi Kani301375e2017-08-23 16:54:47 -06003646 const char *owner;
Borislav Petkov360b7f32010-10-15 19:25:38 +02003647 int err = -ENODEV;
Borislav Petkov3f37a362016-05-06 19:44:27 +02003648 int i;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003649
Toshi Kani301375e2017-08-23 16:54:47 -06003650 owner = edac_get_owner();
3651 if (owner && strncmp(owner, EDAC_MOD_STR, sizeof(EDAC_MOD_STR)))
3652 return -EBUSY;
3653
Yazen Ghannam1bd99002017-01-27 11:24:23 -06003654 if (!x86_match_cpu(amd64_cpuids))
3655 return -ENODEV;
3656
Hans Rosenfeld9653a5c2010-10-29 17:14:31 +02003657 if (amd_cache_northbridges() < 0)
Yazen Ghannam1bd99002017-01-27 11:24:23 -06003658 return -ENODEV;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003659
Borislav Petkov6ba92fe2016-06-16 01:13:18 +02003660 opstate_init();
3661
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003662 err = -ENOMEM;
Kees Cook6396bb22018-06-12 14:03:40 -07003663 ecc_stngs = kcalloc(amd_nb_num(), sizeof(ecc_stngs[0]), GFP_KERNEL);
Borislav Petkov2ec591a2015-02-17 10:58:34 +01003664 if (!ecc_stngs)
Borislav Petkova9f0fbe2011-03-29 18:10:53 +02003665 goto err_free;
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003666
Borislav Petkov50542252009-12-11 18:14:40 +01003667 msrs = msrs_alloc();
Borislav Petkov56b34b92009-12-21 18:13:01 +01003668 if (!msrs)
Borislav Petkov360b7f32010-10-15 19:25:38 +02003669 goto err_free;
Borislav Petkov50542252009-12-11 18:14:40 +01003670
Yazen Ghannambdcee772019-02-28 15:36:10 +00003671 compute_num_umcs();
3672
Yazen Ghannam2287c632017-01-13 09:52:19 -06003673 for (i = 0; i < amd_nb_num(); i++) {
3674 err = probe_one_instance(i);
3675 if (err) {
Borislav Petkov3f37a362016-05-06 19:44:27 +02003676 /* unwind properly */
3677 while (--i >= 0)
3678 remove_one_instance(i);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003679
Borislav Petkov3f37a362016-05-06 19:44:27 +02003680 goto err_pci;
3681 }
Yazen Ghannam2287c632017-01-13 09:52:19 -06003682 }
Doug Thompson7d6034d2009-04-27 20:01:01 +02003683
Yazen Ghannam4688c9b2017-01-27 11:24:22 -06003684 if (!edac_has_mcs()) {
3685 err = -ENODEV;
3686 goto err_pci;
3687 }
3688
Yazen Ghannam234365f2017-01-24 16:32:25 -06003689 /* register stuff with EDAC MCE */
3690 if (report_gart_errors)
3691 amd_report_gart_errors(true);
3692
3693 if (boot_cpu_data.x86 >= 0x17)
3694 amd_register_ecc_decoder(decode_umc_error);
3695 else
3696 amd_register_ecc_decoder(decode_bus_error);
3697
Borislav Petkov360b7f32010-10-15 19:25:38 +02003698 setup_pci_device();
Tomasz Palaf5b10c42014-11-02 11:22:12 +01003699
3700#ifdef CONFIG_X86_32
3701 amd64_err("%s on 32-bit is unsupported. USE AT YOUR OWN RISK!\n", EDAC_MOD_STR);
3702#endif
3703
Borislav Petkovde0336b2016-04-27 12:21:21 +02003704 printk(KERN_INFO "AMD64 EDAC driver v%s\n", EDAC_AMD64_VERSION);
3705
Borislav Petkov360b7f32010-10-15 19:25:38 +02003706 return 0;
Borislav Petkov56b34b92009-12-21 18:13:01 +01003707
Borislav Petkov56b34b92009-12-21 18:13:01 +01003708err_pci:
3709 msrs_free(msrs);
3710 msrs = NULL;
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003711
Borislav Petkov360b7f32010-10-15 19:25:38 +02003712err_free:
Borislav Petkov360b7f32010-10-15 19:25:38 +02003713 kfree(ecc_stngs);
3714 ecc_stngs = NULL;
3715
Doug Thompson7d6034d2009-04-27 20:01:01 +02003716 return err;
3717}
3718
3719static void __exit amd64_edac_exit(void)
3720{
Borislav Petkov3f37a362016-05-06 19:44:27 +02003721 int i;
3722
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003723 if (pci_ctl)
3724 edac_pci_release_generic_ctl(pci_ctl);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003725
Yazen Ghannam234365f2017-01-24 16:32:25 -06003726 /* unregister from EDAC MCE */
3727 amd_report_gart_errors(false);
3728
3729 if (boot_cpu_data.x86 >= 0x17)
3730 amd_unregister_ecc_decoder(decode_umc_error);
3731 else
3732 amd_unregister_ecc_decoder(decode_bus_error);
3733
Borislav Petkov3f37a362016-05-06 19:44:27 +02003734 for (i = 0; i < amd_nb_num(); i++)
3735 remove_one_instance(i);
Borislav Petkov50542252009-12-11 18:14:40 +01003736
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003737 kfree(ecc_stngs);
3738 ecc_stngs = NULL;
3739
Borislav Petkov50542252009-12-11 18:14:40 +01003740 msrs_free(msrs);
3741 msrs = NULL;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003742}
3743
3744module_init(amd64_edac_init);
3745module_exit(amd64_edac_exit);
3746
3747MODULE_LICENSE("GPL");
3748MODULE_AUTHOR("SoftwareBitMaker: Doug Thompson, "
3749 "Dave Peterson, Thayne Harbaugh");
3750MODULE_DESCRIPTION("MC support for AMD64 memory controllers - "
3751 EDAC_AMD64_VERSION);
3752
3753module_param(edac_op_state, int, 0444);
3754MODULE_PARM_DESC(edac_op_state, "EDAC Error Reporting state: 0=Poll,1=NMI");