blob: 873437be86d9c2d571953ef7624c06a0bb868f78 [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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Perches956b9ba12012-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 Ghannamfc00c6a2019-02-28 15:36:12 +0000791/*
792 * The Address Mask should be a contiguous set of bits in the non-interleaved
793 * case. So to check for CS interleaving, find the most- and least-significant
794 * bits of the mask, generate a contiguous bitmask, and compare the two.
795 */
796static bool f17_cs_interleaved(struct amd64_pvt *pvt, u8 ctrl, int cs)
797{
798 u32 mask = pvt->csels[ctrl].csmasks[cs >> 1];
799 u32 msb = fls(mask) - 1, lsb = ffs(mask) - 1;
800 u32 test_mask = GENMASK(msb, lsb);
801
802 edac_dbg(1, "mask=0x%08x test_mask=0x%08x\n", mask, test_mask);
803
804 return mask ^ test_mask;
805}
806
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600807static void debug_display_dimm_sizes_df(struct amd64_pvt *pvt, u8 ctrl)
808{
Yazen Ghannameb77e6b2017-04-27 12:11:54 -0500809 int dimm, size0, size1, cs0, cs1;
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600810
811 edac_printk(KERN_DEBUG, EDAC_MC, "UMC%d chip selects:\n", ctrl);
812
813 for (dimm = 0; dimm < 4; dimm++) {
814 size0 = 0;
Yazen Ghannameb77e6b2017-04-27 12:11:54 -0500815 cs0 = dimm * 2;
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600816
Yazen Ghannameb77e6b2017-04-27 12:11:54 -0500817 if (csrow_enabled(cs0, ctrl, pvt))
818 size0 = pvt->ops->dbam_to_cs(pvt, ctrl, 0, cs0);
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600819
820 size1 = 0;
Yazen Ghannameb77e6b2017-04-27 12:11:54 -0500821 cs1 = dimm * 2 + 1;
822
Yazen Ghannamfc00c6a2019-02-28 15:36:12 +0000823 if (csrow_enabled(cs1, ctrl, pvt)) {
824 /*
825 * CS interleaving is only supported if both CSes have
826 * the same amount of memory. Because they are
827 * interleaved, it will look like both CSes have the
828 * full amount of memory. Save the size for both as
829 * half the amount we found on CS0, if interleaved.
830 */
831 if (f17_cs_interleaved(pvt, ctrl, cs1))
832 size1 = size0 = (size0 >> 1);
833 else
834 size1 = pvt->ops->dbam_to_cs(pvt, ctrl, 0, cs1);
835 }
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600836
837 amd64_info(EDAC_MC ": %d: %5dMB %d: %5dMB\n",
Yazen Ghannameb77e6b2017-04-27 12:11:54 -0500838 cs0, size0,
839 cs1, size1);
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600840 }
841}
842
843static void __dump_misc_regs_df(struct amd64_pvt *pvt)
844{
845 struct amd64_umc *umc;
846 u32 i, tmp, umc_base;
847
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +0000848 for_each_umc(i) {
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600849 umc_base = get_umc_base(i);
850 umc = &pvt->umc[i];
851
852 edac_dbg(1, "UMC%d DIMM cfg: 0x%x\n", i, umc->dimm_cfg);
853 edac_dbg(1, "UMC%d UMC cfg: 0x%x\n", i, umc->umc_cfg);
854 edac_dbg(1, "UMC%d SDP ctrl: 0x%x\n", i, umc->sdp_ctrl);
855 edac_dbg(1, "UMC%d ECC ctrl: 0x%x\n", i, umc->ecc_ctrl);
856
857 amd_smn_read(pvt->mc_node_id, umc_base + UMCCH_ECC_BAD_SYMBOL, &tmp);
858 edac_dbg(1, "UMC%d ECC bad symbol: 0x%x\n", i, tmp);
859
860 amd_smn_read(pvt->mc_node_id, umc_base + UMCCH_UMC_CAP, &tmp);
861 edac_dbg(1, "UMC%d UMC cap: 0x%x\n", i, tmp);
862 edac_dbg(1, "UMC%d UMC cap high: 0x%x\n", i, umc->umc_cap_hi);
863
864 edac_dbg(1, "UMC%d ECC capable: %s, ChipKill ECC capable: %s\n",
865 i, (umc->umc_cap_hi & BIT(30)) ? "yes" : "no",
866 (umc->umc_cap_hi & BIT(31)) ? "yes" : "no");
867 edac_dbg(1, "UMC%d All DIMMs support ECC: %s\n",
868 i, (umc->umc_cfg & BIT(12)) ? "yes" : "no");
869 edac_dbg(1, "UMC%d x4 DIMMs present: %s\n",
870 i, (umc->dimm_cfg & BIT(6)) ? "yes" : "no");
871 edac_dbg(1, "UMC%d x16 DIMMs present: %s\n",
872 i, (umc->dimm_cfg & BIT(7)) ? "yes" : "no");
873
874 if (pvt->dram_type == MEM_LRDDR4) {
875 amd_smn_read(pvt->mc_node_id, umc_base + UMCCH_ADDR_CFG, &tmp);
876 edac_dbg(1, "UMC%d LRDIMM %dx rank multiply\n",
877 i, 1 << ((tmp >> 4) & 0x3));
878 }
879
880 debug_display_dimm_sizes_df(pvt, i);
881 }
882
883 edac_dbg(1, "F0x104 (DRAM Hole Address): 0x%08x, base: 0x%08x\n",
884 pvt->dhar, dhar_base(pvt));
885}
886
Doug Thompson2da11652009-04-27 16:09:09 +0200887/* Display and decode various NB registers for debug purposes. */
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600888static void __dump_misc_regs(struct amd64_pvt *pvt)
Doug Thompson2da11652009-04-27 16:09:09 +0200889{
Joe Perches956b9ba12012-04-29 17:08:39 -0300890 edac_dbg(1, "F3xE8 (NB Cap): 0x%08x\n", pvt->nbcap);
Doug Thompson2da11652009-04-27 16:09:09 +0200891
Joe Perches956b9ba12012-04-29 17:08:39 -0300892 edac_dbg(1, " NB two channel DRAM capable: %s\n",
893 (pvt->nbcap & NBCAP_DCT_DUAL) ? "yes" : "no");
Borislav Petkov68798e12009-11-03 16:18:33 +0100894
Joe Perches956b9ba12012-04-29 17:08:39 -0300895 edac_dbg(1, " ECC capable: %s, ChipKill ECC capable: %s\n",
896 (pvt->nbcap & NBCAP_SECDED) ? "yes" : "no",
897 (pvt->nbcap & NBCAP_CHIPKILL) ? "yes" : "no");
Borislav Petkov68798e12009-11-03 16:18:33 +0100898
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100899 debug_dump_dramcfg_low(pvt, pvt->dclr0, 0);
Doug Thompson2da11652009-04-27 16:09:09 +0200900
Joe Perches956b9ba12012-04-29 17:08:39 -0300901 edac_dbg(1, "F3xB0 (Online Spare): 0x%08x\n", pvt->online_spare);
Doug Thompson2da11652009-04-27 16:09:09 +0200902
Joe Perches956b9ba12012-04-29 17:08:39 -0300903 edac_dbg(1, "F1xF0 (DRAM Hole Address): 0x%08x, base: 0x%08x, offset: 0x%08x\n",
904 pvt->dhar, dhar_base(pvt),
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200905 (pvt->fam == 0xf) ? k8_dhar_offset(pvt)
906 : f10_dhar_offset(pvt));
Doug Thompson2da11652009-04-27 16:09:09 +0200907
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100908 debug_display_dimm_sizes(pvt, 0);
Borislav Petkov4d796362011-02-03 15:59:57 +0100909
Borislav Petkov8de1d912009-10-16 13:39:30 +0200910 /* everything below this point is Fam10h and above */
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200911 if (pvt->fam == 0xf)
Doug Thompson2da11652009-04-27 16:09:09 +0200912 return;
Borislav Petkov4d796362011-02-03 15:59:57 +0100913
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100914 debug_display_dimm_sizes(pvt, 1);
Doug Thompson2da11652009-04-27 16:09:09 +0200915
Borislav Petkov8de1d912009-10-16 13:39:30 +0200916 /* Only if NOT ganged does dclr1 have valid info */
Borislav Petkov68798e12009-11-03 16:18:33 +0100917 if (!dct_ganging_enabled(pvt))
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100918 debug_dump_dramcfg_low(pvt, pvt->dclr1, 1);
Doug Thompson2da11652009-04-27 16:09:09 +0200919}
920
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600921/* Display and decode various NB registers for debug purposes. */
922static void dump_misc_regs(struct amd64_pvt *pvt)
923{
924 if (pvt->umc)
925 __dump_misc_regs_df(pvt);
926 else
927 __dump_misc_regs(pvt);
928
929 edac_dbg(1, " DramHoleValid: %s\n", dhar_valid(pvt) ? "yes" : "no");
930
Yazen Ghannam78359612019-02-28 15:36:11 +0000931 amd64_info("using x%u syndromes.\n", pvt->ecc_sym_sz);
Yazen Ghannam07ed82e2016-11-28 08:50:21 -0600932}
933
Doug Thompson94be4bf2009-04-27 16:12:00 +0200934/*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500935 * See BKDG, F2x[1,0][5C:40], F2[1,0][6C:60]
Doug Thompson94be4bf2009-04-27 16:12:00 +0200936 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100937static void prep_chip_selects(struct amd64_pvt *pvt)
Doug Thompson94be4bf2009-04-27 16:12:00 +0200938{
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500939 if (pvt->fam == 0xf && pvt->ext_model < K8_REV_F) {
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100940 pvt->csels[0].b_cnt = pvt->csels[1].b_cnt = 8;
941 pvt->csels[0].m_cnt = pvt->csels[1].m_cnt = 8;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100942 } else if (pvt->fam == 0x15 && pvt->model == 0x30) {
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500943 pvt->csels[0].b_cnt = pvt->csels[1].b_cnt = 4;
944 pvt->csels[0].m_cnt = pvt->csels[1].m_cnt = 2;
Borislav Petkov9d858bb2009-09-21 14:35:51 +0200945 } else {
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100946 pvt->csels[0].b_cnt = pvt->csels[1].b_cnt = 8;
947 pvt->csels[0].m_cnt = pvt->csels[1].m_cnt = 4;
Doug Thompson94be4bf2009-04-27 16:12:00 +0200948 }
949}
950
951/*
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100952 * Function 2 Offset F10_DCSB0; read in the DCS Base and DCS Mask registers
Doug Thompson94be4bf2009-04-27 16:12:00 +0200953 */
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200954static void read_dct_base_mask(struct amd64_pvt *pvt)
Doug Thompson94be4bf2009-04-27 16:12:00 +0200955{
Borislav Petkov8de9930a42019-04-25 16:30:34 +0200956 int base_reg0, base_reg1, mask_reg0, mask_reg1, cs;
Doug Thompson94be4bf2009-04-27 16:12:00 +0200957
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100958 prep_chip_selects(pvt);
Doug Thompson94be4bf2009-04-27 16:12:00 +0200959
Borislav Petkov8de9930a42019-04-25 16:30:34 +0200960 if (pvt->umc) {
961 base_reg0 = get_umc_base(0) + UMCCH_BASE_ADDR;
962 base_reg1 = get_umc_base(1) + UMCCH_BASE_ADDR;
963 mask_reg0 = get_umc_base(0) + UMCCH_ADDR_MASK;
964 mask_reg1 = get_umc_base(1) + UMCCH_ADDR_MASK;
965 } else {
966 base_reg0 = DCSB0;
967 base_reg1 = DCSB1;
968 mask_reg0 = DCSM0;
969 mask_reg1 = DCSM1;
970 }
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -0500971
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100972 for_each_chip_select(cs, 0, pvt) {
Borislav Petkov8de9930a42019-04-25 16:30:34 +0200973 int reg0 = base_reg0 + (cs * 4);
974 int reg1 = base_reg1 + (cs * 4);
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100975 u32 *base0 = &pvt->csels[0].csbases[cs];
976 u32 *base1 = &pvt->csels[1].csbases[cs];
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200977
Borislav Petkov8de9930a42019-04-25 16:30:34 +0200978 if (pvt->umc) {
979 if (!amd_smn_read(pvt->mc_node_id, reg0, base0))
980 edac_dbg(0, " DCSB0[%d]=0x%08x reg: 0x%x\n",
981 cs, *base0, reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +0200982
Borislav Petkov8de9930a42019-04-25 16:30:34 +0200983 if (!amd_smn_read(pvt->mc_node_id, reg1, base1))
984 edac_dbg(0, " DCSB1[%d]=0x%08x reg: 0x%x\n",
985 cs, *base1, reg1);
986 } else {
987 if (!amd64_read_dct_pci_cfg(pvt, 0, reg0, base0))
988 edac_dbg(0, " DCSB0[%d]=0x%08x reg: F2x%x\n",
989 cs, *base0, reg0);
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200990
Borislav Petkov8de9930a42019-04-25 16:30:34 +0200991 if (pvt->fam == 0xf)
992 continue;
993
994 if (!amd64_read_dct_pci_cfg(pvt, 1, reg0, base1))
995 edac_dbg(0, " DCSB1[%d]=0x%08x reg: F2x%x\n",
996 cs, *base1, (pvt->fam == 0x10) ? reg1
997 : reg0);
998 }
Doug Thompson94be4bf2009-04-27 16:12:00 +0200999 }
1000
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001001 for_each_chip_select_mask(cs, 0, pvt) {
Borislav Petkov8de9930a42019-04-25 16:30:34 +02001002 int reg0 = mask_reg0 + (cs * 4);
1003 int reg1 = mask_reg1 + (cs * 4);
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001004 u32 *mask0 = &pvt->csels[0].csmasks[cs];
1005 u32 *mask1 = &pvt->csels[1].csmasks[cs];
Borislav Petkovb2b0c602010-10-08 18:32:29 +02001006
Borislav Petkov8de9930a42019-04-25 16:30:34 +02001007 if (pvt->umc) {
1008 if (!amd_smn_read(pvt->mc_node_id, reg0, mask0))
1009 edac_dbg(0, " DCSM0[%d]=0x%08x reg: 0x%x\n",
1010 cs, *mask0, reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +02001011
Borislav Petkov8de9930a42019-04-25 16:30:34 +02001012 if (!amd_smn_read(pvt->mc_node_id, reg1, mask1))
1013 edac_dbg(0, " DCSM1[%d]=0x%08x reg: 0x%x\n",
1014 cs, *mask1, reg1);
1015 } else {
1016 if (!amd64_read_dct_pci_cfg(pvt, 0, reg0, mask0))
1017 edac_dbg(0, " DCSM0[%d]=0x%08x reg: F2x%x\n",
1018 cs, *mask0, reg0);
Borislav Petkovb2b0c602010-10-08 18:32:29 +02001019
Borislav Petkov8de9930a42019-04-25 16:30:34 +02001020 if (pvt->fam == 0xf)
1021 continue;
1022
1023 if (!amd64_read_dct_pci_cfg(pvt, 1, reg0, mask1))
1024 edac_dbg(0, " DCSM1[%d]=0x%08x reg: F2x%x\n",
1025 cs, *mask1, (pvt->fam == 0x10) ? reg1
1026 : reg0);
1027 }
Doug Thompson94be4bf2009-04-27 16:12:00 +02001028 }
1029}
1030
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001031static void determine_memory_type(struct amd64_pvt *pvt)
Doug Thompson94be4bf2009-04-27 16:12:00 +02001032{
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001033 u32 dram_ctrl, dcsm;
Doug Thompson94be4bf2009-04-27 16:12:00 +02001034
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001035 switch (pvt->fam) {
1036 case 0xf:
1037 if (pvt->ext_model >= K8_REV_F)
1038 goto ddr3;
1039
1040 pvt->dram_type = (pvt->dclr0 & BIT(18)) ? MEM_DDR : MEM_RDDR;
1041 return;
1042
1043 case 0x10:
Borislav Petkov6b4c0bd2009-11-12 15:37:57 +01001044 if (pvt->dchr0 & DDR3_MODE)
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001045 goto ddr3;
1046
1047 pvt->dram_type = (pvt->dclr0 & BIT(16)) ? MEM_DDR2 : MEM_RDDR2;
1048 return;
1049
1050 case 0x15:
1051 if (pvt->model < 0x60)
1052 goto ddr3;
1053
1054 /*
1055 * Model 0x60h needs special handling:
1056 *
1057 * We use a Chip Select value of '0' to obtain dcsm.
1058 * Theoretically, it is possible to populate LRDIMMs of different
1059 * 'Rank' value on a DCT. But this is not the common case. So,
1060 * it's reasonable to assume all DIMMs are going to be of same
1061 * 'type' until proven otherwise.
1062 */
1063 amd64_read_dct_pci_cfg(pvt, 0, DRAM_CONTROL, &dram_ctrl);
1064 dcsm = pvt->csels[0].csmasks[0];
1065
1066 if (((dram_ctrl >> 8) & 0x7) == 0x2)
1067 pvt->dram_type = MEM_DDR4;
1068 else if (pvt->dclr0 & BIT(16))
1069 pvt->dram_type = MEM_DDR3;
1070 else if (dcsm & 0x3)
1071 pvt->dram_type = MEM_LRDDR3;
Borislav Petkov6b4c0bd2009-11-12 15:37:57 +01001072 else
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001073 pvt->dram_type = MEM_RDDR3;
1074
1075 return;
1076
1077 case 0x16:
1078 goto ddr3;
1079
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05001080 case 0x17:
Pu Wenc4a3e942018-09-27 16:31:28 +02001081 case 0x18:
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05001082 if ((pvt->umc[0].dimm_cfg | pvt->umc[1].dimm_cfg) & BIT(5))
1083 pvt->dram_type = MEM_LRDDR4;
1084 else if ((pvt->umc[0].dimm_cfg | pvt->umc[1].dimm_cfg) & BIT(4))
1085 pvt->dram_type = MEM_RDDR4;
1086 else
1087 pvt->dram_type = MEM_DDR4;
1088 return;
1089
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001090 default:
1091 WARN(1, KERN_ERR "%s: Family??? 0x%x\n", __func__, pvt->fam);
1092 pvt->dram_type = MEM_EMPTY;
Doug Thompson94be4bf2009-04-27 16:12:00 +02001093 }
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001094 return;
Doug Thompson94be4bf2009-04-27 16:12:00 +02001095
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001096ddr3:
1097 pvt->dram_type = (pvt->dclr0 & BIT(16)) ? MEM_DDR3 : MEM_RDDR3;
Doug Thompson94be4bf2009-04-27 16:12:00 +02001098}
1099
Borislav Petkovcb328502010-12-22 14:28:24 +01001100/* Get the number of DCT channels the memory controller is using. */
Doug Thompsonddff8762009-04-27 16:14:52 +02001101static int k8_early_channel_count(struct amd64_pvt *pvt)
1102{
Borislav Petkovcb328502010-12-22 14:28:24 +01001103 int flag;
Doug Thompsonddff8762009-04-27 16:14:52 +02001104
Borislav Petkov9f56da02010-10-01 19:44:53 +02001105 if (pvt->ext_model >= K8_REV_F)
Doug Thompsonddff8762009-04-27 16:14:52 +02001106 /* RevF (NPT) and later */
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001107 flag = pvt->dclr0 & WIDTH_128;
Borislav Petkov9f56da02010-10-01 19:44:53 +02001108 else
Doug Thompsonddff8762009-04-27 16:14:52 +02001109 /* RevE and earlier */
1110 flag = pvt->dclr0 & REVE_WIDTH_128;
Doug Thompsonddff8762009-04-27 16:14:52 +02001111
1112 /* not used */
1113 pvt->dclr1 = 0;
1114
1115 return (flag) ? 2 : 1;
1116}
1117
Borislav Petkov70046622011-01-10 14:37:27 +01001118/* On F10h and later ErrAddr is MC4_ADDR[47:1] */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001119static u64 get_error_address(struct amd64_pvt *pvt, struct mce *m)
Doug Thompsonddff8762009-04-27 16:14:52 +02001120{
Borislav Petkov2ec591a2015-02-17 10:58:34 +01001121 u16 mce_nid = amd_get_nb_id(m->extcpu);
1122 struct mem_ctl_info *mci;
Borislav Petkov70046622011-01-10 14:37:27 +01001123 u8 start_bit = 1;
1124 u8 end_bit = 47;
Borislav Petkov2ec591a2015-02-17 10:58:34 +01001125 u64 addr;
1126
1127 mci = edac_mc_find(mce_nid);
1128 if (!mci)
1129 return 0;
1130
1131 pvt = mci->pvt_info;
Borislav Petkov70046622011-01-10 14:37:27 +01001132
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001133 if (pvt->fam == 0xf) {
Borislav Petkov70046622011-01-10 14:37:27 +01001134 start_bit = 3;
1135 end_bit = 39;
1136 }
1137
Chen, Gong10ef6b02013-10-18 14:29:07 -07001138 addr = m->addr & GENMASK_ULL(end_bit, start_bit);
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001139
1140 /*
1141 * Erratum 637 workaround
1142 */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001143 if (pvt->fam == 0x15) {
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001144 u64 cc6_base, tmp_addr;
1145 u32 tmp;
Daniel J Blueman8b84c8d2012-11-27 14:32:10 +08001146 u8 intlv_en;
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001147
Chen, Gong10ef6b02013-10-18 14:29:07 -07001148 if ((addr & GENMASK_ULL(47, 24)) >> 24 != 0x00fdf7)
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001149 return addr;
1150
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001151
1152 amd64_read_pci_cfg(pvt->F1, DRAM_LOCAL_NODE_LIM, &tmp);
1153 intlv_en = tmp >> 21 & 0x7;
1154
1155 /* add [47:27] + 3 trailing bits */
Chen, Gong10ef6b02013-10-18 14:29:07 -07001156 cc6_base = (tmp & GENMASK_ULL(20, 0)) << 3;
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001157
1158 /* reverse and add DramIntlvEn */
1159 cc6_base |= intlv_en ^ 0x7;
1160
1161 /* pin at [47:24] */
1162 cc6_base <<= 24;
1163
1164 if (!intlv_en)
Chen, Gong10ef6b02013-10-18 14:29:07 -07001165 return cc6_base | (addr & GENMASK_ULL(23, 0));
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001166
1167 amd64_read_pci_cfg(pvt->F1, DRAM_LOCAL_NODE_BASE, &tmp);
1168
1169 /* faster log2 */
Chen, Gong10ef6b02013-10-18 14:29:07 -07001170 tmp_addr = (addr & GENMASK_ULL(23, 12)) << __fls(intlv_en + 1);
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001171
1172 /* OR DramIntlvSel into bits [14:12] */
Chen, Gong10ef6b02013-10-18 14:29:07 -07001173 tmp_addr |= (tmp & GENMASK_ULL(23, 21)) >> 9;
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001174
1175 /* add remaining [11:0] bits from original MC4_ADDR */
Chen, Gong10ef6b02013-10-18 14:29:07 -07001176 tmp_addr |= addr & GENMASK_ULL(11, 0);
Borislav Petkovc1ae6832011-03-30 15:42:10 +02001177
1178 return cc6_base | tmp_addr;
1179 }
1180
1181 return addr;
Doug Thompsonddff8762009-04-27 16:14:52 +02001182}
1183
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001184static struct pci_dev *pci_get_related_function(unsigned int vendor,
1185 unsigned int device,
1186 struct pci_dev *related)
1187{
1188 struct pci_dev *dev = NULL;
1189
1190 while ((dev = pci_get_device(vendor, device, dev))) {
1191 if (pci_domain_nr(dev->bus) == pci_domain_nr(related->bus) &&
1192 (dev->bus->number == related->bus->number) &&
1193 (PCI_SLOT(dev->devfn) == PCI_SLOT(related->devfn)))
1194 break;
1195 }
1196
1197 return dev;
1198}
1199
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001200static void read_dram_base_limit_regs(struct amd64_pvt *pvt, unsigned range)
Doug Thompsonddff8762009-04-27 16:14:52 +02001201{
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001202 struct amd_northbridge *nb;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001203 struct pci_dev *f1 = NULL;
1204 unsigned int pci_func;
Borislav Petkov71d2a322011-02-21 19:37:24 +01001205 int off = range << 3;
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001206 u32 llim;
Doug Thompsonddff8762009-04-27 16:14:52 +02001207
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001208 amd64_read_pci_cfg(pvt->F1, DRAM_BASE_LO + off, &pvt->ranges[range].base.lo);
1209 amd64_read_pci_cfg(pvt->F1, DRAM_LIMIT_LO + off, &pvt->ranges[range].lim.lo);
Doug Thompsonddff8762009-04-27 16:14:52 +02001210
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001211 if (pvt->fam == 0xf)
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001212 return;
Doug Thompsonddff8762009-04-27 16:14:52 +02001213
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001214 if (!dram_rw(pvt, range))
1215 return;
Doug Thompsonddff8762009-04-27 16:14:52 +02001216
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001217 amd64_read_pci_cfg(pvt->F1, DRAM_BASE_HI + off, &pvt->ranges[range].base.hi);
1218 amd64_read_pci_cfg(pvt->F1, DRAM_LIMIT_HI + off, &pvt->ranges[range].lim.hi);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001219
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001220 /* F15h: factor in CC6 save area by reading dst node's limit reg */
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001221 if (pvt->fam != 0x15)
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001222 return;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001223
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001224 nb = node_to_amd_nb(dram_dst_node(pvt, range));
1225 if (WARN_ON(!nb))
1226 return;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001227
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001228 if (pvt->model == 0x60)
1229 pci_func = PCI_DEVICE_ID_AMD_15H_M60H_NB_F1;
1230 else if (pvt->model == 0x30)
1231 pci_func = PCI_DEVICE_ID_AMD_15H_M30H_NB_F1;
1232 else
1233 pci_func = PCI_DEVICE_ID_AMD_15H_NB_F1;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001234
1235 f1 = pci_get_related_function(nb->misc->vendor, pci_func, nb->misc);
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001236 if (WARN_ON(!f1))
1237 return;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001238
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001239 amd64_read_pci_cfg(f1, DRAM_LOCAL_NODE_LIM, &llim);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001240
Chen, Gong10ef6b02013-10-18 14:29:07 -07001241 pvt->ranges[range].lim.lo &= GENMASK_ULL(15, 0);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001242
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001243 /* {[39:27],111b} */
1244 pvt->ranges[range].lim.lo |= ((llim & 0x1fff) << 3 | 0x7) << 16;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001245
Chen, Gong10ef6b02013-10-18 14:29:07 -07001246 pvt->ranges[range].lim.hi &= GENMASK_ULL(7, 0);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001247
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001248 /* [47:40] */
1249 pvt->ranges[range].lim.hi |= llim >> 13;
1250
1251 pci_dev_put(f1);
Doug Thompsonddff8762009-04-27 16:14:52 +02001252}
1253
Borislav Petkovf192c7b2011-01-10 14:24:32 +01001254static void k8_map_sysaddr_to_csrow(struct mem_ctl_info *mci, u64 sys_addr,
Borislav Petkov33ca0642012-08-30 18:01:36 +02001255 struct err_info *err)
Doug Thompsonddff8762009-04-27 16:14:52 +02001256{
Borislav Petkovf192c7b2011-01-10 14:24:32 +01001257 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompsonddff8762009-04-27 16:14:52 +02001258
Borislav Petkov33ca0642012-08-30 18:01:36 +02001259 error_address_to_page_and_offset(sys_addr, err);
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001260
1261 /*
1262 * Find out which node the error address belongs to. This may be
1263 * different from the node that detected the error.
1264 */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001265 err->src_mci = find_mc_by_sys_addr(mci, sys_addr);
1266 if (!err->src_mci) {
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001267 amd64_mc_err(mci, "failed to map error addr 0x%lx to a node\n",
1268 (unsigned long)sys_addr);
Borislav Petkov33ca0642012-08-30 18:01:36 +02001269 err->err_code = ERR_NODE;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001270 return;
1271 }
1272
1273 /* Now map the sys_addr to a CSROW */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001274 err->csrow = sys_addr_to_csrow(err->src_mci, sys_addr);
1275 if (err->csrow < 0) {
1276 err->err_code = ERR_CSROW;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001277 return;
1278 }
1279
Doug Thompsonddff8762009-04-27 16:14:52 +02001280 /* CHIPKILL enabled */
Borislav Petkovf192c7b2011-01-10 14:24:32 +01001281 if (pvt->nbcfg & NBCFG_CHIPKILL) {
Borislav Petkov33ca0642012-08-30 18:01:36 +02001282 err->channel = get_channel_from_ecc_syndrome(mci, err->syndrome);
1283 if (err->channel < 0) {
Doug Thompsonddff8762009-04-27 16:14:52 +02001284 /*
1285 * Syndrome didn't map, so we don't know which of the
1286 * 2 DIMMs is in error. So we need to ID 'both' of them
1287 * as suspect.
1288 */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001289 amd64_mc_warn(err->src_mci, "unknown syndrome 0x%04x - "
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001290 "possible error reporting race\n",
Borislav Petkov33ca0642012-08-30 18:01:36 +02001291 err->syndrome);
1292 err->err_code = ERR_CHANNEL;
Doug Thompsonddff8762009-04-27 16:14:52 +02001293 return;
1294 }
1295 } else {
1296 /*
1297 * non-chipkill ecc mode
1298 *
1299 * The k8 documentation is unclear about how to determine the
1300 * channel number when using non-chipkill memory. This method
1301 * was obtained from email communication with someone at AMD.
1302 * (Wish the email was placed in this comment - norsk)
1303 */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001304 err->channel = ((sys_addr & BIT(3)) != 0);
Doug Thompsonddff8762009-04-27 16:14:52 +02001305 }
Doug Thompsonddff8762009-04-27 16:14:52 +02001306}
1307
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001308static int ddr2_cs_size(unsigned i, bool dct_width)
Doug Thompsonddff8762009-04-27 16:14:52 +02001309{
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001310 unsigned shift = 0;
Doug Thompsonddff8762009-04-27 16:14:52 +02001311
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001312 if (i <= 2)
1313 shift = i;
1314 else if (!(i & 0x1))
1315 shift = i >> 1;
Borislav Petkov1433eb92009-10-21 13:44:36 +02001316 else
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001317 shift = (i + 1) >> 1;
Doug Thompsonddff8762009-04-27 16:14:52 +02001318
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001319 return 128 << (shift + !!dct_width);
1320}
1321
1322static int k8_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001323 unsigned cs_mode, int cs_mask_nr)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001324{
1325 u32 dclr = dct ? pvt->dclr1 : pvt->dclr0;
1326
1327 if (pvt->ext_model >= K8_REV_F) {
1328 WARN_ON(cs_mode > 11);
1329 return ddr2_cs_size(cs_mode, dclr & WIDTH_128);
1330 }
1331 else if (pvt->ext_model >= K8_REV_D) {
Borislav Petkov11b0a312011-11-09 21:28:43 +01001332 unsigned diff;
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001333 WARN_ON(cs_mode > 10);
1334
Borislav Petkov11b0a312011-11-09 21:28:43 +01001335 /*
1336 * the below calculation, besides trying to win an obfuscated C
1337 * contest, maps cs_mode values to DIMM chip select sizes. The
1338 * mappings are:
1339 *
1340 * cs_mode CS size (mb)
1341 * ======= ============
1342 * 0 32
1343 * 1 64
1344 * 2 128
1345 * 3 128
1346 * 4 256
1347 * 5 512
1348 * 6 256
1349 * 7 512
1350 * 8 1024
1351 * 9 1024
1352 * 10 2048
1353 *
1354 * Basically, it calculates a value with which to shift the
1355 * smallest CS size of 32MB.
1356 *
1357 * ddr[23]_cs_size have a similar purpose.
1358 */
1359 diff = cs_mode/3 + (unsigned)(cs_mode > 5);
1360
1361 return 32 << (cs_mode - diff);
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001362 }
1363 else {
1364 WARN_ON(cs_mode > 6);
1365 return 32 << cs_mode;
1366 }
Doug Thompsonddff8762009-04-27 16:14:52 +02001367}
1368
Doug Thompson1afd3c92009-04-27 16:16:50 +02001369/*
1370 * Get the number of DCT channels in use.
1371 *
1372 * Return:
1373 * number of Memory Channels in operation
1374 * Pass back:
1375 * contents of the DCL0_LOW register
1376 */
Borislav Petkov7d20d142011-01-07 17:58:04 +01001377static int f1x_early_channel_count(struct amd64_pvt *pvt)
Doug Thompson1afd3c92009-04-27 16:16:50 +02001378{
Borislav Petkov6ba5dcd2009-10-13 19:26:55 +02001379 int i, j, channels = 0;
Doug Thompsonddff8762009-04-27 16:14:52 +02001380
Borislav Petkov7d20d142011-01-07 17:58:04 +01001381 /* On F10h, if we are in 128 bit mode, then we are using 2 channels */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001382 if (pvt->fam == 0x10 && (pvt->dclr0 & WIDTH_128))
Borislav Petkov7d20d142011-01-07 17:58:04 +01001383 return 2;
Doug Thompson1afd3c92009-04-27 16:16:50 +02001384
1385 /*
Borislav Petkovd16149e2009-10-16 19:55:49 +02001386 * Need to check if in unganged mode: In such, there are 2 channels,
1387 * but they are not in 128 bit mode and thus the above 'dclr0' status
1388 * bit will be OFF.
Doug Thompson1afd3c92009-04-27 16:16:50 +02001389 *
1390 * Need to check DCT0[0] and DCT1[0] to see if only one of them has
1391 * their CSEnable bit on. If so, then SINGLE DIMM case.
1392 */
Joe Perches956b9ba12012-04-29 17:08:39 -03001393 edac_dbg(0, "Data width is not 128 bits - need more decoding\n");
Doug Thompson1afd3c92009-04-27 16:16:50 +02001394
1395 /*
1396 * Check DRAM Bank Address Mapping values for each DIMM to see if there
1397 * is more than just one DIMM present in unganged mode. Need to check
1398 * both controllers since DIMMs can be placed in either one.
1399 */
Borislav Petkov525a1b22010-12-21 15:53:27 +01001400 for (i = 0; i < 2; i++) {
1401 u32 dbam = (i ? pvt->dbam1 : pvt->dbam0);
Doug Thompson1afd3c92009-04-27 16:16:50 +02001402
Wan Wei57a30852009-08-07 17:04:49 +02001403 for (j = 0; j < 4; j++) {
1404 if (DBAM_DIMM(j, dbam) > 0) {
1405 channels++;
1406 break;
1407 }
1408 }
Doug Thompson1afd3c92009-04-27 16:16:50 +02001409 }
1410
Borislav Petkovd16149e2009-10-16 19:55:49 +02001411 if (channels > 2)
1412 channels = 2;
1413
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02001414 amd64_info("MCT channel count: %d\n", channels);
Doug Thompson1afd3c92009-04-27 16:16:50 +02001415
1416 return channels;
Doug Thompson1afd3c92009-04-27 16:16:50 +02001417}
1418
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001419static int f17_early_channel_count(struct amd64_pvt *pvt)
1420{
1421 int i, channels = 0;
1422
1423 /* SDP Control bit 31 (SdpInit) is clear for unused UMC channels */
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00001424 for_each_umc(i)
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001425 channels += !!(pvt->umc[i].sdp_ctrl & UMC_SDP_INIT);
1426
1427 amd64_info("MCT channel count: %d\n", channels);
1428
1429 return channels;
1430}
1431
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001432static int ddr3_cs_size(unsigned i, bool dct_width)
Doug Thompson1afd3c92009-04-27 16:16:50 +02001433{
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001434 unsigned shift = 0;
1435 int cs_size = 0;
1436
1437 if (i == 0 || i == 3 || i == 4)
1438 cs_size = -1;
1439 else if (i <= 2)
1440 shift = i;
1441 else if (i == 12)
1442 shift = 7;
1443 else if (!(i & 0x1))
1444 shift = i >> 1;
1445 else
1446 shift = (i + 1) >> 1;
1447
1448 if (cs_size != -1)
1449 cs_size = (128 * (1 << !!dct_width)) << shift;
1450
1451 return cs_size;
1452}
1453
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001454static int ddr3_lrdimm_cs_size(unsigned i, unsigned rank_multiply)
1455{
1456 unsigned shift = 0;
1457 int cs_size = 0;
1458
1459 if (i < 4 || i == 6)
1460 cs_size = -1;
1461 else if (i == 12)
1462 shift = 7;
1463 else if (!(i & 0x1))
1464 shift = i >> 1;
1465 else
1466 shift = (i + 1) >> 1;
1467
1468 if (cs_size != -1)
1469 cs_size = rank_multiply * (128 << shift);
1470
1471 return cs_size;
1472}
1473
1474static int ddr4_cs_size(unsigned i)
1475{
1476 int cs_size = 0;
1477
1478 if (i == 0)
1479 cs_size = -1;
1480 else if (i == 1)
1481 cs_size = 1024;
1482 else
1483 /* Min cs_size = 1G */
1484 cs_size = 1024 * (1 << (i >> 1));
1485
1486 return cs_size;
1487}
1488
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001489static int f10_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001490 unsigned cs_mode, int cs_mask_nr)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001491{
1492 u32 dclr = dct ? pvt->dclr1 : pvt->dclr0;
1493
1494 WARN_ON(cs_mode > 11);
Borislav Petkov1433eb92009-10-21 13:44:36 +02001495
1496 if (pvt->dchr0 & DDR3_MODE || pvt->dchr1 & DDR3_MODE)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001497 return ddr3_cs_size(cs_mode, dclr & WIDTH_128);
Borislav Petkov1433eb92009-10-21 13:44:36 +02001498 else
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001499 return ddr2_cs_size(cs_mode, dclr & WIDTH_128);
1500}
Borislav Petkov1433eb92009-10-21 13:44:36 +02001501
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001502/*
1503 * F15h supports only 64bit DCT interfaces
1504 */
1505static int f15_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001506 unsigned cs_mode, int cs_mask_nr)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001507{
1508 WARN_ON(cs_mode > 12);
1509
1510 return ddr3_cs_size(cs_mode, false);
Doug Thompson1afd3c92009-04-27 16:16:50 +02001511}
1512
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001513/* F15h M60h supports DDR4 mapping as well.. */
1514static int f15_m60h_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
1515 unsigned cs_mode, int cs_mask_nr)
1516{
1517 int cs_size;
1518 u32 dcsm = pvt->csels[dct].csmasks[cs_mask_nr];
1519
1520 WARN_ON(cs_mode > 12);
1521
1522 if (pvt->dram_type == MEM_DDR4) {
1523 if (cs_mode > 9)
1524 return -1;
1525
1526 cs_size = ddr4_cs_size(cs_mode);
1527 } else if (pvt->dram_type == MEM_LRDDR3) {
1528 unsigned rank_multiply = dcsm & 0xf;
1529
1530 if (rank_multiply == 3)
1531 rank_multiply = 4;
1532 cs_size = ddr3_lrdimm_cs_size(cs_mode, rank_multiply);
1533 } else {
1534 /* Minimum cs size is 512mb for F15hM60h*/
1535 if (cs_mode == 0x1)
1536 return -1;
1537
1538 cs_size = ddr3_cs_size(cs_mode, false);
1539 }
1540
1541 return cs_size;
1542}
1543
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001544/*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001545 * F16h and F15h model 30h have only limited cs_modes.
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001546 */
1547static int f16_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001548 unsigned cs_mode, int cs_mask_nr)
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001549{
1550 WARN_ON(cs_mode > 12);
1551
1552 if (cs_mode == 6 || cs_mode == 8 ||
1553 cs_mode == 9 || cs_mode == 12)
1554 return -1;
1555 else
1556 return ddr3_cs_size(cs_mode, false);
1557}
1558
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05001559static int f17_base_addr_to_cs_size(struct amd64_pvt *pvt, u8 umc,
1560 unsigned int cs_mode, int csrow_nr)
1561{
1562 u32 base_addr = pvt->csels[umc].csbases[csrow_nr];
1563
1564 /* Each mask is used for every two base addresses. */
1565 u32 addr_mask = pvt->csels[umc].csmasks[csrow_nr >> 1];
1566
1567 /* Register [31:1] = Address [39:9]. Size is in kBs here. */
1568 u32 size = ((addr_mask >> 1) - (base_addr >> 1) + 1) >> 1;
1569
1570 edac_dbg(1, "BaseAddr: 0x%x, AddrMask: 0x%x\n", base_addr, addr_mask);
1571
1572 /* Return size in MBs. */
1573 return size >> 10;
1574}
1575
Borislav Petkov5a5d2372011-01-17 17:52:57 +01001576static void read_dram_ctl_register(struct amd64_pvt *pvt)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001577{
Doug Thompson6163b5d2009-04-27 16:20:17 +02001578
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001579 if (pvt->fam == 0xf)
Borislav Petkov5a5d2372011-01-17 17:52:57 +01001580 return;
1581
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001582 if (!amd64_read_pci_cfg(pvt->F2, DCT_SEL_LO, &pvt->dct_sel_lo)) {
Joe Perches956b9ba12012-04-29 17:08:39 -03001583 edac_dbg(0, "F2x110 (DCTSelLow): 0x%08x, High range addrs at: 0x%x\n",
1584 pvt->dct_sel_lo, dct_sel_baseaddr(pvt));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001585
Joe Perches956b9ba12012-04-29 17:08:39 -03001586 edac_dbg(0, " DCTs operate in %s mode\n",
1587 (dct_ganging_enabled(pvt) ? "ganged" : "unganged"));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001588
Borislav Petkov72381bd2009-10-09 19:14:43 +02001589 if (!dct_ganging_enabled(pvt))
Joe Perches956b9ba12012-04-29 17:08:39 -03001590 edac_dbg(0, " Address range split per DCT: %s\n",
1591 (dct_high_range_enabled(pvt) ? "yes" : "no"));
Borislav Petkov72381bd2009-10-09 19:14:43 +02001592
Joe Perches956b9ba12012-04-29 17:08:39 -03001593 edac_dbg(0, " data interleave for ECC: %s, DRAM cleared since last warm reset: %s\n",
1594 (dct_data_intlv_enabled(pvt) ? "enabled" : "disabled"),
1595 (dct_memory_cleared(pvt) ? "yes" : "no"));
Borislav Petkov72381bd2009-10-09 19:14:43 +02001596
Joe Perches956b9ba12012-04-29 17:08:39 -03001597 edac_dbg(0, " channel interleave: %s, "
1598 "interleave bits selector: 0x%x\n",
1599 (dct_interleave_enabled(pvt) ? "enabled" : "disabled"),
1600 dct_sel_interleave_addr(pvt));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001601 }
1602
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001603 amd64_read_pci_cfg(pvt->F2, DCT_SEL_HI, &pvt->dct_sel_hi);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001604}
1605
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001606/*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001607 * Determine channel (DCT) based on the interleaving mode (see F15h M30h BKDG,
1608 * 2.10.12 Memory Interleaving Modes).
1609 */
1610static u8 f15_m30h_determine_channel(struct amd64_pvt *pvt, u64 sys_addr,
1611 u8 intlv_en, int num_dcts_intlv,
1612 u32 dct_sel)
1613{
1614 u8 channel = 0;
1615 u8 select;
1616
1617 if (!(intlv_en))
1618 return (u8)(dct_sel);
1619
1620 if (num_dcts_intlv == 2) {
1621 select = (sys_addr >> 8) & 0x3;
1622 channel = select ? 0x3 : 0;
Aravind Gopalakrishnan9d0e8d82014-01-21 15:03:36 -06001623 } else if (num_dcts_intlv == 4) {
1624 u8 intlv_addr = dct_sel_interleave_addr(pvt);
1625 switch (intlv_addr) {
1626 case 0x4:
1627 channel = (sys_addr >> 8) & 0x3;
1628 break;
1629 case 0x5:
1630 channel = (sys_addr >> 9) & 0x3;
1631 break;
1632 }
1633 }
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001634 return channel;
1635}
1636
1637/*
Borislav Petkov229a7a12010-12-09 18:57:54 +01001638 * Determine channel (DCT) based on the interleaving mode: F10h BKDG, 2.8.9 Memory
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001639 * Interleaving Modes.
1640 */
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001641static u8 f1x_determine_channel(struct amd64_pvt *pvt, u64 sys_addr,
Borislav Petkov229a7a12010-12-09 18:57:54 +01001642 bool hi_range_sel, u8 intlv_en)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001643{
Borislav Petkov151fa712011-02-21 19:33:10 +01001644 u8 dct_sel_high = (pvt->dct_sel_lo >> 1) & 1;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001645
1646 if (dct_ganging_enabled(pvt))
Borislav Petkov229a7a12010-12-09 18:57:54 +01001647 return 0;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001648
Borislav Petkov229a7a12010-12-09 18:57:54 +01001649 if (hi_range_sel)
1650 return dct_sel_high;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001651
Borislav Petkov229a7a12010-12-09 18:57:54 +01001652 /*
1653 * see F2x110[DctSelIntLvAddr] - channel interleave mode
1654 */
1655 if (dct_interleave_enabled(pvt)) {
1656 u8 intlv_addr = dct_sel_interleave_addr(pvt);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001657
Borislav Petkov229a7a12010-12-09 18:57:54 +01001658 /* return DCT select function: 0=DCT0, 1=DCT1 */
1659 if (!intlv_addr)
1660 return sys_addr >> 6 & 1;
1661
1662 if (intlv_addr & 0x2) {
1663 u8 shift = intlv_addr & 0x1 ? 9 : 6;
Yazen Ghannamdc0a50a82016-08-03 10:59:15 -04001664 u32 temp = hweight_long((u32) ((sys_addr >> 16) & 0x1F)) & 1;
Borislav Petkov229a7a12010-12-09 18:57:54 +01001665
1666 return ((sys_addr >> shift) & 1) ^ temp;
1667 }
1668
Yazen Ghannamdc0a50a82016-08-03 10:59:15 -04001669 if (intlv_addr & 0x4) {
1670 u8 shift = intlv_addr & 0x1 ? 9 : 8;
1671
1672 return (sys_addr >> shift) & 1;
1673 }
1674
Borislav Petkov229a7a12010-12-09 18:57:54 +01001675 return (sys_addr >> (12 + hweight8(intlv_en))) & 1;
1676 }
1677
1678 if (dct_high_range_enabled(pvt))
1679 return ~dct_sel_high & 1;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001680
1681 return 0;
1682}
1683
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001684/* Convert the sys_addr to the normalized DCT address */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08001685static u64 f1x_get_norm_dct_addr(struct amd64_pvt *pvt, u8 range,
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001686 u64 sys_addr, bool hi_rng,
1687 u32 dct_sel_base_addr)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001688{
1689 u64 chan_off;
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001690 u64 dram_base = get_dram_base(pvt, range);
1691 u64 hole_off = f10_dhar_offset(pvt);
Dan Carpenter6f3508f2016-01-20 12:54:51 +03001692 u64 dct_sel_base_off = (u64)(pvt->dct_sel_hi & 0xFFFFFC00) << 16;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001693
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001694 if (hi_rng) {
1695 /*
1696 * if
1697 * base address of high range is below 4Gb
1698 * (bits [47:27] at [31:11])
1699 * DRAM address space on this DCT is hoisted above 4Gb &&
1700 * sys_addr > 4Gb
1701 *
1702 * remove hole offset from sys_addr
1703 * else
1704 * remove high range offset from sys_addr
1705 */
1706 if ((!(dct_sel_base_addr >> 16) ||
1707 dct_sel_base_addr < dhar_base(pvt)) &&
Borislav Petkov972ea172011-02-21 19:43:02 +01001708 dhar_valid(pvt) &&
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001709 (sys_addr >= BIT_64(32)))
Borislav Petkovbc21fa52010-11-11 17:29:13 +01001710 chan_off = hole_off;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001711 else
1712 chan_off = dct_sel_base_off;
1713 } else {
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001714 /*
1715 * if
1716 * we have a valid hole &&
1717 * sys_addr > 4Gb
1718 *
1719 * remove hole
1720 * else
1721 * remove dram base to normalize to DCT address
1722 */
Borislav Petkov972ea172011-02-21 19:43:02 +01001723 if (dhar_valid(pvt) && (sys_addr >= BIT_64(32)))
Borislav Petkovbc21fa52010-11-11 17:29:13 +01001724 chan_off = hole_off;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001725 else
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001726 chan_off = dram_base;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001727 }
1728
Chen, Gong10ef6b02013-10-18 14:29:07 -07001729 return (sys_addr & GENMASK_ULL(47,6)) - (chan_off & GENMASK_ULL(47,23));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001730}
1731
Doug Thompson6163b5d2009-04-27 16:20:17 +02001732/*
1733 * checks if the csrow passed in is marked as SPARED, if so returns the new
1734 * spare row
1735 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001736static int f10_process_possible_spare(struct amd64_pvt *pvt, u8 dct, int csrow)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001737{
Borislav Petkov614ec9d2011-01-13 18:02:22 +01001738 int tmp_cs;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001739
Borislav Petkov614ec9d2011-01-13 18:02:22 +01001740 if (online_spare_swap_done(pvt, dct) &&
1741 csrow == online_spare_bad_dramcs(pvt, dct)) {
1742
1743 for_each_chip_select(tmp_cs, dct, pvt) {
1744 if (chip_select_base(tmp_cs, dct, pvt) & 0x2) {
1745 csrow = tmp_cs;
1746 break;
1747 }
1748 }
Doug Thompson6163b5d2009-04-27 16:20:17 +02001749 }
1750 return csrow;
1751}
1752
1753/*
1754 * Iterate over the DRAM DCT "base" and "mask" registers looking for a
1755 * SystemAddr match on the specified 'ChannelSelect' and 'NodeID'
1756 *
1757 * Return:
1758 * -EINVAL: NOT FOUND
1759 * 0..csrow = Chip-Select Row
1760 */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08001761static int f1x_lookup_addr_in_dct(u64 in_addr, u8 nid, u8 dct)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001762{
1763 struct mem_ctl_info *mci;
1764 struct amd64_pvt *pvt;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001765 u64 cs_base, cs_mask;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001766 int cs_found = -EINVAL;
1767 int csrow;
1768
Borislav Petkov2ec591a2015-02-17 10:58:34 +01001769 mci = edac_mc_find(nid);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001770 if (!mci)
1771 return cs_found;
1772
1773 pvt = mci->pvt_info;
1774
Joe Perches956b9ba12012-04-29 17:08:39 -03001775 edac_dbg(1, "input addr: 0x%llx, DCT: %d\n", in_addr, dct);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001776
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001777 for_each_chip_select(csrow, dct, pvt) {
1778 if (!csrow_enabled(csrow, dct, pvt))
Doug Thompson6163b5d2009-04-27 16:20:17 +02001779 continue;
1780
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001781 get_cs_base_and_mask(pvt, csrow, dct, &cs_base, &cs_mask);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001782
Joe Perches956b9ba12012-04-29 17:08:39 -03001783 edac_dbg(1, " CSROW=%d CSBase=0x%llx CSMask=0x%llx\n",
1784 csrow, cs_base, cs_mask);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001785
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001786 cs_mask = ~cs_mask;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001787
Joe Perches956b9ba12012-04-29 17:08:39 -03001788 edac_dbg(1, " (InputAddr & ~CSMask)=0x%llx (CSBase & ~CSMask)=0x%llx\n",
1789 (in_addr & cs_mask), (cs_base & cs_mask));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001790
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001791 if ((in_addr & cs_mask) == (cs_base & cs_mask)) {
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001792 if (pvt->fam == 0x15 && pvt->model >= 0x30) {
1793 cs_found = csrow;
1794 break;
1795 }
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001796 cs_found = f10_process_possible_spare(pvt, dct, csrow);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001797
Joe Perches956b9ba12012-04-29 17:08:39 -03001798 edac_dbg(1, " MATCH csrow=%d\n", cs_found);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001799 break;
1800 }
1801 }
1802 return cs_found;
1803}
1804
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001805/*
1806 * See F2x10C. Non-interleaved graphics framebuffer memory under the 16G is
1807 * swapped with a region located at the bottom of memory so that the GPU can use
1808 * the interleaved region and thus two channels.
1809 */
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001810static u64 f1x_swap_interleaved_region(struct amd64_pvt *pvt, u64 sys_addr)
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001811{
1812 u32 swap_reg, swap_base, swap_limit, rgn_size, tmp_addr;
1813
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001814 if (pvt->fam == 0x10) {
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001815 /* only revC3 and revE have that feature */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001816 if (pvt->model < 4 || (pvt->model < 0xa && pvt->stepping < 3))
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001817 return sys_addr;
1818 }
1819
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001820 amd64_read_pci_cfg(pvt->F2, SWAP_INTLV_REG, &swap_reg);
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001821
1822 if (!(swap_reg & 0x1))
1823 return sys_addr;
1824
1825 swap_base = (swap_reg >> 3) & 0x7f;
1826 swap_limit = (swap_reg >> 11) & 0x7f;
1827 rgn_size = (swap_reg >> 20) & 0x7f;
1828 tmp_addr = sys_addr >> 27;
1829
1830 if (!(sys_addr >> 34) &&
1831 (((tmp_addr >= swap_base) &&
1832 (tmp_addr <= swap_limit)) ||
1833 (tmp_addr < rgn_size)))
1834 return sys_addr ^ (u64)swap_base << 27;
1835
1836 return sys_addr;
1837}
1838
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001839/* For a given @dram_range, check if @sys_addr falls within it. */
Borislav Petkove761359a2011-02-21 19:49:01 +01001840static int f1x_match_to_this_node(struct amd64_pvt *pvt, unsigned range,
Borislav Petkov33ca0642012-08-30 18:01:36 +02001841 u64 sys_addr, int *chan_sel)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001842{
Borislav Petkov229a7a12010-12-09 18:57:54 +01001843 int cs_found = -EINVAL;
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001844 u64 chan_addr;
Borislav Petkov5d4b58e2011-01-13 16:01:13 +01001845 u32 dct_sel_base;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001846 u8 channel;
Borislav Petkov229a7a12010-12-09 18:57:54 +01001847 bool high_range = false;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001848
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001849 u8 node_id = dram_dst_node(pvt, range);
Borislav Petkov229a7a12010-12-09 18:57:54 +01001850 u8 intlv_en = dram_intlv_en(pvt, range);
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001851 u32 intlv_sel = dram_intlv_sel(pvt, range);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001852
Joe Perches956b9ba12012-04-29 17:08:39 -03001853 edac_dbg(1, "(range %d) SystemAddr= 0x%llx Limit=0x%llx\n",
1854 range, sys_addr, get_dram_limit(pvt, range));
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001855
Borislav Petkov355fba62011-01-17 13:03:26 +01001856 if (dhar_valid(pvt) &&
1857 dhar_base(pvt) <= sys_addr &&
1858 sys_addr < BIT_64(32)) {
1859 amd64_warn("Huh? Address is in the MMIO hole: 0x%016llx\n",
1860 sys_addr);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001861 return -EINVAL;
Borislav Petkov355fba62011-01-17 13:03:26 +01001862 }
1863
Borislav Petkovf030ddf2011-04-08 15:05:21 +02001864 if (intlv_en && (intlv_sel != ((sys_addr >> 12) & intlv_en)))
Borislav Petkov355fba62011-01-17 13:03:26 +01001865 return -EINVAL;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001866
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001867 sys_addr = f1x_swap_interleaved_region(pvt, sys_addr);
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001868
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001869 dct_sel_base = dct_sel_baseaddr(pvt);
1870
1871 /*
1872 * check whether addresses >= DctSelBaseAddr[47:27] are to be used to
1873 * select between DCT0 and DCT1.
1874 */
1875 if (dct_high_range_enabled(pvt) &&
1876 !dct_ganging_enabled(pvt) &&
1877 ((sys_addr >> 27) >= (dct_sel_base >> 11)))
Borislav Petkov229a7a12010-12-09 18:57:54 +01001878 high_range = true;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001879
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001880 channel = f1x_determine_channel(pvt, sys_addr, high_range, intlv_en);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001881
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001882 chan_addr = f1x_get_norm_dct_addr(pvt, range, sys_addr,
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001883 high_range, dct_sel_base);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001884
Borislav Petkove2f79db2011-01-13 14:57:34 +01001885 /* Remove node interleaving, see F1x120 */
1886 if (intlv_en)
1887 chan_addr = ((chan_addr >> (12 + hweight8(intlv_en))) << 12) |
1888 (chan_addr & 0xfff);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001889
Borislav Petkov5d4b58e2011-01-13 16:01:13 +01001890 /* remove channel interleave */
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001891 if (dct_interleave_enabled(pvt) &&
1892 !dct_high_range_enabled(pvt) &&
1893 !dct_ganging_enabled(pvt)) {
Borislav Petkov5d4b58e2011-01-13 16:01:13 +01001894
1895 if (dct_sel_interleave_addr(pvt) != 1) {
1896 if (dct_sel_interleave_addr(pvt) == 0x3)
1897 /* hash 9 */
1898 chan_addr = ((chan_addr >> 10) << 9) |
1899 (chan_addr & 0x1ff);
1900 else
1901 /* A[6] or hash 6 */
1902 chan_addr = ((chan_addr >> 7) << 6) |
1903 (chan_addr & 0x3f);
1904 } else
1905 /* A[12] */
1906 chan_addr = ((chan_addr >> 13) << 12) |
1907 (chan_addr & 0xfff);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001908 }
1909
Joe Perches956b9ba12012-04-29 17:08:39 -03001910 edac_dbg(1, " Normalized DCT addr: 0x%llx\n", chan_addr);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001911
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001912 cs_found = f1x_lookup_addr_in_dct(chan_addr, node_id, channel);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001913
Borislav Petkov33ca0642012-08-30 18:01:36 +02001914 if (cs_found >= 0)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001915 *chan_sel = channel;
Borislav Petkov33ca0642012-08-30 18:01:36 +02001916
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001917 return cs_found;
1918}
1919
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001920static int f15_m30h_match_to_this_node(struct amd64_pvt *pvt, unsigned range,
1921 u64 sys_addr, int *chan_sel)
1922{
1923 int cs_found = -EINVAL;
1924 int num_dcts_intlv = 0;
1925 u64 chan_addr, chan_offset;
1926 u64 dct_base, dct_limit;
1927 u32 dct_cont_base_reg, dct_cont_limit_reg, tmp;
1928 u8 channel, alias_channel, leg_mmio_hole, dct_sel, dct_offset_en;
1929
1930 u64 dhar_offset = f10_dhar_offset(pvt);
1931 u8 intlv_addr = dct_sel_interleave_addr(pvt);
1932 u8 node_id = dram_dst_node(pvt, range);
1933 u8 intlv_en = dram_intlv_en(pvt, range);
1934
1935 amd64_read_pci_cfg(pvt->F1, DRAM_CONT_BASE, &dct_cont_base_reg);
1936 amd64_read_pci_cfg(pvt->F1, DRAM_CONT_LIMIT, &dct_cont_limit_reg);
1937
1938 dct_offset_en = (u8) ((dct_cont_base_reg >> 3) & BIT(0));
1939 dct_sel = (u8) ((dct_cont_base_reg >> 4) & 0x7);
1940
1941 edac_dbg(1, "(range %d) SystemAddr= 0x%llx Limit=0x%llx\n",
1942 range, sys_addr, get_dram_limit(pvt, range));
1943
1944 if (!(get_dram_base(pvt, range) <= sys_addr) &&
1945 !(get_dram_limit(pvt, range) >= sys_addr))
1946 return -EINVAL;
1947
1948 if (dhar_valid(pvt) &&
1949 dhar_base(pvt) <= sys_addr &&
1950 sys_addr < BIT_64(32)) {
1951 amd64_warn("Huh? Address is in the MMIO hole: 0x%016llx\n",
1952 sys_addr);
1953 return -EINVAL;
1954 }
1955
1956 /* Verify sys_addr is within DCT Range. */
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05001957 dct_base = (u64) dct_sel_baseaddr(pvt);
1958 dct_limit = (dct_cont_limit_reg >> 11) & 0x1FFF;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001959
1960 if (!(dct_cont_base_reg & BIT(0)) &&
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05001961 !(dct_base <= (sys_addr >> 27) &&
1962 dct_limit >= (sys_addr >> 27)))
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001963 return -EINVAL;
1964
1965 /* Verify number of dct's that participate in channel interleaving. */
1966 num_dcts_intlv = (int) hweight8(intlv_en);
1967
1968 if (!(num_dcts_intlv % 2 == 0) || (num_dcts_intlv > 4))
1969 return -EINVAL;
1970
Yazen Ghannamdc0a50a82016-08-03 10:59:15 -04001971 if (pvt->model >= 0x60)
1972 channel = f1x_determine_channel(pvt, sys_addr, false, intlv_en);
1973 else
1974 channel = f15_m30h_determine_channel(pvt, sys_addr, intlv_en,
1975 num_dcts_intlv, dct_sel);
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001976
1977 /* Verify we stay within the MAX number of channels allowed */
Aravind Gopalakrishnan7f3f5242013-12-04 11:40:11 -06001978 if (channel > 3)
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001979 return -EINVAL;
1980
1981 leg_mmio_hole = (u8) (dct_cont_base_reg >> 1 & BIT(0));
1982
1983 /* Get normalized DCT addr */
1984 if (leg_mmio_hole && (sys_addr >= BIT_64(32)))
1985 chan_offset = dhar_offset;
1986 else
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05001987 chan_offset = dct_base << 27;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001988
1989 chan_addr = sys_addr - chan_offset;
1990
1991 /* remove channel interleave */
1992 if (num_dcts_intlv == 2) {
1993 if (intlv_addr == 0x4)
1994 chan_addr = ((chan_addr >> 9) << 8) |
1995 (chan_addr & 0xff);
1996 else if (intlv_addr == 0x5)
1997 chan_addr = ((chan_addr >> 10) << 9) |
1998 (chan_addr & 0x1ff);
1999 else
2000 return -EINVAL;
2001
2002 } else if (num_dcts_intlv == 4) {
2003 if (intlv_addr == 0x4)
2004 chan_addr = ((chan_addr >> 10) << 8) |
2005 (chan_addr & 0xff);
2006 else if (intlv_addr == 0x5)
2007 chan_addr = ((chan_addr >> 11) << 9) |
2008 (chan_addr & 0x1ff);
2009 else
2010 return -EINVAL;
2011 }
2012
2013 if (dct_offset_en) {
2014 amd64_read_pci_cfg(pvt->F1,
2015 DRAM_CONT_HIGH_OFF + (int) channel * 4,
2016 &tmp);
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05002017 chan_addr += (u64) ((tmp >> 11) & 0xfff) << 27;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002018 }
2019
2020 f15h_select_dct(pvt, channel);
2021
2022 edac_dbg(1, " Normalized DCT addr: 0x%llx\n", chan_addr);
2023
2024 /*
2025 * Find Chip select:
2026 * if channel = 3, then alias it to 1. This is because, in F15 M30h,
2027 * there is support for 4 DCT's, but only 2 are currently functional.
2028 * They are DCT0 and DCT3. But we have read all registers of DCT3 into
2029 * pvt->csels[1]. So we need to use '1' here to get correct info.
2030 * Refer F15 M30h BKDG Section 2.10 and 2.10.3 for clarifications.
2031 */
2032 alias_channel = (channel == 3) ? 1 : channel;
2033
2034 cs_found = f1x_lookup_addr_in_dct(chan_addr, node_id, alias_channel);
2035
2036 if (cs_found >= 0)
2037 *chan_sel = alias_channel;
2038
2039 return cs_found;
2040}
2041
2042static int f1x_translate_sysaddr_to_cs(struct amd64_pvt *pvt,
2043 u64 sys_addr,
2044 int *chan_sel)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002045{
Borislav Petkove761359a2011-02-21 19:49:01 +01002046 int cs_found = -EINVAL;
2047 unsigned range;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002048
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002049 for (range = 0; range < DRAM_RANGES; range++) {
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002050 if (!dram_rw(pvt, range))
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002051 continue;
2052
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002053 if (pvt->fam == 0x15 && pvt->model >= 0x30)
2054 cs_found = f15_m30h_match_to_this_node(pvt, range,
2055 sys_addr,
2056 chan_sel);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002057
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002058 else if ((get_dram_base(pvt, range) <= sys_addr) &&
2059 (get_dram_limit(pvt, range) >= sys_addr)) {
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01002060 cs_found = f1x_match_to_this_node(pvt, range,
Borislav Petkov33ca0642012-08-30 18:01:36 +02002061 sys_addr, chan_sel);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002062 if (cs_found >= 0)
2063 break;
2064 }
2065 }
2066 return cs_found;
2067}
2068
2069/*
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002070 * For reference see "2.8.5 Routing DRAM Requests" in F10 BKDG. This code maps
2071 * a @sys_addr to NodeID, DCT (channel) and chip select (CSROW).
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002072 *
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002073 * The @sys_addr is usually an error address received from the hardware
2074 * (MCX_ADDR).
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002075 */
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01002076static void f1x_map_sysaddr_to_csrow(struct mem_ctl_info *mci, u64 sys_addr,
Borislav Petkov33ca0642012-08-30 18:01:36 +02002077 struct err_info *err)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002078{
2079 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002080
Borislav Petkov33ca0642012-08-30 18:01:36 +02002081 error_address_to_page_and_offset(sys_addr, err);
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03002082
Borislav Petkov33ca0642012-08-30 18:01:36 +02002083 err->csrow = f1x_translate_sysaddr_to_cs(pvt, sys_addr, &err->channel);
2084 if (err->csrow < 0) {
2085 err->err_code = ERR_CSROW;
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002086 return;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002087 }
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002088
Borislav Petkovbdc30a02009-11-13 15:10:43 +01002089 /*
2090 * We need the syndromes for channel detection only when we're
2091 * ganged. Otherwise @chan should already contain the channel at
2092 * this point.
2093 */
Borislav Petkova97fa682010-12-23 14:07:18 +01002094 if (dct_ganging_enabled(pvt))
Borislav Petkov33ca0642012-08-30 18:01:36 +02002095 err->channel = get_channel_from_ecc_syndrome(mci, err->syndrome);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002096}
2097
2098/*
Borislav Petkov8566c4d2009-10-16 13:48:28 +02002099 * debug routine to display the memory sizes of all logical DIMMs and its
Borislav Petkovcb328502010-12-22 14:28:24 +01002100 * CSROWs
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002101 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002102static void debug_display_dimm_sizes(struct amd64_pvt *pvt, u8 ctrl)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002103{
Borislav Petkovbb89f5a2012-09-12 18:06:00 +02002104 int dimm, size0, size1;
Borislav Petkov525a1b22010-12-21 15:53:27 +01002105 u32 *dcsb = ctrl ? pvt->csels[1].csbases : pvt->csels[0].csbases;
2106 u32 dbam = ctrl ? pvt->dbam1 : pvt->dbam0;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002107
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002108 if (pvt->fam == 0xf) {
Borislav Petkov8566c4d2009-10-16 13:48:28 +02002109 /* K8 families < revF not supported yet */
Borislav Petkov1433eb92009-10-21 13:44:36 +02002110 if (pvt->ext_model < K8_REV_F)
Borislav Petkov8566c4d2009-10-16 13:48:28 +02002111 return;
2112 else
2113 WARN_ON(ctrl != 0);
2114 }
2115
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002116 if (pvt->fam == 0x10) {
2117 dbam = (ctrl && !dct_ganging_enabled(pvt)) ? pvt->dbam1
2118 : pvt->dbam0;
2119 dcsb = (ctrl && !dct_ganging_enabled(pvt)) ?
2120 pvt->csels[1].csbases :
2121 pvt->csels[0].csbases;
2122 } else if (ctrl) {
2123 dbam = pvt->dbam0;
2124 dcsb = pvt->csels[1].csbases;
2125 }
Joe Perches956b9ba12012-04-29 17:08:39 -03002126 edac_dbg(1, "F2x%d80 (DRAM Bank Address Mapping): 0x%08x\n",
2127 ctrl, dbam);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002128
Borislav Petkov8566c4d2009-10-16 13:48:28 +02002129 edac_printk(KERN_DEBUG, EDAC_MC, "DCT%d chip selects:\n", ctrl);
2130
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002131 /* Dump memory sizes for DIMM and its CSROWs */
2132 for (dimm = 0; dimm < 4; dimm++) {
2133
2134 size0 = 0;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002135 if (dcsb[dimm*2] & DCSB_CS_ENABLE)
Yazen Ghannam07ed82e2016-11-28 08:50:21 -06002136 /*
2137 * For F15m60h, we need multiplier for LRDIMM cs_size
2138 * calculation. We pass dimm value to the dbam_to_cs
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002139 * mapper so we can find the multiplier from the
2140 * corresponding DCSM.
2141 */
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01002142 size0 = pvt->ops->dbam_to_cs(pvt, ctrl,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002143 DBAM_DIMM(dimm, dbam),
2144 dimm);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002145
2146 size1 = 0;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002147 if (dcsb[dimm*2 + 1] & DCSB_CS_ENABLE)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01002148 size1 = pvt->ops->dbam_to_cs(pvt, ctrl,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002149 DBAM_DIMM(dimm, dbam),
2150 dimm);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002151
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002152 amd64_info(EDAC_MC ": %d: %5dMB %d: %5dMB\n",
Borislav Petkovbb89f5a2012-09-12 18:06:00 +02002153 dimm * 2, size0,
2154 dimm * 2 + 1, size1);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02002155 }
2156}
2157
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002158static struct amd64_family_type family_types[] = {
Doug Thompson4d376072009-04-27 16:25:05 +02002159 [K8_CPUS] = {
Borislav Petkov0092b202010-10-01 19:20:05 +02002160 .ctl_name = "K8",
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002161 .f1_id = PCI_DEVICE_ID_AMD_K8_NB_ADDRMAP,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002162 .f2_id = PCI_DEVICE_ID_AMD_K8_NB_MEMCTL,
Doug Thompson4d376072009-04-27 16:25:05 +02002163 .ops = {
Borislav Petkov1433eb92009-10-21 13:44:36 +02002164 .early_channel_count = k8_early_channel_count,
Borislav Petkov1433eb92009-10-21 13:44:36 +02002165 .map_sysaddr_to_csrow = k8_map_sysaddr_to_csrow,
2166 .dbam_to_cs = k8_dbam_to_chip_select,
Doug Thompson4d376072009-04-27 16:25:05 +02002167 }
2168 },
2169 [F10_CPUS] = {
Borislav Petkov0092b202010-10-01 19:20:05 +02002170 .ctl_name = "F10h",
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002171 .f1_id = PCI_DEVICE_ID_AMD_10H_NB_MAP,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002172 .f2_id = PCI_DEVICE_ID_AMD_10H_NB_DRAM,
Doug Thompson4d376072009-04-27 16:25:05 +02002173 .ops = {
Borislav Petkov7d20d142011-01-07 17:58:04 +01002174 .early_channel_count = f1x_early_channel_count,
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01002175 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
Borislav Petkov1433eb92009-10-21 13:44:36 +02002176 .dbam_to_cs = f10_dbam_to_chip_select,
Borislav Petkovb2b0c602010-10-08 18:32:29 +02002177 }
2178 },
2179 [F15_CPUS] = {
2180 .ctl_name = "F15h",
Borislav Petkovdf71a052011-01-19 18:15:10 +01002181 .f1_id = PCI_DEVICE_ID_AMD_15H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002182 .f2_id = PCI_DEVICE_ID_AMD_15H_NB_F2,
Borislav Petkovb2b0c602010-10-08 18:32:29 +02002183 .ops = {
Borislav Petkov7d20d142011-01-07 17:58:04 +01002184 .early_channel_count = f1x_early_channel_count,
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01002185 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01002186 .dbam_to_cs = f15_dbam_to_chip_select,
Doug Thompson4d376072009-04-27 16:25:05 +02002187 }
2188 },
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002189 [F15_M30H_CPUS] = {
2190 .ctl_name = "F15h_M30h",
2191 .f1_id = PCI_DEVICE_ID_AMD_15H_M30H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002192 .f2_id = PCI_DEVICE_ID_AMD_15H_M30H_NB_F2,
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002193 .ops = {
2194 .early_channel_count = f1x_early_channel_count,
2195 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
2196 .dbam_to_cs = f16_dbam_to_chip_select,
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002197 }
2198 },
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002199 [F15_M60H_CPUS] = {
2200 .ctl_name = "F15h_M60h",
2201 .f1_id = PCI_DEVICE_ID_AMD_15H_M60H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002202 .f2_id = PCI_DEVICE_ID_AMD_15H_M60H_NB_F2,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002203 .ops = {
2204 .early_channel_count = f1x_early_channel_count,
2205 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
2206 .dbam_to_cs = f15_m60h_dbam_to_chip_select,
2207 }
2208 },
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05002209 [F16_CPUS] = {
2210 .ctl_name = "F16h",
2211 .f1_id = PCI_DEVICE_ID_AMD_16H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002212 .f2_id = PCI_DEVICE_ID_AMD_16H_NB_F2,
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05002213 .ops = {
2214 .early_channel_count = f1x_early_channel_count,
2215 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
2216 .dbam_to_cs = f16_dbam_to_chip_select,
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05002217 }
2218 },
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06002219 [F16_M30H_CPUS] = {
2220 .ctl_name = "F16h_M30h",
2221 .f1_id = PCI_DEVICE_ID_AMD_16H_M30H_NB_F1,
Borislav Petkov3f37a362016-05-06 19:44:27 +02002222 .f2_id = PCI_DEVICE_ID_AMD_16H_M30H_NB_F2,
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06002223 .ops = {
2224 .early_channel_count = f1x_early_channel_count,
2225 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
2226 .dbam_to_cs = f16_dbam_to_chip_select,
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06002227 }
2228 },
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05002229 [F17_CPUS] = {
2230 .ctl_name = "F17h",
2231 .f0_id = PCI_DEVICE_ID_AMD_17H_DF_F0,
2232 .f6_id = PCI_DEVICE_ID_AMD_17H_DF_F6,
2233 .ops = {
2234 .early_channel_count = f17_early_channel_count,
2235 .dbam_to_cs = f17_base_addr_to_cs_size,
2236 }
2237 },
Michael Jin8960de42018-08-16 15:28:40 -04002238 [F17_M10H_CPUS] = {
2239 .ctl_name = "F17h_M10h",
2240 .f0_id = PCI_DEVICE_ID_AMD_17H_M10H_DF_F0,
2241 .f6_id = PCI_DEVICE_ID_AMD_17H_M10H_DF_F6,
2242 .ops = {
2243 .early_channel_count = f17_early_channel_count,
2244 .dbam_to_cs = f17_base_addr_to_cs_size,
2245 }
2246 },
Yazen Ghannam6e8462392019-02-28 15:36:09 +00002247 [F17_M30H_CPUS] = {
2248 .ctl_name = "F17h_M30h",
2249 .f0_id = PCI_DEVICE_ID_AMD_17H_M30H_DF_F0,
2250 .f6_id = PCI_DEVICE_ID_AMD_17H_M30H_DF_F6,
2251 .ops = {
2252 .early_channel_count = f17_early_channel_count,
2253 .dbam_to_cs = f17_base_addr_to_cs_size,
2254 }
2255 },
Doug Thompson4d376072009-04-27 16:25:05 +02002256};
2257
Doug Thompsonb1289d62009-04-27 16:37:05 +02002258/*
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002259 * These are tables of eigenvectors (one per line) which can be used for the
2260 * construction of the syndrome tables. The modified syndrome search algorithm
2261 * uses those to find the symbol in error and thus the DIMM.
Doug Thompsonb1289d62009-04-27 16:37:05 +02002262 *
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002263 * Algorithm courtesy of Ross LaFetra from AMD.
Doug Thompsonb1289d62009-04-27 16:37:05 +02002264 */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002265static const u16 x4_vectors[] = {
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002266 0x2f57, 0x1afe, 0x66cc, 0xdd88,
2267 0x11eb, 0x3396, 0x7f4c, 0xeac8,
2268 0x0001, 0x0002, 0x0004, 0x0008,
2269 0x1013, 0x3032, 0x4044, 0x8088,
2270 0x106b, 0x30d6, 0x70fc, 0xe0a8,
2271 0x4857, 0xc4fe, 0x13cc, 0x3288,
2272 0x1ac5, 0x2f4a, 0x5394, 0xa1e8,
2273 0x1f39, 0x251e, 0xbd6c, 0x6bd8,
2274 0x15c1, 0x2a42, 0x89ac, 0x4758,
2275 0x2b03, 0x1602, 0x4f0c, 0xca08,
2276 0x1f07, 0x3a0e, 0x6b04, 0xbd08,
2277 0x8ba7, 0x465e, 0x244c, 0x1cc8,
2278 0x2b87, 0x164e, 0x642c, 0xdc18,
2279 0x40b9, 0x80de, 0x1094, 0x20e8,
2280 0x27db, 0x1eb6, 0x9dac, 0x7b58,
2281 0x11c1, 0x2242, 0x84ac, 0x4c58,
2282 0x1be5, 0x2d7a, 0x5e34, 0xa718,
2283 0x4b39, 0x8d1e, 0x14b4, 0x28d8,
2284 0x4c97, 0xc87e, 0x11fc, 0x33a8,
2285 0x8e97, 0x497e, 0x2ffc, 0x1aa8,
2286 0x16b3, 0x3d62, 0x4f34, 0x8518,
2287 0x1e2f, 0x391a, 0x5cac, 0xf858,
2288 0x1d9f, 0x3b7a, 0x572c, 0xfe18,
2289 0x15f5, 0x2a5a, 0x5264, 0xa3b8,
2290 0x1dbb, 0x3b66, 0x715c, 0xe3f8,
2291 0x4397, 0xc27e, 0x17fc, 0x3ea8,
2292 0x1617, 0x3d3e, 0x6464, 0xb8b8,
2293 0x23ff, 0x12aa, 0xab6c, 0x56d8,
2294 0x2dfb, 0x1ba6, 0x913c, 0x7328,
2295 0x185d, 0x2ca6, 0x7914, 0x9e28,
2296 0x171b, 0x3e36, 0x7d7c, 0xebe8,
2297 0x4199, 0x82ee, 0x19f4, 0x2e58,
2298 0x4807, 0xc40e, 0x130c, 0x3208,
2299 0x1905, 0x2e0a, 0x5804, 0xac08,
2300 0x213f, 0x132a, 0xadfc, 0x5ba8,
2301 0x19a9, 0x2efe, 0xb5cc, 0x6f88,
Doug Thompsonb1289d62009-04-27 16:37:05 +02002302};
2303
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002304static const u16 x8_vectors[] = {
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002305 0x0145, 0x028a, 0x2374, 0x43c8, 0xa1f0, 0x0520, 0x0a40, 0x1480,
2306 0x0211, 0x0422, 0x0844, 0x1088, 0x01b0, 0x44e0, 0x23c0, 0xed80,
2307 0x1011, 0x0116, 0x022c, 0x0458, 0x08b0, 0x8c60, 0x2740, 0x4e80,
2308 0x0411, 0x0822, 0x1044, 0x0158, 0x02b0, 0x2360, 0x46c0, 0xab80,
2309 0x0811, 0x1022, 0x012c, 0x0258, 0x04b0, 0x4660, 0x8cc0, 0x2780,
2310 0x2071, 0x40e2, 0xa0c4, 0x0108, 0x0210, 0x0420, 0x0840, 0x1080,
2311 0x4071, 0x80e2, 0x0104, 0x0208, 0x0410, 0x0820, 0x1040, 0x2080,
2312 0x8071, 0x0102, 0x0204, 0x0408, 0x0810, 0x1020, 0x2040, 0x4080,
2313 0x019d, 0x03d6, 0x136c, 0x2198, 0x50b0, 0xb2e0, 0x0740, 0x0e80,
2314 0x0189, 0x03ea, 0x072c, 0x0e58, 0x1cb0, 0x56e0, 0x37c0, 0xf580,
2315 0x01fd, 0x0376, 0x06ec, 0x0bb8, 0x1110, 0x2220, 0x4440, 0x8880,
2316 0x0163, 0x02c6, 0x1104, 0x0758, 0x0eb0, 0x2be0, 0x6140, 0xc280,
2317 0x02fd, 0x01c6, 0x0b5c, 0x1108, 0x07b0, 0x25a0, 0x8840, 0x6180,
2318 0x0801, 0x012e, 0x025c, 0x04b8, 0x1370, 0x26e0, 0x57c0, 0xb580,
2319 0x0401, 0x0802, 0x015c, 0x02b8, 0x22b0, 0x13e0, 0x7140, 0xe280,
2320 0x0201, 0x0402, 0x0804, 0x01b8, 0x11b0, 0x31a0, 0x8040, 0x7180,
2321 0x0101, 0x0202, 0x0404, 0x0808, 0x1010, 0x2020, 0x4040, 0x8080,
2322 0x0001, 0x0002, 0x0004, 0x0008, 0x0010, 0x0020, 0x0040, 0x0080,
2323 0x0100, 0x0200, 0x0400, 0x0800, 0x1000, 0x2000, 0x4000, 0x8000,
2324};
2325
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002326static int decode_syndrome(u16 syndrome, const u16 *vectors, unsigned num_vecs,
Borislav Petkovd34a6ec2011-02-23 17:41:50 +01002327 unsigned v_dim)
Doug Thompsonb1289d62009-04-27 16:37:05 +02002328{
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002329 unsigned int i, err_sym;
Doug Thompsonb1289d62009-04-27 16:37:05 +02002330
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002331 for (err_sym = 0; err_sym < num_vecs / v_dim; err_sym++) {
2332 u16 s = syndrome;
Borislav Petkovd34a6ec2011-02-23 17:41:50 +01002333 unsigned v_idx = err_sym * v_dim;
2334 unsigned v_end = (err_sym + 1) * v_dim;
Doug Thompsonb1289d62009-04-27 16:37:05 +02002335
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002336 /* walk over all 16 bits of the syndrome */
2337 for (i = 1; i < (1U << 16); i <<= 1) {
2338
2339 /* if bit is set in that eigenvector... */
2340 if (v_idx < v_end && vectors[v_idx] & i) {
2341 u16 ev_comp = vectors[v_idx++];
2342
2343 /* ... and bit set in the modified syndrome, */
2344 if (s & i) {
2345 /* remove it. */
2346 s ^= ev_comp;
2347
2348 if (!s)
2349 return err_sym;
2350 }
2351
2352 } else if (s & i)
2353 /* can't get to zero, move to next symbol */
2354 break;
2355 }
Doug Thompsonb1289d62009-04-27 16:37:05 +02002356 }
2357
Joe Perches956b9ba12012-04-29 17:08:39 -03002358 edac_dbg(0, "syndrome(%x) not found\n", syndrome);
Doug Thompsonb1289d62009-04-27 16:37:05 +02002359 return -1;
2360}
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002361
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002362static int map_err_sym_to_channel(int err_sym, int sym_size)
2363{
2364 if (sym_size == 4)
2365 switch (err_sym) {
2366 case 0x20:
2367 case 0x21:
2368 return 0;
2369 break;
2370 case 0x22:
2371 case 0x23:
2372 return 1;
2373 break;
2374 default:
2375 return err_sym >> 4;
2376 break;
2377 }
2378 /* x8 symbols */
2379 else
2380 switch (err_sym) {
2381 /* imaginary bits not in a DIMM */
2382 case 0x10:
2383 WARN(1, KERN_ERR "Invalid error symbol: 0x%x\n",
2384 err_sym);
2385 return -1;
2386 break;
2387
2388 case 0x11:
2389 return 0;
2390 break;
2391 case 0x12:
2392 return 1;
2393 break;
2394 default:
2395 return err_sym >> 3;
2396 break;
2397 }
2398 return -1;
2399}
2400
2401static int get_channel_from_ecc_syndrome(struct mem_ctl_info *mci, u16 syndrome)
2402{
2403 struct amd64_pvt *pvt = mci->pvt_info;
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002404 int err_sym = -1;
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002405
Borislav Petkova3b7db02011-01-19 20:35:12 +01002406 if (pvt->ecc_sym_sz == 8)
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002407 err_sym = decode_syndrome(syndrome, x8_vectors,
2408 ARRAY_SIZE(x8_vectors),
Borislav Petkova3b7db02011-01-19 20:35:12 +01002409 pvt->ecc_sym_sz);
2410 else if (pvt->ecc_sym_sz == 4)
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002411 err_sym = decode_syndrome(syndrome, x4_vectors,
2412 ARRAY_SIZE(x4_vectors),
Borislav Petkova3b7db02011-01-19 20:35:12 +01002413 pvt->ecc_sym_sz);
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002414 else {
Borislav Petkova3b7db02011-01-19 20:35:12 +01002415 amd64_warn("Illegal syndrome type: %u\n", pvt->ecc_sym_sz);
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002416 return err_sym;
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002417 }
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002418
Borislav Petkova3b7db02011-01-19 20:35:12 +01002419 return map_err_sym_to_channel(err_sym, pvt->ecc_sym_sz);
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002420}
2421
Yazen Ghanname70984d2016-11-17 17:57:31 -05002422static void __log_ecc_error(struct mem_ctl_info *mci, struct err_info *err,
Borislav Petkov33ca0642012-08-30 18:01:36 +02002423 u8 ecc_type)
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002424{
Borislav Petkov33ca0642012-08-30 18:01:36 +02002425 enum hw_event_mc_err_type err_type;
2426 const char *string;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002427
Borislav Petkov33ca0642012-08-30 18:01:36 +02002428 if (ecc_type == 2)
2429 err_type = HW_EVENT_ERR_CORRECTED;
2430 else if (ecc_type == 1)
2431 err_type = HW_EVENT_ERR_UNCORRECTED;
Yazen Ghannamd12a9692016-11-17 17:57:32 -05002432 else if (ecc_type == 3)
2433 err_type = HW_EVENT_ERR_DEFERRED;
Borislav Petkov33ca0642012-08-30 18:01:36 +02002434 else {
2435 WARN(1, "Something is rotten in the state of Denmark.\n");
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002436 return;
2437 }
2438
Borislav Petkov33ca0642012-08-30 18:01:36 +02002439 switch (err->err_code) {
2440 case DECODE_OK:
2441 string = "";
2442 break;
2443 case ERR_NODE:
2444 string = "Failed to map error addr to a node";
2445 break;
2446 case ERR_CSROW:
2447 string = "Failed to map error addr to a csrow";
2448 break;
2449 case ERR_CHANNEL:
Yazen Ghannam713ad542016-11-28 12:59:53 -06002450 string = "Unknown syndrome - possible error reporting race";
2451 break;
2452 case ERR_SYND:
2453 string = "MCA_SYND not valid - unknown syndrome and csrow";
2454 break;
2455 case ERR_NORM_ADDR:
2456 string = "Cannot decode normalized address";
Borislav Petkov33ca0642012-08-30 18:01:36 +02002457 break;
2458 default:
2459 string = "WTF error";
2460 break;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002461 }
Borislav Petkov33ca0642012-08-30 18:01:36 +02002462
2463 edac_mc_handle_error(err_type, mci, 1,
2464 err->page, err->offset, err->syndrome,
2465 err->csrow, err->channel, -1,
2466 string, "");
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002467}
2468
Borislav Petkovdf781d02013-12-15 17:29:44 +01002469static inline void decode_bus_error(int node_id, struct mce *m)
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002470{
Daniel J Blueman0c510cc2015-02-17 11:34:38 +08002471 struct mem_ctl_info *mci;
2472 struct amd64_pvt *pvt;
Borislav Petkovf192c7b2011-01-10 14:24:32 +01002473 u8 ecc_type = (m->status >> 45) & 0x3;
Borislav Petkov66fed2d2012-08-09 18:41:07 +02002474 u8 xec = XEC(m->status, 0x1f);
2475 u16 ec = EC(m->status);
Borislav Petkov33ca0642012-08-30 18:01:36 +02002476 u64 sys_addr;
2477 struct err_info err;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002478
Daniel J Blueman0c510cc2015-02-17 11:34:38 +08002479 mci = edac_mc_find(node_id);
2480 if (!mci)
2481 return;
2482
2483 pvt = mci->pvt_info;
2484
Borislav Petkov66fed2d2012-08-09 18:41:07 +02002485 /* Bail out early if this was an 'observed' error */
Borislav Petkov5980bb92011-01-07 16:26:49 +01002486 if (PP(ec) == NBSL_PP_OBS)
Borislav Petkovb70ef012009-06-25 19:32:38 +02002487 return;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002488
Borislav Petkovecaf5602009-07-23 16:32:01 +02002489 /* Do only ECC errors */
2490 if (xec && xec != F10_NBSL_EXT_ERR_ECC)
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002491 return;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002492
Borislav Petkov33ca0642012-08-30 18:01:36 +02002493 memset(&err, 0, sizeof(err));
2494
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002495 sys_addr = get_error_address(pvt, m);
Borislav Petkov33ca0642012-08-30 18:01:36 +02002496
Borislav Petkovecaf5602009-07-23 16:32:01 +02002497 if (ecc_type == 2)
Borislav Petkov33ca0642012-08-30 18:01:36 +02002498 err.syndrome = extract_syndrome(m->status);
2499
2500 pvt->ops->map_sysaddr_to_csrow(mci, sys_addr, &err);
2501
Yazen Ghanname70984d2016-11-17 17:57:31 -05002502 __log_ecc_error(mci, &err, ecc_type);
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002503}
2504
Doug Thompson0ec449e2009-04-27 19:41:25 +02002505/*
Yazen Ghannam713ad542016-11-28 12:59:53 -06002506 * To find the UMC channel represented by this bank we need to match on its
2507 * instance_id. The instance_id of a bank is held in the lower 32 bits of its
2508 * IPID.
Yazen Ghannambdcee772019-02-28 15:36:10 +00002509 *
2510 * Currently, we can derive the channel number by looking at the 6th nibble in
2511 * the instance_id. For example, instance_id=0xYXXXXX where Y is the channel
2512 * number.
Yazen Ghannam713ad542016-11-28 12:59:53 -06002513 */
Yazen Ghannambdcee772019-02-28 15:36:10 +00002514static int find_umc_channel(struct mce *m)
Yazen Ghannam713ad542016-11-28 12:59:53 -06002515{
Yazen Ghannambdcee772019-02-28 15:36:10 +00002516 return (m->ipid & GENMASK(31, 0)) >> 20;
Yazen Ghannam713ad542016-11-28 12:59:53 -06002517}
2518
2519static void decode_umc_error(int node_id, struct mce *m)
2520{
2521 u8 ecc_type = (m->status >> 45) & 0x3;
2522 struct mem_ctl_info *mci;
2523 struct amd64_pvt *pvt;
2524 struct err_info err;
2525 u64 sys_addr;
2526
2527 mci = edac_mc_find(node_id);
2528 if (!mci)
2529 return;
2530
2531 pvt = mci->pvt_info;
2532
2533 memset(&err, 0, sizeof(err));
2534
2535 if (m->status & MCI_STATUS_DEFERRED)
2536 ecc_type = 3;
2537
Yazen Ghannambdcee772019-02-28 15:36:10 +00002538 err.channel = find_umc_channel(m);
Yazen Ghannam713ad542016-11-28 12:59:53 -06002539
2540 if (umc_normaddr_to_sysaddr(m->addr, pvt->mc_node_id, err.channel, &sys_addr)) {
2541 err.err_code = ERR_NORM_ADDR;
2542 goto log_error;
2543 }
2544
2545 error_address_to_page_and_offset(sys_addr, &err);
2546
2547 if (!(m->status & MCI_STATUS_SYNDV)) {
2548 err.err_code = ERR_SYND;
2549 goto log_error;
2550 }
2551
2552 if (ecc_type == 2) {
2553 u8 length = (m->synd >> 18) & 0x3f;
2554
2555 if (length)
2556 err.syndrome = (m->synd >> 32) & GENMASK(length - 1, 0);
2557 else
2558 err.err_code = ERR_CHANNEL;
2559 }
2560
2561 err.csrow = m->synd & 0x7;
2562
2563log_error:
2564 __log_ecc_error(mci, &err, ecc_type);
2565}
2566
2567/*
Borislav Petkov3f37a362016-05-06 19:44:27 +02002568 * Use pvt->F3 which contains the F3 CPU PCI device to get the related
2569 * F1 (AddrMap) and F2 (Dct) devices. Return negative value on error.
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002570 * Reserve F0 and F6 on systems with a UMC.
Doug Thompson0ec449e2009-04-27 19:41:25 +02002571 */
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002572static int
2573reserve_mc_sibling_devs(struct amd64_pvt *pvt, u16 pci_id1, u16 pci_id2)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002574{
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002575 if (pvt->umc) {
2576 pvt->F0 = pci_get_related_function(pvt->F3->vendor, pci_id1, pvt->F3);
2577 if (!pvt->F0) {
Borislav Petkov5246c542016-12-01 11:35:07 +01002578 amd64_err("F0 not found, device 0x%x (broken BIOS?)\n", pci_id1);
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002579 return -ENODEV;
2580 }
2581
2582 pvt->F6 = pci_get_related_function(pvt->F3->vendor, pci_id2, pvt->F3);
2583 if (!pvt->F6) {
2584 pci_dev_put(pvt->F0);
2585 pvt->F0 = NULL;
2586
Borislav Petkov5246c542016-12-01 11:35:07 +01002587 amd64_err("F6 not found: device 0x%x (broken BIOS?)\n", pci_id2);
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002588 return -ENODEV;
2589 }
Borislav Petkov5246c542016-12-01 11:35:07 +01002590
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002591 edac_dbg(1, "F0: %s\n", pci_name(pvt->F0));
2592 edac_dbg(1, "F3: %s\n", pci_name(pvt->F3));
2593 edac_dbg(1, "F6: %s\n", pci_name(pvt->F6));
2594
2595 return 0;
2596 }
2597
Doug Thompson0ec449e2009-04-27 19:41:25 +02002598 /* Reserve the ADDRESS MAP Device */
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002599 pvt->F1 = pci_get_related_function(pvt->F3->vendor, pci_id1, pvt->F3);
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002600 if (!pvt->F1) {
Borislav Petkov5246c542016-12-01 11:35:07 +01002601 amd64_err("F1 not found: device 0x%x (broken BIOS?)\n", pci_id1);
Borislav Petkovbbd0c1f62010-10-01 19:27:58 +02002602 return -ENODEV;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002603 }
2604
Borislav Petkov3f37a362016-05-06 19:44:27 +02002605 /* Reserve the DCT Device */
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002606 pvt->F2 = pci_get_related_function(pvt->F3->vendor, pci_id2, pvt->F3);
Borislav Petkov3f37a362016-05-06 19:44:27 +02002607 if (!pvt->F2) {
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002608 pci_dev_put(pvt->F1);
2609 pvt->F1 = NULL;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002610
Borislav Petkov5246c542016-12-01 11:35:07 +01002611 amd64_err("F2 not found: device 0x%x (broken BIOS?)\n", pci_id2);
2612 return -ENODEV;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002613 }
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002614
Joe Perches956b9ba12012-04-29 17:08:39 -03002615 edac_dbg(1, "F1: %s\n", pci_name(pvt->F1));
2616 edac_dbg(1, "F2: %s\n", pci_name(pvt->F2));
2617 edac_dbg(1, "F3: %s\n", pci_name(pvt->F3));
Doug Thompson0ec449e2009-04-27 19:41:25 +02002618
2619 return 0;
2620}
2621
Borislav Petkov360b7f32010-10-15 19:25:38 +02002622static void free_mc_sibling_devs(struct amd64_pvt *pvt)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002623{
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05002624 if (pvt->umc) {
2625 pci_dev_put(pvt->F0);
2626 pci_dev_put(pvt->F6);
2627 } else {
2628 pci_dev_put(pvt->F1);
2629 pci_dev_put(pvt->F2);
2630 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002631}
2632
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002633static void determine_ecc_sym_sz(struct amd64_pvt *pvt)
2634{
2635 pvt->ecc_sym_sz = 4;
2636
2637 if (pvt->umc) {
2638 u8 i;
2639
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00002640 for_each_umc(i) {
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002641 /* Check enabled channels only: */
Yazen Ghannam78359612019-02-28 15:36:11 +00002642 if (pvt->umc[i].sdp_ctrl & UMC_SDP_INIT) {
2643 if (pvt->umc[i].ecc_ctrl & BIT(9)) {
2644 pvt->ecc_sym_sz = 16;
2645 return;
2646 } else if (pvt->umc[i].ecc_ctrl & BIT(7)) {
2647 pvt->ecc_sym_sz = 8;
2648 return;
2649 }
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002650 }
2651 }
Yazen Ghannam78359612019-02-28 15:36:11 +00002652 } else if (pvt->fam >= 0x10) {
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002653 u32 tmp;
2654
2655 amd64_read_pci_cfg(pvt->F3, EXT_NB_MCA_CFG, &tmp);
2656 /* F16h has only DCT0, so no need to read dbam1. */
2657 if (pvt->fam != 0x16)
2658 amd64_read_dct_pci_cfg(pvt, 1, DBAM0, &pvt->dbam1);
2659
2660 /* F10h, revD and later can do x8 ECC too. */
2661 if ((pvt->fam > 0x10 || pvt->model > 7) && tmp & BIT(25))
2662 pvt->ecc_sym_sz = 8;
2663 }
2664}
2665
2666/*
2667 * Retrieve the hardware registers of the memory controller.
2668 */
2669static void __read_mc_regs_df(struct amd64_pvt *pvt)
2670{
2671 u8 nid = pvt->mc_node_id;
2672 struct amd64_umc *umc;
2673 u32 i, umc_base;
2674
2675 /* Read registers from each UMC */
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00002676 for_each_umc(i) {
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002677
2678 umc_base = get_umc_base(i);
2679 umc = &pvt->umc[i];
2680
Yazen Ghannam07ed82e2016-11-28 08:50:21 -06002681 amd_smn_read(nid, umc_base + UMCCH_DIMM_CFG, &umc->dimm_cfg);
2682 amd_smn_read(nid, umc_base + UMCCH_UMC_CFG, &umc->umc_cfg);
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002683 amd_smn_read(nid, umc_base + UMCCH_SDP_CTRL, &umc->sdp_ctrl);
2684 amd_smn_read(nid, umc_base + UMCCH_ECC_CTRL, &umc->ecc_ctrl);
Yazen Ghannam07ed82e2016-11-28 08:50:21 -06002685 amd_smn_read(nid, umc_base + UMCCH_UMC_CAP_HI, &umc->umc_cap_hi);
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002686 }
2687}
2688
Doug Thompson0ec449e2009-04-27 19:41:25 +02002689/*
2690 * Retrieve the hardware registers of the memory controller (this includes the
2691 * 'Address Map' and 'Misc' device regs)
2692 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02002693static void read_mc_regs(struct amd64_pvt *pvt)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002694{
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002695 unsigned int range;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002696 u64 msr_val;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002697
2698 /*
2699 * Retrieve TOP_MEM and TOP_MEM2; no masking off of reserved bits since
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002700 * those are Read-As-Zero.
Doug Thompson0ec449e2009-04-27 19:41:25 +02002701 */
Borislav Petkove97f8bb2009-10-12 15:27:45 +02002702 rdmsrl(MSR_K8_TOP_MEM1, pvt->top_mem);
Joe Perches956b9ba12012-04-29 17:08:39 -03002703 edac_dbg(0, " TOP_MEM: 0x%016llx\n", pvt->top_mem);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002704
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002705 /* Check first whether TOP_MEM2 is enabled: */
Doug Thompson0ec449e2009-04-27 19:41:25 +02002706 rdmsrl(MSR_K8_SYSCFG, msr_val);
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002707 if (msr_val & BIT(21)) {
Borislav Petkove97f8bb2009-10-12 15:27:45 +02002708 rdmsrl(MSR_K8_TOP_MEM2, pvt->top_mem2);
Joe Perches956b9ba12012-04-29 17:08:39 -03002709 edac_dbg(0, " TOP_MEM2: 0x%016llx\n", pvt->top_mem2);
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002710 } else {
Joe Perches956b9ba12012-04-29 17:08:39 -03002711 edac_dbg(0, " TOP_MEM2 disabled\n");
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002712 }
2713
2714 if (pvt->umc) {
2715 __read_mc_regs_df(pvt);
2716 amd64_read_pci_cfg(pvt->F0, DF_DHAR, &pvt->dhar);
2717
2718 goto skip;
2719 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002720
Borislav Petkov5980bb92011-01-07 16:26:49 +01002721 amd64_read_pci_cfg(pvt->F3, NBCAP, &pvt->nbcap);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002722
Borislav Petkov5a5d2372011-01-17 17:52:57 +01002723 read_dram_ctl_register(pvt);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002724
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002725 for (range = 0; range < DRAM_RANGES; range++) {
2726 u8 rw;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002727
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002728 /* read settings for this DRAM range */
2729 read_dram_base_limit_regs(pvt, range);
Borislav Petkove97f8bb2009-10-12 15:27:45 +02002730
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002731 rw = dram_rw(pvt, range);
2732 if (!rw)
2733 continue;
2734
Joe Perches956b9ba12012-04-29 17:08:39 -03002735 edac_dbg(1, " DRAM range[%d], base: 0x%016llx; limit: 0x%016llx\n",
2736 range,
2737 get_dram_base(pvt, range),
2738 get_dram_limit(pvt, range));
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002739
Joe Perches956b9ba12012-04-29 17:08:39 -03002740 edac_dbg(1, " IntlvEn=%s; Range access: %s%s IntlvSel=%d DstNode=%d\n",
2741 dram_intlv_en(pvt, range) ? "Enabled" : "Disabled",
2742 (rw & 0x1) ? "R" : "-",
2743 (rw & 0x2) ? "W" : "-",
2744 dram_intlv_sel(pvt, range),
2745 dram_dst_node(pvt, range));
Doug Thompson0ec449e2009-04-27 19:41:25 +02002746 }
2747
Borislav Petkovbc21fa52010-11-11 17:29:13 +01002748 amd64_read_pci_cfg(pvt->F1, DHAR, &pvt->dhar);
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002749 amd64_read_dct_pci_cfg(pvt, 0, DBAM0, &pvt->dbam0);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002750
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002751 amd64_read_pci_cfg(pvt->F3, F10_ONLINE_SPARE, &pvt->online_spare);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002752
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002753 amd64_read_dct_pci_cfg(pvt, 0, DCLR0, &pvt->dclr0);
2754 amd64_read_dct_pci_cfg(pvt, 0, DCHR0, &pvt->dchr0);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002755
Borislav Petkov78da1212010-12-22 19:31:45 +01002756 if (!dct_ganging_enabled(pvt)) {
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002757 amd64_read_dct_pci_cfg(pvt, 1, DCLR0, &pvt->dclr1);
2758 amd64_read_dct_pci_cfg(pvt, 1, DCHR0, &pvt->dchr1);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002759 }
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002760
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002761skip:
2762 read_dct_base_mask(pvt);
2763
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002764 determine_memory_type(pvt);
2765 edac_dbg(1, " DIMM type: %s\n", edac_mem_types[pvt->dram_type]);
Borislav Petkova3b7db02011-01-19 20:35:12 +01002766
Yazen Ghannamb64ce7c2016-11-17 17:57:37 -05002767 determine_ecc_sym_sz(pvt);
Borislav Petkova3b7db02011-01-19 20:35:12 +01002768
Borislav Petkovb2b0c602010-10-08 18:32:29 +02002769 dump_misc_regs(pvt);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002770}
2771
2772/*
2773 * NOTE: CPU Revision Dependent code
2774 *
2775 * Input:
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002776 * @csrow_nr ChipSelect Row Number (0..NUM_CHIPSELECTS-1)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002777 * k8 private pointer to -->
2778 * DRAM Bank Address mapping register
2779 * node_id
2780 * DCL register where dual_channel_active is
2781 *
2782 * The DBAM register consists of 4 sets of 4 bits each definitions:
2783 *
2784 * Bits: CSROWs
2785 * 0-3 CSROWs 0 and 1
2786 * 4-7 CSROWs 2 and 3
2787 * 8-11 CSROWs 4 and 5
2788 * 12-15 CSROWs 6 and 7
2789 *
2790 * Values range from: 0 to 15
2791 * The meaning of the values depends on CPU revision and dual-channel state,
2792 * see relevant BKDG more info.
2793 *
2794 * The memory controller provides for total of only 8 CSROWs in its current
2795 * architecture. Each "pair" of CSROWs normally represents just one DIMM in
2796 * single channel or two (2) DIMMs in dual channel mode.
2797 *
2798 * The following code logic collapses the various tables for CSROW based on CPU
2799 * revision.
2800 *
2801 * Returns:
2802 * The number of PAGE_SIZE pages on the specified CSROW number it
2803 * encompasses
2804 *
2805 */
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002806static u32 get_csrow_nr_pages(struct amd64_pvt *pvt, u8 dct, int csrow_nr_orig)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002807{
Ashish Shenoyf92cae42012-02-22 17:20:38 -08002808 u32 dbam = dct ? pvt->dbam1 : pvt->dbam0;
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002809 int csrow_nr = csrow_nr_orig;
2810 u32 cs_mode, nr_pages;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002811
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002812 if (!pvt->umc)
2813 csrow_nr >>= 1;
Borislav Petkov10de6492012-09-12 19:00:38 +02002814
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002815 cs_mode = DBAM_DIMM(csrow_nr, dbam);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002816
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002817 nr_pages = pvt->ops->dbam_to_cs(pvt, dct, cs_mode, csrow_nr);
2818 nr_pages <<= 20 - PAGE_SHIFT;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002819
Borislav Petkov10de6492012-09-12 19:00:38 +02002820 edac_dbg(0, "csrow: %d, channel: %d, DBAM idx: %d\n",
Yazen Ghannameb77e6b2017-04-27 12:11:54 -05002821 csrow_nr_orig, dct, cs_mode);
Borislav Petkov10de6492012-09-12 19:00:38 +02002822 edac_dbg(0, "nr_pages/channel: %u\n", nr_pages);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002823
2824 return nr_pages;
2825}
2826
2827/*
2828 * Initialize the array of csrow attribute instances, based on the values
2829 * from pci config hardware registers.
2830 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02002831static int init_csrows(struct mem_ctl_info *mci)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002832{
Borislav Petkov10de6492012-09-12 19:00:38 +02002833 struct amd64_pvt *pvt = mci->pvt_info;
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06002834 enum edac_type edac_mode = EDAC_NONE;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002835 struct csrow_info *csrow;
Mauro Carvalho Chehabde3910eb2012-04-24 15:05:43 -03002836 struct dimm_info *dimm;
Borislav Petkov10de6492012-09-12 19:00:38 +02002837 int i, j, empty = 1;
Mauro Carvalho Chehaba895bf82012-01-28 09:09:38 -03002838 int nr_pages = 0;
Borislav Petkov10de6492012-09-12 19:00:38 +02002839 u32 val;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002840
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06002841 if (!pvt->umc) {
2842 amd64_read_pci_cfg(pvt->F3, NBCFG, &val);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002843
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06002844 pvt->nbcfg = val;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002845
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06002846 edac_dbg(0, "node %d, NBCFG=0x%08x[ChipKillEccCap: %d|DramEccEn: %d]\n",
2847 pvt->mc_node_id, val,
2848 !!(val & NBCFG_CHIPKILL), !!(val & NBCFG_ECC_ENABLE));
2849 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002850
Borislav Petkov10de6492012-09-12 19:00:38 +02002851 /*
2852 * We iterate over DCT0 here but we look at DCT1 in parallel, if needed.
2853 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002854 for_each_chip_select(i, 0, pvt) {
Borislav Petkov10de6492012-09-12 19:00:38 +02002855 bool row_dct0 = !!csrow_enabled(i, 0, pvt);
2856 bool row_dct1 = false;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002857
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002858 if (pvt->fam != 0xf)
Borislav Petkov10de6492012-09-12 19:00:38 +02002859 row_dct1 = !!csrow_enabled(i, 1, pvt);
2860
2861 if (!row_dct0 && !row_dct1)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002862 continue;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002863
Borislav Petkov10de6492012-09-12 19:00:38 +02002864 csrow = mci->csrows[i];
Doug Thompson0ec449e2009-04-27 19:41:25 +02002865 empty = 0;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002866
Borislav Petkov10de6492012-09-12 19:00:38 +02002867 edac_dbg(1, "MC node: %d, csrow: %d\n",
2868 pvt->mc_node_id, i);
2869
Mauro Carvalho Chehab1eef1282013-03-11 09:07:46 -03002870 if (row_dct0) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002871 nr_pages = get_csrow_nr_pages(pvt, 0, i);
Mauro Carvalho Chehab1eef1282013-03-11 09:07:46 -03002872 csrow->channels[0]->dimm->nr_pages = nr_pages;
2873 }
Borislav Petkov10de6492012-09-12 19:00:38 +02002874
2875 /* K8 has only one DCT */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002876 if (pvt->fam != 0xf && row_dct1) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002877 int row_dct1_pages = get_csrow_nr_pages(pvt, 1, i);
Mauro Carvalho Chehab1eef1282013-03-11 09:07:46 -03002878
2879 csrow->channels[1]->dimm->nr_pages = row_dct1_pages;
2880 nr_pages += row_dct1_pages;
2881 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002882
Borislav Petkov10de6492012-09-12 19:00:38 +02002883 edac_dbg(1, "Total csrow%d pages: %u\n", i, nr_pages);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002884
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06002885 /* Determine DIMM ECC mode: */
2886 if (pvt->umc) {
2887 if (mci->edac_ctl_cap & EDAC_FLAG_S4ECD4ED)
2888 edac_mode = EDAC_S4ECD4ED;
2889 else if (mci->edac_ctl_cap & EDAC_FLAG_SECDED)
2890 edac_mode = EDAC_SECDED;
2891
2892 } else if (pvt->nbcfg & NBCFG_ECC_ENABLE) {
2893 edac_mode = (pvt->nbcfg & NBCFG_CHIPKILL)
2894 ? EDAC_S4ECD4ED
2895 : EDAC_SECDED;
2896 }
Mauro Carvalho Chehab084a4fc2012-01-27 18:38:08 -03002897
2898 for (j = 0; j < pvt->channel_count; j++) {
Mauro Carvalho Chehabde3910eb2012-04-24 15:05:43 -03002899 dimm = csrow->channels[j]->dimm;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002900 dimm->mtype = pvt->dram_type;
Mauro Carvalho Chehabde3910eb2012-04-24 15:05:43 -03002901 dimm->edac_mode = edac_mode;
Mauro Carvalho Chehab084a4fc2012-01-27 18:38:08 -03002902 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002903 }
2904
2905 return empty;
2906}
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002907
Borislav Petkov06724532009-09-16 13:05:46 +02002908/* get all cores on this DCT */
Daniel J Blueman8b84c8d2012-11-27 14:32:10 +08002909static void get_cpus_on_this_dct_cpumask(struct cpumask *mask, u16 nid)
Doug Thompsonf9431992009-04-27 19:46:08 +02002910{
Borislav Petkov06724532009-09-16 13:05:46 +02002911 int cpu;
Doug Thompsonf9431992009-04-27 19:46:08 +02002912
Borislav Petkov06724532009-09-16 13:05:46 +02002913 for_each_online_cpu(cpu)
2914 if (amd_get_nb_id(cpu) == nid)
2915 cpumask_set_cpu(cpu, mask);
Doug Thompsonf9431992009-04-27 19:46:08 +02002916}
2917
2918/* check MCG_CTL on all the cpus on this node */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002919static bool nb_mce_bank_enabled_on_node(u16 nid)
Doug Thompsonf9431992009-04-27 19:46:08 +02002920{
Rusty Russellba578cb2009-11-03 14:56:35 +10302921 cpumask_var_t mask;
Borislav Petkov50542252009-12-11 18:14:40 +01002922 int cpu, nbe;
Borislav Petkov06724532009-09-16 13:05:46 +02002923 bool ret = false;
Doug Thompsonf9431992009-04-27 19:46:08 +02002924
Rusty Russellba578cb2009-11-03 14:56:35 +10302925 if (!zalloc_cpumask_var(&mask, GFP_KERNEL)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002926 amd64_warn("%s: Error allocating mask\n", __func__);
Rusty Russellba578cb2009-11-03 14:56:35 +10302927 return false;
2928 }
Borislav Petkov06724532009-09-16 13:05:46 +02002929
Rusty Russellba578cb2009-11-03 14:56:35 +10302930 get_cpus_on_this_dct_cpumask(mask, nid);
Borislav Petkov06724532009-09-16 13:05:46 +02002931
Rusty Russellba578cb2009-11-03 14:56:35 +10302932 rdmsr_on_cpus(mask, MSR_IA32_MCG_CTL, msrs);
Borislav Petkov06724532009-09-16 13:05:46 +02002933
Rusty Russellba578cb2009-11-03 14:56:35 +10302934 for_each_cpu(cpu, mask) {
Borislav Petkov50542252009-12-11 18:14:40 +01002935 struct msr *reg = per_cpu_ptr(msrs, cpu);
Borislav Petkov5980bb92011-01-07 16:26:49 +01002936 nbe = reg->l & MSR_MCGCTL_NBE;
Borislav Petkov06724532009-09-16 13:05:46 +02002937
Joe Perches956b9ba12012-04-29 17:08:39 -03002938 edac_dbg(0, "core: %u, MCG_CTL: 0x%llx, NB MSR is %s\n",
2939 cpu, reg->q,
2940 (nbe ? "enabled" : "disabled"));
Borislav Petkov06724532009-09-16 13:05:46 +02002941
2942 if (!nbe)
2943 goto out;
Borislav Petkov06724532009-09-16 13:05:46 +02002944 }
2945 ret = true;
2946
2947out:
Rusty Russellba578cb2009-11-03 14:56:35 +10302948 free_cpumask_var(mask);
Doug Thompsonf9431992009-04-27 19:46:08 +02002949 return ret;
2950}
2951
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002952static int toggle_ecc_err_reporting(struct ecc_settings *s, u16 nid, bool on)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002953{
2954 cpumask_var_t cmask;
Borislav Petkov50542252009-12-11 18:14:40 +01002955 int cpu;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002956
2957 if (!zalloc_cpumask_var(&cmask, GFP_KERNEL)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002958 amd64_warn("%s: error allocating mask\n", __func__);
Pan Bian0de278842016-12-04 14:07:18 +08002959 return -ENOMEM;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002960 }
2961
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002962 get_cpus_on_this_dct_cpumask(cmask, nid);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002963
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002964 rdmsr_on_cpus(cmask, MSR_IA32_MCG_CTL, msrs);
2965
2966 for_each_cpu(cpu, cmask) {
2967
Borislav Petkov50542252009-12-11 18:14:40 +01002968 struct msr *reg = per_cpu_ptr(msrs, cpu);
2969
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002970 if (on) {
Borislav Petkov5980bb92011-01-07 16:26:49 +01002971 if (reg->l & MSR_MCGCTL_NBE)
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002972 s->flags.nb_mce_enable = 1;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002973
Borislav Petkov5980bb92011-01-07 16:26:49 +01002974 reg->l |= MSR_MCGCTL_NBE;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002975 } else {
2976 /*
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01002977 * Turn off NB MCE reporting only when it was off before
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002978 */
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002979 if (!s->flags.nb_mce_enable)
Borislav Petkov5980bb92011-01-07 16:26:49 +01002980 reg->l &= ~MSR_MCGCTL_NBE;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002981 }
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002982 }
2983 wrmsr_on_cpus(cmask, MSR_IA32_MCG_CTL, msrs);
2984
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002985 free_cpumask_var(cmask);
2986
2987 return 0;
2988}
2989
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002990static bool enable_ecc_error_reporting(struct ecc_settings *s, u16 nid,
Borislav Petkov2299ef72010-10-15 17:44:04 +02002991 struct pci_dev *F3)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002992{
Borislav Petkov2299ef72010-10-15 17:44:04 +02002993 bool ret = true;
Borislav Petkovc9f4f262010-12-22 19:48:20 +01002994 u32 value, mask = 0x3; /* UECC/CECC enable */
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002995
Borislav Petkov2299ef72010-10-15 17:44:04 +02002996 if (toggle_ecc_err_reporting(s, nid, ON)) {
2997 amd64_warn("Error enabling ECC reporting over MCGCTL!\n");
2998 return false;
2999 }
3000
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003001 amd64_read_pci_cfg(F3, NBCTL, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003002
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003003 s->old_nbctl = value & mask;
3004 s->nbctl_valid = true;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003005
3006 value |= mask;
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003007 amd64_write_pci_cfg(F3, NBCTL, value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003008
Borislav Petkova97fa682010-12-23 14:07:18 +01003009 amd64_read_pci_cfg(F3, NBCFG, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003010
Joe Perches956b9ba12012-04-29 17:08:39 -03003011 edac_dbg(0, "1: node %d, NBCFG=0x%08x[DramEccEn: %d]\n",
3012 nid, value, !!(value & NBCFG_ECC_ENABLE));
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003013
Borislav Petkova97fa682010-12-23 14:07:18 +01003014 if (!(value & NBCFG_ECC_ENABLE)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003015 amd64_warn("DRAM ECC disabled on this node, enabling...\n");
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003016
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003017 s->flags.nb_ecc_prev = 0;
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01003018
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003019 /* Attempt to turn on DRAM ECC Enable */
Borislav Petkova97fa682010-12-23 14:07:18 +01003020 value |= NBCFG_ECC_ENABLE;
3021 amd64_write_pci_cfg(F3, NBCFG, value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003022
Borislav Petkova97fa682010-12-23 14:07:18 +01003023 amd64_read_pci_cfg(F3, NBCFG, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003024
Borislav Petkova97fa682010-12-23 14:07:18 +01003025 if (!(value & NBCFG_ECC_ENABLE)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003026 amd64_warn("Hardware rejected DRAM ECC enable,"
3027 "check memory DIMM configuration.\n");
Borislav Petkov2299ef72010-10-15 17:44:04 +02003028 ret = false;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003029 } else {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003030 amd64_info("Hardware accepted DRAM ECC Enable\n");
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003031 }
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01003032 } else {
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003033 s->flags.nb_ecc_prev = 1;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003034 }
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01003035
Joe Perches956b9ba12012-04-29 17:08:39 -03003036 edac_dbg(0, "2: node %d, NBCFG=0x%08x[DramEccEn: %d]\n",
3037 nid, value, !!(value & NBCFG_ECC_ENABLE));
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003038
Borislav Petkov2299ef72010-10-15 17:44:04 +02003039 return ret;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003040}
3041
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08003042static void restore_ecc_error_reporting(struct ecc_settings *s, u16 nid,
Borislav Petkov360b7f32010-10-15 19:25:38 +02003043 struct pci_dev *F3)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003044{
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003045 u32 value, mask = 0x3; /* UECC/CECC enable */
3046
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003047 if (!s->nbctl_valid)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003048 return;
3049
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003050 amd64_read_pci_cfg(F3, NBCTL, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003051 value &= ~mask;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003052 value |= s->old_nbctl;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003053
Borislav Petkovc9f4f262010-12-22 19:48:20 +01003054 amd64_write_pci_cfg(F3, NBCTL, value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003055
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003056 /* restore previous BIOS DRAM ECC "off" setting we force-enabled */
3057 if (!s->flags.nb_ecc_prev) {
Borislav Petkova97fa682010-12-23 14:07:18 +01003058 amd64_read_pci_cfg(F3, NBCFG, &value);
3059 value &= ~NBCFG_ECC_ENABLE;
3060 amd64_write_pci_cfg(F3, NBCFG, value);
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01003061 }
3062
3063 /* restore the NB Enable MCGCTL bit */
Borislav Petkov2299ef72010-10-15 17:44:04 +02003064 if (toggle_ecc_err_reporting(s, nid, OFF))
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003065 amd64_warn("Error restoring NB MCGCTL settings!\n");
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01003066}
3067
Doug Thompsonf9431992009-04-27 19:46:08 +02003068/*
Borislav Petkov2299ef72010-10-15 17:44:04 +02003069 * EDAC requires that the BIOS have ECC enabled before
3070 * taking over the processing of ECC errors. A command line
3071 * option allows to force-enable hardware ECC later in
3072 * enable_ecc_error_reporting().
Doug Thompsonf9431992009-04-27 19:46:08 +02003073 */
Borislav Petkovcab4d272010-02-11 17:15:57 +01003074static const char *ecc_msg =
3075 "ECC disabled in the BIOS or no ECC capability, module will not load.\n"
3076 " Either enable ECC checking or force module loading by setting "
3077 "'ecc_enable_override'.\n"
3078 " (Note that use of the override may cause unknown side effects.)\n";
Borislav Petkovbe3468e2009-08-05 15:47:22 +02003079
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08003080static bool ecc_enabled(struct pci_dev *F3, u16 nid)
Doug Thompsonf9431992009-04-27 19:46:08 +02003081{
Borislav Petkov06724532009-09-16 13:05:46 +02003082 bool nb_mce_en = false;
Yazen Ghannam196b79f2016-11-17 17:57:34 -05003083 u8 ecc_en = 0, i;
3084 u32 value;
Doug Thompsonf9431992009-04-27 19:46:08 +02003085
Yazen Ghannam196b79f2016-11-17 17:57:34 -05003086 if (boot_cpu_data.x86 >= 0x17) {
3087 u8 umc_en_mask = 0, ecc_en_mask = 0;
Doug Thompsonf9431992009-04-27 19:46:08 +02003088
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00003089 for_each_umc(i) {
Yazen Ghannam196b79f2016-11-17 17:57:34 -05003090 u32 base = get_umc_base(i);
3091
3092 /* Only check enabled UMCs. */
3093 if (amd_smn_read(nid, base + UMCCH_SDP_CTRL, &value))
3094 continue;
3095
3096 if (!(value & UMC_SDP_INIT))
3097 continue;
3098
3099 umc_en_mask |= BIT(i);
3100
3101 if (amd_smn_read(nid, base + UMCCH_UMC_CAP_HI, &value))
3102 continue;
3103
3104 if (value & UMC_ECC_ENABLED)
3105 ecc_en_mask |= BIT(i);
3106 }
3107
3108 /* Check whether at least one UMC is enabled: */
3109 if (umc_en_mask)
3110 ecc_en = umc_en_mask == ecc_en_mask;
Yazen Ghannam11ab1ca2017-01-27 11:24:19 -06003111 else
3112 edac_dbg(0, "Node %d: No enabled UMCs.\n", nid);
Yazen Ghannam196b79f2016-11-17 17:57:34 -05003113
3114 /* Assume UMC MCA banks are enabled. */
3115 nb_mce_en = true;
3116 } else {
3117 amd64_read_pci_cfg(F3, NBCFG, &value);
3118
3119 ecc_en = !!(value & NBCFG_ECC_ENABLE);
3120
3121 nb_mce_en = nb_mce_bank_enabled_on_node(nid);
3122 if (!nb_mce_en)
Yazen Ghannam11ab1ca2017-01-27 11:24:19 -06003123 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 -05003124 MSR_IA32_MCG_CTL, nid);
3125 }
3126
Yazen Ghannam11ab1ca2017-01-27 11:24:19 -06003127 amd64_info("Node %d: DRAM ECC %s.\n",
3128 nid, (ecc_en ? "enabled" : "disabled"));
Doug Thompsonf9431992009-04-27 19:46:08 +02003129
Borislav Petkov2299ef72010-10-15 17:44:04 +02003130 if (!ecc_en || !nb_mce_en) {
Yazen Ghannam11ab1ca2017-01-27 11:24:19 -06003131 amd64_info("%s", ecc_msg);
Borislav Petkov2299ef72010-10-15 17:44:04 +02003132 return false;
Borislav Petkov43f5e682009-12-21 18:55:18 +01003133 }
Borislav Petkov2299ef72010-10-15 17:44:04 +02003134 return true;
Doug Thompsonf9431992009-04-27 19:46:08 +02003135}
3136
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003137static inline void
3138f17h_determine_edac_ctl_cap(struct mem_ctl_info *mci, struct amd64_pvt *pvt)
3139{
3140 u8 i, ecc_en = 1, cpk_en = 1;
3141
Yazen Ghannam4d30d2b2019-02-28 15:36:10 +00003142 for_each_umc(i) {
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003143 if (pvt->umc[i].sdp_ctrl & UMC_SDP_INIT) {
3144 ecc_en &= !!(pvt->umc[i].umc_cap_hi & UMC_ECC_ENABLED);
3145 cpk_en &= !!(pvt->umc[i].umc_cap_hi & UMC_ECC_CHIPKILL_CAP);
3146 }
3147 }
3148
3149 /* Set chipkill only if ECC is enabled: */
3150 if (ecc_en) {
3151 mci->edac_ctl_cap |= EDAC_FLAG_SECDED;
3152
3153 if (cpk_en)
3154 mci->edac_ctl_cap |= EDAC_FLAG_S4ECD4ED;
3155 }
3156}
3157
Borislav Petkovdf71a052011-01-19 18:15:10 +01003158static void setup_mci_misc_attrs(struct mem_ctl_info *mci,
3159 struct amd64_family_type *fam)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003160{
3161 struct amd64_pvt *pvt = mci->pvt_info;
3162
3163 mci->mtype_cap = MEM_FLAG_DDR2 | MEM_FLAG_RDDR2;
3164 mci->edac_ctl_cap = EDAC_FLAG_NONE;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003165
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003166 if (pvt->umc) {
3167 f17h_determine_edac_ctl_cap(mci, pvt);
3168 } else {
3169 if (pvt->nbcap & NBCAP_SECDED)
3170 mci->edac_ctl_cap |= EDAC_FLAG_SECDED;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003171
Yazen Ghannam2d09d8f2016-11-29 08:51:56 -06003172 if (pvt->nbcap & NBCAP_CHIPKILL)
3173 mci->edac_ctl_cap |= EDAC_FLAG_S4ECD4ED;
3174 }
Doug Thompson7d6034d2009-04-27 20:01:01 +02003175
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003176 mci->edac_cap = determine_edac_cap(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003177 mci->mod_name = EDAC_MOD_STR;
Borislav Petkovdf71a052011-01-19 18:15:10 +01003178 mci->ctl_name = fam->ctl_name;
Yazen Ghanname7934b72016-11-17 17:57:30 -05003179 mci->dev_name = pci_name(pvt->F3);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003180 mci->ctl_page_to_phys = NULL;
3181
Doug Thompson7d6034d2009-04-27 20:01:01 +02003182 /* memory scrubber interface */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003183 mci->set_sdram_scrub_rate = set_scrub_rate;
3184 mci->get_sdram_scrub_rate = get_scrub_rate;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003185}
3186
Borislav Petkov0092b202010-10-01 19:20:05 +02003187/*
3188 * returns a pointer to the family descriptor on success, NULL otherwise.
3189 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003190static struct amd64_family_type *per_family_init(struct amd64_pvt *pvt)
Borislav Petkov395ae782010-10-01 18:38:19 +02003191{
Borislav Petkov0092b202010-10-01 19:20:05 +02003192 struct amd64_family_type *fam_type = NULL;
3193
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003194 pvt->ext_model = boot_cpu_data.x86_model >> 4;
Jia Zhangb3991512018-01-01 09:52:10 +08003195 pvt->stepping = boot_cpu_data.x86_stepping;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003196 pvt->model = boot_cpu_data.x86_model;
3197 pvt->fam = boot_cpu_data.x86;
3198
3199 switch (pvt->fam) {
Borislav Petkov395ae782010-10-01 18:38:19 +02003200 case 0xf:
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003201 fam_type = &family_types[K8_CPUS];
3202 pvt->ops = &family_types[K8_CPUS].ops;
Borislav Petkov395ae782010-10-01 18:38:19 +02003203 break;
Borislav Petkovdf71a052011-01-19 18:15:10 +01003204
Borislav Petkov395ae782010-10-01 18:38:19 +02003205 case 0x10:
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003206 fam_type = &family_types[F10_CPUS];
3207 pvt->ops = &family_types[F10_CPUS].ops;
Borislav Petkovdf71a052011-01-19 18:15:10 +01003208 break;
3209
3210 case 0x15:
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003211 if (pvt->model == 0x30) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003212 fam_type = &family_types[F15_M30H_CPUS];
3213 pvt->ops = &family_types[F15_M30H_CPUS].ops;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003214 break;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01003215 } else if (pvt->model == 0x60) {
3216 fam_type = &family_types[F15_M60H_CPUS];
3217 pvt->ops = &family_types[F15_M60H_CPUS].ops;
3218 break;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003219 }
3220
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003221 fam_type = &family_types[F15_CPUS];
3222 pvt->ops = &family_types[F15_CPUS].ops;
Borislav Petkov395ae782010-10-01 18:38:19 +02003223 break;
3224
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05003225 case 0x16:
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06003226 if (pvt->model == 0x30) {
3227 fam_type = &family_types[F16_M30H_CPUS];
3228 pvt->ops = &family_types[F16_M30H_CPUS].ops;
3229 break;
3230 }
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003231 fam_type = &family_types[F16_CPUS];
3232 pvt->ops = &family_types[F16_CPUS].ops;
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05003233 break;
3234
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05003235 case 0x17:
Michael Jin8960de42018-08-16 15:28:40 -04003236 if (pvt->model >= 0x10 && pvt->model <= 0x2f) {
3237 fam_type = &family_types[F17_M10H_CPUS];
3238 pvt->ops = &family_types[F17_M10H_CPUS].ops;
3239 break;
Yazen Ghannam6e8462392019-02-28 15:36:09 +00003240 } else if (pvt->model >= 0x30 && pvt->model <= 0x3f) {
3241 fam_type = &family_types[F17_M30H_CPUS];
3242 pvt->ops = &family_types[F17_M30H_CPUS].ops;
3243 break;
Michael Jin8960de42018-08-16 15:28:40 -04003244 }
Pu Wenc4a3e942018-09-27 16:31:28 +02003245 /* fall through */
3246 case 0x18:
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05003247 fam_type = &family_types[F17_CPUS];
3248 pvt->ops = &family_types[F17_CPUS].ops;
Pu Wenc4a3e942018-09-27 16:31:28 +02003249
3250 if (pvt->fam == 0x18)
3251 family_types[F17_CPUS].ctl_name = "F18h";
Yazen Ghannamf1cbbec2016-11-17 17:57:35 -05003252 break;
3253
Borislav Petkov395ae782010-10-01 18:38:19 +02003254 default:
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003255 amd64_err("Unsupported family!\n");
Borislav Petkov0092b202010-10-01 19:20:05 +02003256 return NULL;
Borislav Petkov395ae782010-10-01 18:38:19 +02003257 }
Borislav Petkov0092b202010-10-01 19:20:05 +02003258
Borislav Petkovdf71a052011-01-19 18:15:10 +01003259 amd64_info("%s %sdetected (node %d).\n", fam_type->ctl_name,
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05003260 (pvt->fam == 0xf ?
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02003261 (pvt->ext_model >= K8_REV_F ? "revF or later "
3262 : "revE or earlier ")
3263 : ""), pvt->mc_node_id);
Borislav Petkov0092b202010-10-01 19:20:05 +02003264 return fam_type;
Borislav Petkov395ae782010-10-01 18:38:19 +02003265}
3266
Takashi Iwaie339f1e2015-02-04 11:48:53 +01003267static const struct attribute_group *amd64_edac_attr_groups[] = {
3268#ifdef CONFIG_EDAC_DEBUG
3269 &amd64_edac_dbg_group,
3270#endif
3271#ifdef CONFIG_EDAC_AMD64_ERROR_INJECTION
3272 &amd64_edac_inj_group,
3273#endif
3274 NULL
3275};
3276
Yazen Ghannambdcee772019-02-28 15:36:10 +00003277/* Set the number of Unified Memory Controllers in the system. */
3278static void compute_num_umcs(void)
3279{
3280 u8 model = boot_cpu_data.x86_model;
3281
3282 if (boot_cpu_data.x86 < 0x17)
3283 return;
3284
3285 if (model >= 0x30 && model <= 0x3f)
3286 num_umcs = 8;
3287 else
3288 num_umcs = 2;
3289
3290 edac_dbg(1, "Number of UMCs: %x", num_umcs);
3291}
3292
Borislav Petkov3f37a362016-05-06 19:44:27 +02003293static int init_one_instance(unsigned int nid)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003294{
Borislav Petkov3f37a362016-05-06 19:44:27 +02003295 struct pci_dev *F3 = node_to_amd_nb(nid)->misc;
Borislav Petkov0092b202010-10-01 19:20:05 +02003296 struct amd64_family_type *fam_type = NULL;
Borislav Petkov360b7f32010-10-15 19:25:38 +02003297 struct mem_ctl_info *mci = NULL;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03003298 struct edac_mc_layer layers[2];
Borislav Petkov3f37a362016-05-06 19:44:27 +02003299 struct amd64_pvt *pvt = NULL;
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003300 u16 pci_id1, pci_id2;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003301 int err = 0, ret;
3302
3303 ret = -ENOMEM;
3304 pvt = kzalloc(sizeof(struct amd64_pvt), GFP_KERNEL);
3305 if (!pvt)
Borislav Petkov360b7f32010-10-15 19:25:38 +02003306 goto err_ret;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003307
Borislav Petkov360b7f32010-10-15 19:25:38 +02003308 pvt->mc_node_id = nid;
Borislav Petkov3f37a362016-05-06 19:44:27 +02003309 pvt->F3 = F3;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003310
Borislav Petkov395ae782010-10-01 18:38:19 +02003311 ret = -EINVAL;
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003312 fam_type = per_family_init(pvt);
Borislav Petkov0092b202010-10-01 19:20:05 +02003313 if (!fam_type)
Borislav Petkov395ae782010-10-01 18:38:19 +02003314 goto err_free;
3315
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003316 if (pvt->fam >= 0x17) {
Yazen Ghannambdcee772019-02-28 15:36:10 +00003317 pvt->umc = kcalloc(num_umcs, sizeof(struct amd64_umc), GFP_KERNEL);
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003318 if (!pvt->umc) {
3319 ret = -ENOMEM;
3320 goto err_free;
3321 }
3322
3323 pci_id1 = fam_type->f0_id;
3324 pci_id2 = fam_type->f6_id;
3325 } else {
3326 pci_id1 = fam_type->f1_id;
3327 pci_id2 = fam_type->f2_id;
3328 }
3329
3330 err = reserve_mc_sibling_devs(pvt, pci_id1, pci_id2);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003331 if (err)
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003332 goto err_post_init;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003333
Borislav Petkov360b7f32010-10-15 19:25:38 +02003334 read_mc_regs(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003335
Doug Thompson7d6034d2009-04-27 20:01:01 +02003336 /*
3337 * We need to determine how many memory channels there are. Then use
3338 * that information for calculating the size of the dynamic instance
Borislav Petkov360b7f32010-10-15 19:25:38 +02003339 * tables in the 'mci' structure.
Doug Thompson7d6034d2009-04-27 20:01:01 +02003340 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02003341 ret = -EINVAL;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003342 pvt->channel_count = pvt->ops->early_channel_count(pvt);
3343 if (pvt->channel_count < 0)
Borislav Petkov360b7f32010-10-15 19:25:38 +02003344 goto err_siblings;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003345
3346 ret = -ENOMEM;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03003347 layers[0].type = EDAC_MC_LAYER_CHIP_SELECT;
3348 layers[0].size = pvt->csels[0].b_cnt;
3349 layers[0].is_virt_csrow = true;
3350 layers[1].type = EDAC_MC_LAYER_CHANNEL;
Borislav Petkovf0a56c42013-07-23 20:01:23 +02003351
3352 /*
3353 * Always allocate two channels since we can have setups with DIMMs on
3354 * only one channel. Also, this simplifies handling later for the price
3355 * of a couple of KBs tops.
Yazen Ghannam869adc42019-03-25 20:33:30 +00003356 *
3357 * On Fam17h+, the number of controllers may be greater than two. So set
3358 * the size equal to the maximum number of UMCs.
Borislav Petkovf0a56c42013-07-23 20:01:23 +02003359 */
Yazen Ghannam869adc42019-03-25 20:33:30 +00003360 if (pvt->fam >= 0x17)
3361 layers[1].size = num_umcs;
3362 else
3363 layers[1].size = 2;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03003364 layers[1].is_virt_csrow = false;
Borislav Petkovf0a56c42013-07-23 20:01:23 +02003365
Mauro Carvalho Chehabca0907b2012-05-02 14:37:00 -03003366 mci = edac_mc_alloc(nid, ARRAY_SIZE(layers), layers, 0);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003367 if (!mci)
Borislav Petkov360b7f32010-10-15 19:25:38 +02003368 goto err_siblings;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003369
3370 mci->pvt_info = pvt;
Borislav Petkov3f37a362016-05-06 19:44:27 +02003371 mci->pdev = &pvt->F3->dev;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003372
Borislav Petkovdf71a052011-01-19 18:15:10 +01003373 setup_mci_misc_attrs(mci, fam_type);
Borislav Petkov360b7f32010-10-15 19:25:38 +02003374
3375 if (init_csrows(mci))
Doug Thompson7d6034d2009-04-27 20:01:01 +02003376 mci->edac_cap = EDAC_FLAG_NONE;
3377
Doug Thompson7d6034d2009-04-27 20:01:01 +02003378 ret = -ENODEV;
Takashi Iwaie339f1e2015-02-04 11:48:53 +01003379 if (edac_mc_add_mc_with_groups(mci, amd64_edac_attr_groups)) {
Joe Perches956b9ba12012-04-29 17:08:39 -03003380 edac_dbg(1, "failed edac_mc_add_mc()\n");
Doug Thompson7d6034d2009-04-27 20:01:01 +02003381 goto err_add_mc;
3382 }
3383
Doug Thompson7d6034d2009-04-27 20:01:01 +02003384 return 0;
3385
3386err_add_mc:
3387 edac_mc_free(mci);
3388
Borislav Petkov360b7f32010-10-15 19:25:38 +02003389err_siblings:
3390 free_mc_sibling_devs(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003391
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003392err_post_init:
3393 if (pvt->fam >= 0x17)
3394 kfree(pvt->umc);
3395
Borislav Petkov360b7f32010-10-15 19:25:38 +02003396err_free:
3397 kfree(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003398
Borislav Petkov360b7f32010-10-15 19:25:38 +02003399err_ret:
Doug Thompson7d6034d2009-04-27 20:01:01 +02003400 return ret;
3401}
3402
Borislav Petkov3f37a362016-05-06 19:44:27 +02003403static int probe_one_instance(unsigned int nid)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003404{
Borislav Petkov2299ef72010-10-15 17:44:04 +02003405 struct pci_dev *F3 = node_to_amd_nb(nid)->misc;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003406 struct ecc_settings *s;
Borislav Petkov3f37a362016-05-06 19:44:27 +02003407 int ret;
Borislav Petkovb8cfa022010-10-01 19:35:38 +02003408
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003409 ret = -ENOMEM;
3410 s = kzalloc(sizeof(struct ecc_settings), GFP_KERNEL);
3411 if (!s)
Borislav Petkov2299ef72010-10-15 17:44:04 +02003412 goto err_out;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003413
3414 ecc_stngs[nid] = s;
3415
Borislav Petkov2299ef72010-10-15 17:44:04 +02003416 if (!ecc_enabled(F3, nid)) {
Yazen Ghannam4688c9b2017-01-27 11:24:22 -06003417 ret = 0;
Borislav Petkov2299ef72010-10-15 17:44:04 +02003418
3419 if (!ecc_enable_override)
3420 goto err_enable;
3421
Yazen Ghannam044e7a42016-11-22 15:40:16 -06003422 if (boot_cpu_data.x86 >= 0x17) {
3423 amd64_warn("Forcing ECC on is not recommended on newer systems. Please enable ECC in BIOS.");
3424 goto err_enable;
3425 } else
3426 amd64_warn("Forcing ECC on!\n");
Borislav Petkov2299ef72010-10-15 17:44:04 +02003427
3428 if (!enable_ecc_error_reporting(s, nid, F3))
3429 goto err_enable;
3430 }
3431
Borislav Petkov3f37a362016-05-06 19:44:27 +02003432 ret = init_one_instance(nid);
Borislav Petkov360b7f32010-10-15 19:25:38 +02003433 if (ret < 0) {
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003434 amd64_err("Error probing instance: %d\n", nid);
Yazen Ghannam044e7a42016-11-22 15:40:16 -06003435
3436 if (boot_cpu_data.x86 < 0x17)
3437 restore_ecc_error_reporting(s, nid, F3);
Yazen Ghannam2b9b2c42017-01-24 16:32:24 -06003438
3439 goto err_enable;
Borislav Petkov360b7f32010-10-15 19:25:38 +02003440 }
Doug Thompson7d6034d2009-04-27 20:01:01 +02003441
3442 return ret;
Borislav Petkov2299ef72010-10-15 17:44:04 +02003443
3444err_enable:
3445 kfree(s);
3446 ecc_stngs[nid] = NULL;
3447
3448err_out:
3449 return ret;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003450}
3451
Borislav Petkov3f37a362016-05-06 19:44:27 +02003452static void remove_one_instance(unsigned int nid)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003453{
Borislav Petkov360b7f32010-10-15 19:25:38 +02003454 struct pci_dev *F3 = node_to_amd_nb(nid)->misc;
3455 struct ecc_settings *s = ecc_stngs[nid];
Borislav Petkov3f37a362016-05-06 19:44:27 +02003456 struct mem_ctl_info *mci;
3457 struct amd64_pvt *pvt;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003458
Borislav Petkov3f37a362016-05-06 19:44:27 +02003459 mci = find_mci_by_dev(&F3->dev);
Borislav Petkova4b4bed2013-08-10 13:54:48 +02003460 WARN_ON(!mci);
3461
Doug Thompson7d6034d2009-04-27 20:01:01 +02003462 /* Remove from EDAC CORE tracking list */
Borislav Petkov3f37a362016-05-06 19:44:27 +02003463 mci = edac_mc_del_mc(&F3->dev);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003464 if (!mci)
3465 return;
3466
3467 pvt = mci->pvt_info;
3468
Borislav Petkov360b7f32010-10-15 19:25:38 +02003469 restore_ecc_error_reporting(s, nid, F3);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003470
Borislav Petkov360b7f32010-10-15 19:25:38 +02003471 free_mc_sibling_devs(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003472
Borislav Petkov360b7f32010-10-15 19:25:38 +02003473 kfree(ecc_stngs[nid]);
3474 ecc_stngs[nid] = NULL;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003475
Doug Thompson7d6034d2009-04-27 20:01:01 +02003476 /* Free the EDAC CORE resources */
Borislav Petkov8f68ed92009-12-21 15:15:59 +01003477 mci->pvt_info = NULL;
Borislav Petkov8f68ed92009-12-21 15:15:59 +01003478
3479 kfree(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003480 edac_mc_free(mci);
3481}
3482
Borislav Petkov360b7f32010-10-15 19:25:38 +02003483static void setup_pci_device(void)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003484{
3485 struct mem_ctl_info *mci;
3486 struct amd64_pvt *pvt;
3487
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003488 if (pci_ctl)
Doug Thompson7d6034d2009-04-27 20:01:01 +02003489 return;
3490
Borislav Petkov2ec591a2015-02-17 10:58:34 +01003491 mci = edac_mc_find(0);
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003492 if (!mci)
3493 return;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003494
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003495 pvt = mci->pvt_info;
Yazen Ghannam936fc3a2016-11-17 17:57:36 -05003496 if (pvt->umc)
3497 pci_ctl = edac_pci_create_generic_ctl(&pvt->F0->dev, EDAC_MOD_STR);
3498 else
3499 pci_ctl = edac_pci_create_generic_ctl(&pvt->F2->dev, EDAC_MOD_STR);
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003500 if (!pci_ctl) {
3501 pr_warn("%s(): Unable to create PCI control\n", __func__);
3502 pr_warn("%s(): PCI error report via EDAC not set\n", __func__);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003503 }
3504}
3505
Yazen Ghannamd6efab72016-09-15 19:07:17 -05003506static const struct x86_cpu_id amd64_cpuids[] = {
3507 { X86_VENDOR_AMD, 0xF, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
3508 { X86_VENDOR_AMD, 0x10, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
3509 { X86_VENDOR_AMD, 0x15, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
3510 { X86_VENDOR_AMD, 0x16, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
Yazen Ghannam95d3af62016-11-17 17:57:43 -05003511 { X86_VENDOR_AMD, 0x17, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
Pu Wenc4a3e942018-09-27 16:31:28 +02003512 { X86_VENDOR_HYGON, 0x18, X86_MODEL_ANY, X86_FEATURE_ANY, 0 },
Yazen Ghannamd6efab72016-09-15 19:07:17 -05003513 { }
3514};
3515MODULE_DEVICE_TABLE(x86cpu, amd64_cpuids);
3516
Doug Thompson7d6034d2009-04-27 20:01:01 +02003517static int __init amd64_edac_init(void)
3518{
Toshi Kani301375e2017-08-23 16:54:47 -06003519 const char *owner;
Borislav Petkov360b7f32010-10-15 19:25:38 +02003520 int err = -ENODEV;
Borislav Petkov3f37a362016-05-06 19:44:27 +02003521 int i;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003522
Toshi Kani301375e2017-08-23 16:54:47 -06003523 owner = edac_get_owner();
3524 if (owner && strncmp(owner, EDAC_MOD_STR, sizeof(EDAC_MOD_STR)))
3525 return -EBUSY;
3526
Yazen Ghannam1bd99002017-01-27 11:24:23 -06003527 if (!x86_match_cpu(amd64_cpuids))
3528 return -ENODEV;
3529
Hans Rosenfeld9653a5c2010-10-29 17:14:31 +02003530 if (amd_cache_northbridges() < 0)
Yazen Ghannam1bd99002017-01-27 11:24:23 -06003531 return -ENODEV;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003532
Borislav Petkov6ba92fe2016-06-16 01:13:18 +02003533 opstate_init();
3534
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003535 err = -ENOMEM;
Kees Cook6396bb22018-06-12 14:03:40 -07003536 ecc_stngs = kcalloc(amd_nb_num(), sizeof(ecc_stngs[0]), GFP_KERNEL);
Borislav Petkov2ec591a2015-02-17 10:58:34 +01003537 if (!ecc_stngs)
Borislav Petkova9f0fbe2011-03-29 18:10:53 +02003538 goto err_free;
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003539
Borislav Petkov50542252009-12-11 18:14:40 +01003540 msrs = msrs_alloc();
Borislav Petkov56b34b92009-12-21 18:13:01 +01003541 if (!msrs)
Borislav Petkov360b7f32010-10-15 19:25:38 +02003542 goto err_free;
Borislav Petkov50542252009-12-11 18:14:40 +01003543
Yazen Ghannambdcee772019-02-28 15:36:10 +00003544 compute_num_umcs();
3545
Yazen Ghannam2287c632017-01-13 09:52:19 -06003546 for (i = 0; i < amd_nb_num(); i++) {
3547 err = probe_one_instance(i);
3548 if (err) {
Borislav Petkov3f37a362016-05-06 19:44:27 +02003549 /* unwind properly */
3550 while (--i >= 0)
3551 remove_one_instance(i);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003552
Borislav Petkov3f37a362016-05-06 19:44:27 +02003553 goto err_pci;
3554 }
Yazen Ghannam2287c632017-01-13 09:52:19 -06003555 }
Doug Thompson7d6034d2009-04-27 20:01:01 +02003556
Yazen Ghannam4688c9b2017-01-27 11:24:22 -06003557 if (!edac_has_mcs()) {
3558 err = -ENODEV;
3559 goto err_pci;
3560 }
3561
Yazen Ghannam234365f2017-01-24 16:32:25 -06003562 /* register stuff with EDAC MCE */
3563 if (report_gart_errors)
3564 amd_report_gart_errors(true);
3565
3566 if (boot_cpu_data.x86 >= 0x17)
3567 amd_register_ecc_decoder(decode_umc_error);
3568 else
3569 amd_register_ecc_decoder(decode_bus_error);
3570
Borislav Petkov360b7f32010-10-15 19:25:38 +02003571 setup_pci_device();
Tomasz Palaf5b10c42014-11-02 11:22:12 +01003572
3573#ifdef CONFIG_X86_32
3574 amd64_err("%s on 32-bit is unsupported. USE AT YOUR OWN RISK!\n", EDAC_MOD_STR);
3575#endif
3576
Borislav Petkovde0336b2016-04-27 12:21:21 +02003577 printk(KERN_INFO "AMD64 EDAC driver v%s\n", EDAC_AMD64_VERSION);
3578
Borislav Petkov360b7f32010-10-15 19:25:38 +02003579 return 0;
Borislav Petkov56b34b92009-12-21 18:13:01 +01003580
Borislav Petkov56b34b92009-12-21 18:13:01 +01003581err_pci:
3582 msrs_free(msrs);
3583 msrs = NULL;
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003584
Borislav Petkov360b7f32010-10-15 19:25:38 +02003585err_free:
Borislav Petkov360b7f32010-10-15 19:25:38 +02003586 kfree(ecc_stngs);
3587 ecc_stngs = NULL;
3588
Doug Thompson7d6034d2009-04-27 20:01:01 +02003589 return err;
3590}
3591
3592static void __exit amd64_edac_exit(void)
3593{
Borislav Petkov3f37a362016-05-06 19:44:27 +02003594 int i;
3595
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003596 if (pci_ctl)
3597 edac_pci_release_generic_ctl(pci_ctl);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003598
Yazen Ghannam234365f2017-01-24 16:32:25 -06003599 /* unregister from EDAC MCE */
3600 amd_report_gart_errors(false);
3601
3602 if (boot_cpu_data.x86 >= 0x17)
3603 amd_unregister_ecc_decoder(decode_umc_error);
3604 else
3605 amd_unregister_ecc_decoder(decode_bus_error);
3606
Borislav Petkov3f37a362016-05-06 19:44:27 +02003607 for (i = 0; i < amd_nb_num(); i++)
3608 remove_one_instance(i);
Borislav Petkov50542252009-12-11 18:14:40 +01003609
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003610 kfree(ecc_stngs);
3611 ecc_stngs = NULL;
3612
Borislav Petkov50542252009-12-11 18:14:40 +01003613 msrs_free(msrs);
3614 msrs = NULL;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003615}
3616
3617module_init(amd64_edac_init);
3618module_exit(amd64_edac_exit);
3619
3620MODULE_LICENSE("GPL");
3621MODULE_AUTHOR("SoftwareBitMaker: Doug Thompson, "
3622 "Dave Peterson, Thayne Harbaugh");
3623MODULE_DESCRIPTION("MC support for AMD64 memory controllers - "
3624 EDAC_AMD64_VERSION);
3625
3626module_param(edac_op_state, int, 0444);
3627MODULE_PARM_DESC(edac_op_state, "EDAC Error Reporting state: 0=Poll,1=NMI");