blob: 73aea40a9c89592035e08ecf98292dc9aab9df62 [file] [log] [blame]
Doug Thompson2bc65412009-05-04 20:11:14 +02001#include "amd64_edac.h"
Andreas Herrmann23ac4ae2010-09-17 18:03:43 +02002#include <asm/amd_nb.h>
Doug Thompson2bc65412009-05-04 20:11:14 +02003
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01004static struct edac_pci_ctl_info *pci_ctl;
Doug Thompson2bc65412009-05-04 20:11:14 +02005
6static int report_gart_errors;
7module_param(report_gart_errors, int, 0644);
8
9/*
10 * Set by command line parameter. If BIOS has enabled the ECC, this override is
11 * cleared to prevent re-enabling the hardware by this driver.
12 */
13static int ecc_enable_override;
14module_param(ecc_enable_override, int, 0644);
15
Tejun Heoa29d8b82010-02-02 14:39:15 +090016static struct msr __percpu *msrs;
Borislav Petkov50542252009-12-11 18:14:40 +010017
Borislav Petkov360b7f32010-10-15 19:25:38 +020018/*
19 * count successfully initialized driver instances for setup_pci_device()
20 */
21static atomic_t drv_instances = ATOMIC_INIT(0);
22
Borislav Petkov2ec591a2015-02-17 10:58:34 +010023/* Per-node stuff */
Borislav Petkovae7bb7c2010-10-14 16:01:30 +020024static struct ecc_settings **ecc_stngs;
Doug Thompson2bc65412009-05-04 20:11:14 +020025
26/*
Borislav Petkovb70ef012009-06-25 19:32:38 +020027 * Valid scrub rates for the K8 hardware memory scrubber. We map the scrubbing
28 * bandwidth to a valid bit pattern. The 'set' operation finds the 'matching-
29 * or higher value'.
30 *
31 *FIXME: Produce a better mapping/linearisation.
32 */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +080033static const struct scrubrate {
Borislav Petkov39094442010-11-24 19:52:09 +010034 u32 scrubval; /* bit pattern for scrub rate */
35 u32 bandwidth; /* bandwidth consumed (bytes/sec) */
36} scrubrates[] = {
Borislav Petkovb70ef012009-06-25 19:32:38 +020037 { 0x01, 1600000000UL},
38 { 0x02, 800000000UL},
39 { 0x03, 400000000UL},
40 { 0x04, 200000000UL},
41 { 0x05, 100000000UL},
42 { 0x06, 50000000UL},
43 { 0x07, 25000000UL},
44 { 0x08, 12284069UL},
45 { 0x09, 6274509UL},
46 { 0x0A, 3121951UL},
47 { 0x0B, 1560975UL},
48 { 0x0C, 781440UL},
49 { 0x0D, 390720UL},
50 { 0x0E, 195300UL},
51 { 0x0F, 97650UL},
52 { 0x10, 48854UL},
53 { 0x11, 24427UL},
54 { 0x12, 12213UL},
55 { 0x13, 6101UL},
56 { 0x14, 3051UL},
57 { 0x15, 1523UL},
58 { 0x16, 761UL},
59 { 0x00, 0UL}, /* scrubbing off */
60};
61
Borislav Petkov66fed2d2012-08-09 18:41:07 +020062int __amd64_read_pci_cfg_dword(struct pci_dev *pdev, int offset,
63 u32 *val, const char *func)
Borislav Petkovb2b0c602010-10-08 18:32:29 +020064{
65 int err = 0;
66
67 err = pci_read_config_dword(pdev, offset, val);
68 if (err)
69 amd64_warn("%s: error reading F%dx%03x.\n",
70 func, PCI_FUNC(pdev->devfn), offset);
71
72 return err;
73}
74
75int __amd64_write_pci_cfg_dword(struct pci_dev *pdev, int offset,
76 u32 val, const char *func)
77{
78 int err = 0;
79
80 err = pci_write_config_dword(pdev, offset, val);
81 if (err)
82 amd64_warn("%s: error writing to F%dx%03x.\n",
83 func, PCI_FUNC(pdev->devfn), offset);
84
85 return err;
86}
87
88/*
Borislav Petkov73ba8592011-09-19 17:34:45 +020089 * Select DCT to which PCI cfg accesses are routed
90 */
91static void f15h_select_dct(struct amd64_pvt *pvt, u8 dct)
92{
93 u32 reg = 0;
94
95 amd64_read_pci_cfg(pvt->F1, DCT_CFG_SEL, &reg);
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -050096 reg &= (pvt->model == 0x30) ? ~3 : ~1;
Borislav Petkov73ba8592011-09-19 17:34:45 +020097 reg |= dct;
98 amd64_write_pci_cfg(pvt->F1, DCT_CFG_SEL, reg);
99}
100
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500101/*
102 *
103 * Depending on the family, F2 DCT reads need special handling:
104 *
105 * K8: has a single DCT only and no address offsets >= 0x100
106 *
107 * F10h: each DCT has its own set of regs
108 * DCT0 -> F2x040..
109 * DCT1 -> F2x140..
110 *
111 * F16h: has only 1 DCT
112 *
113 * F15h: we select which DCT we access using F1x10C[DctCfgSel]
114 */
115static inline int amd64_read_dct_pci_cfg(struct amd64_pvt *pvt, u8 dct,
116 int offset, u32 *val)
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200117{
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500118 switch (pvt->fam) {
119 case 0xf:
120 if (dct || offset >= 0x100)
121 return -EINVAL;
122 break;
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200123
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500124 case 0x10:
125 if (dct) {
126 /*
127 * Note: If ganging is enabled, barring the regs
128 * F2x[1,0]98 and F2x[1,0]9C; reads reads to F2x1xx
129 * return 0. (cf. Section 2.8.1 F10h BKDG)
130 */
131 if (dct_ganging_enabled(pvt))
132 return 0;
133
134 offset += 0x100;
135 }
136 break;
137
138 case 0x15:
139 /*
140 * F15h: F2x1xx addresses do not map explicitly to DCT1.
141 * We should select which DCT we access using F1x10C[DctCfgSel]
142 */
143 dct = (dct && pvt->model == 0x30) ? 3 : dct;
144 f15h_select_dct(pvt, dct);
145 break;
146
147 case 0x16:
148 if (dct)
149 return -EINVAL;
150 break;
151
152 default:
153 break;
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200154 }
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500155 return amd64_read_pci_cfg(pvt->F2, offset, val);
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200156}
157
Borislav Petkovb70ef012009-06-25 19:32:38 +0200158/*
Doug Thompson2bc65412009-05-04 20:11:14 +0200159 * Memory scrubber control interface. For K8, memory scrubbing is handled by
160 * hardware and can involve L2 cache, dcache as well as the main memory. With
161 * F10, this is extended to L3 cache scrubbing on CPU models sporting that
162 * functionality.
163 *
164 * This causes the "units" for the scrubbing speed to vary from 64 byte blocks
165 * (dram) over to cache lines. This is nasty, so we will use bandwidth in
166 * bytes/sec for the setting.
167 *
168 * Currently, we only do dram scrubbing. If the scrubbing is done in software on
169 * other archs, we might not have access to the caches directly.
170 */
171
172/*
173 * scan the scrub rate mapping table for a close or matching bandwidth value to
174 * issue. If requested is too big, then use last maximum value found.
175 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100176static int __set_scrub_rate(struct pci_dev *ctl, u32 new_bw, u32 min_rate)
Doug Thompson2bc65412009-05-04 20:11:14 +0200177{
178 u32 scrubval;
179 int i;
180
181 /*
182 * map the configured rate (new_bw) to a value specific to the AMD64
183 * memory controller and apply to register. Search for the first
184 * bandwidth entry that is greater or equal than the setting requested
185 * and program that. If at last entry, turn off DRAM scrubbing.
Andrew Morton168bfee2012-10-23 14:09:39 -0700186 *
187 * If no suitable bandwidth is found, turn off DRAM scrubbing entirely
188 * by falling back to the last element in scrubrates[].
Doug Thompson2bc65412009-05-04 20:11:14 +0200189 */
Andrew Morton168bfee2012-10-23 14:09:39 -0700190 for (i = 0; i < ARRAY_SIZE(scrubrates) - 1; i++) {
Doug Thompson2bc65412009-05-04 20:11:14 +0200191 /*
192 * skip scrub rates which aren't recommended
193 * (see F10 BKDG, F3x58)
194 */
Borislav Petkov395ae782010-10-01 18:38:19 +0200195 if (scrubrates[i].scrubval < min_rate)
Doug Thompson2bc65412009-05-04 20:11:14 +0200196 continue;
197
198 if (scrubrates[i].bandwidth <= new_bw)
199 break;
Doug Thompson2bc65412009-05-04 20:11:14 +0200200 }
201
202 scrubval = scrubrates[i].scrubval;
Doug Thompson2bc65412009-05-04 20:11:14 +0200203
Borislav Petkov5980bb92011-01-07 16:26:49 +0100204 pci_write_bits32(ctl, SCRCTRL, scrubval, 0x001F);
Doug Thompson2bc65412009-05-04 20:11:14 +0200205
Borislav Petkov39094442010-11-24 19:52:09 +0100206 if (scrubval)
207 return scrubrates[i].bandwidth;
208
Doug Thompson2bc65412009-05-04 20:11:14 +0200209 return 0;
210}
211
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100212static int set_scrub_rate(struct mem_ctl_info *mci, u32 bw)
Doug Thompson2bc65412009-05-04 20:11:14 +0200213{
214 struct amd64_pvt *pvt = mci->pvt_info;
Borislav Petkov87b3e0e2011-01-19 20:02:38 +0100215 u32 min_scrubrate = 0x5;
Doug Thompson2bc65412009-05-04 20:11:14 +0200216
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200217 if (pvt->fam == 0xf)
Borislav Petkov87b3e0e2011-01-19 20:02:38 +0100218 min_scrubrate = 0x0;
219
Borislav Petkov3f0aba42013-08-24 11:25:00 +0200220 /* Erratum #505 */
221 if (pvt->fam == 0x15 && pvt->model < 0x10)
Borislav Petkov73ba8592011-09-19 17:34:45 +0200222 f15h_select_dct(pvt, 0);
223
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100224 return __set_scrub_rate(pvt->F3, bw, min_scrubrate);
Doug Thompson2bc65412009-05-04 20:11:14 +0200225}
226
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100227static int get_scrub_rate(struct mem_ctl_info *mci)
Doug Thompson2bc65412009-05-04 20:11:14 +0200228{
229 struct amd64_pvt *pvt = mci->pvt_info;
230 u32 scrubval = 0;
Borislav Petkov39094442010-11-24 19:52:09 +0100231 int i, retval = -EINVAL;
Doug Thompson2bc65412009-05-04 20:11:14 +0200232
Borislav Petkov3f0aba42013-08-24 11:25:00 +0200233 /* Erratum #505 */
234 if (pvt->fam == 0x15 && pvt->model < 0x10)
Borislav Petkov73ba8592011-09-19 17:34:45 +0200235 f15h_select_dct(pvt, 0);
236
Borislav Petkov5980bb92011-01-07 16:26:49 +0100237 amd64_read_pci_cfg(pvt->F3, SCRCTRL, &scrubval);
Doug Thompson2bc65412009-05-04 20:11:14 +0200238
239 scrubval = scrubval & 0x001F;
240
Roel Kluin926311f2010-01-11 20:58:21 +0100241 for (i = 0; i < ARRAY_SIZE(scrubrates); i++) {
Doug Thompson2bc65412009-05-04 20:11:14 +0200242 if (scrubrates[i].scrubval == scrubval) {
Borislav Petkov39094442010-11-24 19:52:09 +0100243 retval = scrubrates[i].bandwidth;
Doug Thompson2bc65412009-05-04 20:11:14 +0200244 break;
245 }
246 }
Borislav Petkov39094442010-11-24 19:52:09 +0100247 return retval;
Doug Thompson2bc65412009-05-04 20:11:14 +0200248}
249
Doug Thompson67757632009-04-27 15:53:22 +0200250/*
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200251 * returns true if the SysAddr given by sys_addr matches the
252 * DRAM base/limit associated with node_id
Doug Thompson67757632009-04-27 15:53:22 +0200253 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100254static bool base_limit_match(struct amd64_pvt *pvt, u64 sys_addr, u8 nid)
Doug Thompson67757632009-04-27 15:53:22 +0200255{
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200256 u64 addr;
Doug Thompson67757632009-04-27 15:53:22 +0200257
258 /* The K8 treats this as a 40-bit value. However, bits 63-40 will be
259 * all ones if the most significant implemented address bit is 1.
260 * Here we discard bits 63-40. See section 3.4.2 of AMD publication
261 * 24592: AMD x86-64 Architecture Programmer's Manual Volume 1
262 * Application Programming.
263 */
264 addr = sys_addr & 0x000000ffffffffffull;
265
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200266 return ((addr >= get_dram_base(pvt, nid)) &&
267 (addr <= get_dram_limit(pvt, nid)));
Doug Thompson67757632009-04-27 15:53:22 +0200268}
269
270/*
271 * Attempt to map a SysAddr to a node. On success, return a pointer to the
272 * mem_ctl_info structure for the node that the SysAddr maps to.
273 *
274 * On failure, return NULL.
275 */
276static struct mem_ctl_info *find_mc_by_sys_addr(struct mem_ctl_info *mci,
277 u64 sys_addr)
278{
279 struct amd64_pvt *pvt;
Daniel J Bluemanc7e53012012-11-30 16:44:20 +0800280 u8 node_id;
Doug Thompson67757632009-04-27 15:53:22 +0200281 u32 intlv_en, bits;
282
283 /*
284 * Here we use the DRAM Base (section 3.4.4.1) and DRAM Limit (section
285 * 3.4.4.2) registers to map the SysAddr to a node ID.
286 */
287 pvt = mci->pvt_info;
288
289 /*
290 * The value of this field should be the same for all DRAM Base
291 * registers. Therefore we arbitrarily choose to read it from the
292 * register for node 0.
293 */
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200294 intlv_en = dram_intlv_en(pvt, 0);
Doug Thompson67757632009-04-27 15:53:22 +0200295
296 if (intlv_en == 0) {
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200297 for (node_id = 0; node_id < DRAM_RANGES; node_id++) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100298 if (base_limit_match(pvt, sys_addr, node_id))
Borislav Petkov8edc5442009-09-18 12:39:19 +0200299 goto found;
Doug Thompson67757632009-04-27 15:53:22 +0200300 }
Borislav Petkov8edc5442009-09-18 12:39:19 +0200301 goto err_no_match;
Doug Thompson67757632009-04-27 15:53:22 +0200302 }
303
Borislav Petkov72f158f2009-09-18 12:27:27 +0200304 if (unlikely((intlv_en != 0x01) &&
305 (intlv_en != 0x03) &&
306 (intlv_en != 0x07))) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +0200307 amd64_warn("DRAM Base[IntlvEn] junk value: 0x%x, BIOS bug?\n", intlv_en);
Doug Thompson67757632009-04-27 15:53:22 +0200308 return NULL;
309 }
310
311 bits = (((u32) sys_addr) >> 12) & intlv_en;
312
313 for (node_id = 0; ; ) {
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200314 if ((dram_intlv_sel(pvt, node_id) & intlv_en) == bits)
Doug Thompson67757632009-04-27 15:53:22 +0200315 break; /* intlv_sel field matches */
316
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200317 if (++node_id >= DRAM_RANGES)
Doug Thompson67757632009-04-27 15:53:22 +0200318 goto err_no_match;
319 }
320
321 /* sanity test for sys_addr */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100322 if (unlikely(!base_limit_match(pvt, sys_addr, node_id))) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +0200323 amd64_warn("%s: sys_addr 0x%llx falls outside base/limit address"
324 "range for node %d with node interleaving enabled.\n",
325 __func__, sys_addr, node_id);
Doug Thompson67757632009-04-27 15:53:22 +0200326 return NULL;
327 }
328
329found:
Borislav Petkovb487c332011-02-21 18:55:00 +0100330 return edac_mc_find((int)node_id);
Doug Thompson67757632009-04-27 15:53:22 +0200331
332err_no_match:
Joe Perches956b9ba12012-04-29 17:08:39 -0300333 edac_dbg(2, "sys_addr 0x%lx doesn't match any node\n",
334 (unsigned long)sys_addr);
Doug Thompson67757632009-04-27 15:53:22 +0200335
336 return NULL;
337}
Doug Thompsone2ce7252009-04-27 15:57:12 +0200338
339/*
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100340 * compute the CS base address of the @csrow on the DRAM controller @dct.
341 * For details see F2x[5C:40] in the processor's BKDG
Doug Thompsone2ce7252009-04-27 15:57:12 +0200342 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100343static void get_cs_base_and_mask(struct amd64_pvt *pvt, int csrow, u8 dct,
344 u64 *base, u64 *mask)
Doug Thompsone2ce7252009-04-27 15:57:12 +0200345{
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100346 u64 csbase, csmask, base_bits, mask_bits;
347 u8 addr_shift;
348
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500349 if (pvt->fam == 0xf && pvt->ext_model < K8_REV_F) {
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100350 csbase = pvt->csels[dct].csbases[csrow];
351 csmask = pvt->csels[dct].csmasks[csrow];
Chen, Gong10ef6b02013-10-18 14:29:07 -0700352 base_bits = GENMASK_ULL(31, 21) | GENMASK_ULL(15, 9);
353 mask_bits = GENMASK_ULL(29, 21) | GENMASK_ULL(15, 9);
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100354 addr_shift = 4;
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500355
356 /*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500357 * F16h and F15h, models 30h and later need two addr_shift values:
358 * 8 for high and 6 for low (cf. F16h BKDG).
359 */
360 } else if (pvt->fam == 0x16 ||
361 (pvt->fam == 0x15 && pvt->model >= 0x30)) {
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500362 csbase = pvt->csels[dct].csbases[csrow];
363 csmask = pvt->csels[dct].csmasks[csrow >> 1];
364
Chen, Gong10ef6b02013-10-18 14:29:07 -0700365 *base = (csbase & GENMASK_ULL(15, 5)) << 6;
366 *base |= (csbase & GENMASK_ULL(30, 19)) << 8;
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500367
368 *mask = ~0ULL;
369 /* poke holes for the csmask */
Chen, Gong10ef6b02013-10-18 14:29:07 -0700370 *mask &= ~((GENMASK_ULL(15, 5) << 6) |
371 (GENMASK_ULL(30, 19) << 8));
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500372
Chen, Gong10ef6b02013-10-18 14:29:07 -0700373 *mask |= (csmask & GENMASK_ULL(15, 5)) << 6;
374 *mask |= (csmask & GENMASK_ULL(30, 19)) << 8;
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -0500375
376 return;
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100377 } else {
378 csbase = pvt->csels[dct].csbases[csrow];
379 csmask = pvt->csels[dct].csmasks[csrow >> 1];
380 addr_shift = 8;
381
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200382 if (pvt->fam == 0x15)
Chen, Gong10ef6b02013-10-18 14:29:07 -0700383 base_bits = mask_bits =
384 GENMASK_ULL(30,19) | GENMASK_ULL(13,5);
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100385 else
Chen, Gong10ef6b02013-10-18 14:29:07 -0700386 base_bits = mask_bits =
387 GENMASK_ULL(28,19) | GENMASK_ULL(13,5);
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100388 }
389
390 *base = (csbase & base_bits) << addr_shift;
391
392 *mask = ~0ULL;
393 /* poke holes for the csmask */
394 *mask &= ~(mask_bits << addr_shift);
395 /* OR them in */
396 *mask |= (csmask & mask_bits) << addr_shift;
Doug Thompsone2ce7252009-04-27 15:57:12 +0200397}
398
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100399#define for_each_chip_select(i, dct, pvt) \
400 for (i = 0; i < pvt->csels[dct].b_cnt; i++)
Doug Thompsone2ce7252009-04-27 15:57:12 +0200401
Borislav Petkov614ec9d2011-01-13 18:02:22 +0100402#define chip_select_base(i, dct, pvt) \
403 pvt->csels[dct].csbases[i]
404
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100405#define for_each_chip_select_mask(i, dct, pvt) \
406 for (i = 0; i < pvt->csels[dct].m_cnt; i++)
Doug Thompsone2ce7252009-04-27 15:57:12 +0200407
408/*
409 * @input_addr is an InputAddr associated with the node given by mci. Return the
410 * csrow that input_addr maps to, or -1 on failure (no csrow claims input_addr).
411 */
412static int input_addr_to_csrow(struct mem_ctl_info *mci, u64 input_addr)
413{
414 struct amd64_pvt *pvt;
415 int csrow;
416 u64 base, mask;
417
418 pvt = mci->pvt_info;
419
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100420 for_each_chip_select(csrow, 0, pvt) {
421 if (!csrow_enabled(csrow, 0, pvt))
Doug Thompsone2ce7252009-04-27 15:57:12 +0200422 continue;
423
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100424 get_cs_base_and_mask(pvt, csrow, 0, &base, &mask);
425
426 mask = ~mask;
Doug Thompsone2ce7252009-04-27 15:57:12 +0200427
428 if ((input_addr & mask) == (base & mask)) {
Joe Perches956b9ba12012-04-29 17:08:39 -0300429 edac_dbg(2, "InputAddr 0x%lx matches csrow %d (node %d)\n",
430 (unsigned long)input_addr, csrow,
431 pvt->mc_node_id);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200432
433 return csrow;
434 }
435 }
Joe Perches956b9ba12012-04-29 17:08:39 -0300436 edac_dbg(2, "no matching csrow for InputAddr 0x%lx (MC node %d)\n",
437 (unsigned long)input_addr, pvt->mc_node_id);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200438
439 return -1;
440}
441
442/*
Doug Thompsone2ce7252009-04-27 15:57:12 +0200443 * Obtain info from the DRAM Hole Address Register (section 3.4.8, pub #26094)
444 * for the node represented by mci. Info is passed back in *hole_base,
445 * *hole_offset, and *hole_size. Function returns 0 if info is valid or 1 if
446 * info is invalid. Info may be invalid for either of the following reasons:
447 *
448 * - The revision of the node is not E or greater. In this case, the DRAM Hole
449 * Address Register does not exist.
450 *
451 * - The DramHoleValid bit is cleared in the DRAM Hole Address Register,
452 * indicating that its contents are not valid.
453 *
454 * The values passed back in *hole_base, *hole_offset, and *hole_size are
455 * complete 32-bit values despite the fact that the bitfields in the DHAR
456 * only represent bits 31-24 of the base and offset values.
457 */
458int amd64_get_dram_hole_info(struct mem_ctl_info *mci, u64 *hole_base,
459 u64 *hole_offset, u64 *hole_size)
460{
461 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompsone2ce7252009-04-27 15:57:12 +0200462
463 /* only revE and later have the DRAM Hole Address Register */
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200464 if (pvt->fam == 0xf && pvt->ext_model < K8_REV_E) {
Joe Perches956b9ba12012-04-29 17:08:39 -0300465 edac_dbg(1, " revision %d for node %d does not support DHAR\n",
466 pvt->ext_model, pvt->mc_node_id);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200467 return 1;
468 }
469
Borislav Petkovbc21fa52010-11-11 17:29:13 +0100470 /* valid for Fam10h and above */
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200471 if (pvt->fam >= 0x10 && !dhar_mem_hoist_valid(pvt)) {
Joe Perches956b9ba12012-04-29 17:08:39 -0300472 edac_dbg(1, " Dram Memory Hoisting is DISABLED on this system\n");
Doug Thompsone2ce7252009-04-27 15:57:12 +0200473 return 1;
474 }
475
Borislav Petkovc8e518d2010-12-10 19:49:19 +0100476 if (!dhar_valid(pvt)) {
Joe Perches956b9ba12012-04-29 17:08:39 -0300477 edac_dbg(1, " Dram Memory Hoisting is DISABLED on this node %d\n",
478 pvt->mc_node_id);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200479 return 1;
480 }
481
482 /* This node has Memory Hoisting */
483
484 /* +------------------+--------------------+--------------------+-----
485 * | memory | DRAM hole | relocated |
486 * | [0, (x - 1)] | [x, 0xffffffff] | addresses from |
487 * | | | DRAM hole |
488 * | | | [0x100000000, |
489 * | | | (0x100000000+ |
490 * | | | (0xffffffff-x))] |
491 * +------------------+--------------------+--------------------+-----
492 *
493 * Above is a diagram of physical memory showing the DRAM hole and the
494 * relocated addresses from the DRAM hole. As shown, the DRAM hole
495 * starts at address x (the base address) and extends through address
496 * 0xffffffff. The DRAM Hole Address Register (DHAR) relocates the
497 * addresses in the hole so that they start at 0x100000000.
498 */
499
Borislav Petkov1f316772012-08-10 12:50:50 +0200500 *hole_base = dhar_base(pvt);
501 *hole_size = (1ULL << 32) - *hole_base;
Doug Thompsone2ce7252009-04-27 15:57:12 +0200502
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200503 *hole_offset = (pvt->fam > 0xf) ? f10_dhar_offset(pvt)
504 : k8_dhar_offset(pvt);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200505
Joe Perches956b9ba12012-04-29 17:08:39 -0300506 edac_dbg(1, " DHAR info for node %d base 0x%lx offset 0x%lx size 0x%lx\n",
507 pvt->mc_node_id, (unsigned long)*hole_base,
508 (unsigned long)*hole_offset, (unsigned long)*hole_size);
Doug Thompsone2ce7252009-04-27 15:57:12 +0200509
510 return 0;
511}
512EXPORT_SYMBOL_GPL(amd64_get_dram_hole_info);
513
Doug Thompson93c2df52009-05-04 20:46:50 +0200514/*
515 * Return the DramAddr that the SysAddr given by @sys_addr maps to. It is
516 * assumed that sys_addr maps to the node given by mci.
517 *
518 * The first part of section 3.4.4 (p. 70) shows how the DRAM Base (section
519 * 3.4.4.1) and DRAM Limit (section 3.4.4.2) registers are used to translate a
520 * SysAddr to a DramAddr. If the DRAM Hole Address Register (DHAR) is enabled,
521 * then it is also involved in translating a SysAddr to a DramAddr. Sections
522 * 3.4.8 and 3.5.8.2 describe the DHAR and how it is used for memory hoisting.
523 * These parts of the documentation are unclear. I interpret them as follows:
524 *
525 * When node n receives a SysAddr, it processes the SysAddr as follows:
526 *
527 * 1. It extracts the DRAMBase and DRAMLimit values from the DRAM Base and DRAM
528 * Limit registers for node n. If the SysAddr is not within the range
529 * specified by the base and limit values, then node n ignores the Sysaddr
530 * (since it does not map to node n). Otherwise continue to step 2 below.
531 *
532 * 2. If the DramHoleValid bit of the DHAR for node n is clear, the DHAR is
533 * disabled so skip to step 3 below. Otherwise see if the SysAddr is within
534 * the range of relocated addresses (starting at 0x100000000) from the DRAM
535 * hole. If not, skip to step 3 below. Else get the value of the
536 * DramHoleOffset field from the DHAR. To obtain the DramAddr, subtract the
537 * offset defined by this value from the SysAddr.
538 *
539 * 3. Obtain the base address for node n from the DRAMBase field of the DRAM
540 * Base register for node n. To obtain the DramAddr, subtract the base
541 * address from the SysAddr, as shown near the start of section 3.4.4 (p.70).
542 */
543static u64 sys_addr_to_dram_addr(struct mem_ctl_info *mci, u64 sys_addr)
544{
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200545 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompson93c2df52009-05-04 20:46:50 +0200546 u64 dram_base, hole_base, hole_offset, hole_size, dram_addr;
Borislav Petkov1f316772012-08-10 12:50:50 +0200547 int ret;
Doug Thompson93c2df52009-05-04 20:46:50 +0200548
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200549 dram_base = get_dram_base(pvt, pvt->mc_node_id);
Doug Thompson93c2df52009-05-04 20:46:50 +0200550
551 ret = amd64_get_dram_hole_info(mci, &hole_base, &hole_offset,
552 &hole_size);
553 if (!ret) {
Borislav Petkov1f316772012-08-10 12:50:50 +0200554 if ((sys_addr >= (1ULL << 32)) &&
555 (sys_addr < ((1ULL << 32) + hole_size))) {
Doug Thompson93c2df52009-05-04 20:46:50 +0200556 /* use DHAR to translate SysAddr to DramAddr */
557 dram_addr = sys_addr - hole_offset;
558
Joe Perches956b9ba12012-04-29 17:08:39 -0300559 edac_dbg(2, "using DHAR to translate SysAddr 0x%lx to DramAddr 0x%lx\n",
560 (unsigned long)sys_addr,
561 (unsigned long)dram_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200562
563 return dram_addr;
564 }
565 }
566
567 /*
568 * Translate the SysAddr to a DramAddr as shown near the start of
569 * section 3.4.4 (p. 70). Although sys_addr is a 64-bit value, the k8
570 * only deals with 40-bit values. Therefore we discard bits 63-40 of
571 * sys_addr below. If bit 39 of sys_addr is 1 then the bits we
572 * discard are all 1s. Otherwise the bits we discard are all 0s. See
573 * section 3.4.2 of AMD publication 24592: AMD x86-64 Architecture
574 * Programmer's Manual Volume 1 Application Programming.
575 */
Chen, Gong10ef6b02013-10-18 14:29:07 -0700576 dram_addr = (sys_addr & GENMASK_ULL(39, 0)) - dram_base;
Doug Thompson93c2df52009-05-04 20:46:50 +0200577
Joe Perches956b9ba12012-04-29 17:08:39 -0300578 edac_dbg(2, "using DRAM Base register to translate SysAddr 0x%lx to DramAddr 0x%lx\n",
579 (unsigned long)sys_addr, (unsigned long)dram_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200580 return dram_addr;
581}
582
583/*
584 * @intlv_en is the value of the IntlvEn field from a DRAM Base register
585 * (section 3.4.4.1). Return the number of bits from a SysAddr that are used
586 * for node interleaving.
587 */
588static int num_node_interleave_bits(unsigned intlv_en)
589{
590 static const int intlv_shift_table[] = { 0, 1, 0, 2, 0, 0, 0, 3 };
591 int n;
592
593 BUG_ON(intlv_en > 7);
594 n = intlv_shift_table[intlv_en];
595 return n;
596}
597
598/* Translate the DramAddr given by @dram_addr to an InputAddr. */
599static u64 dram_addr_to_input_addr(struct mem_ctl_info *mci, u64 dram_addr)
600{
601 struct amd64_pvt *pvt;
602 int intlv_shift;
603 u64 input_addr;
604
605 pvt = mci->pvt_info;
606
607 /*
608 * See the start of section 3.4.4 (p. 70, BKDG #26094, K8, revA-E)
609 * concerning translating a DramAddr to an InputAddr.
610 */
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200611 intlv_shift = num_node_interleave_bits(dram_intlv_en(pvt, 0));
Chen, Gong10ef6b02013-10-18 14:29:07 -0700612 input_addr = ((dram_addr >> intlv_shift) & GENMASK_ULL(35, 12)) +
Borislav Petkovf678b8c2010-12-13 19:21:07 +0100613 (dram_addr & 0xfff);
Doug Thompson93c2df52009-05-04 20:46:50 +0200614
Joe Perches956b9ba12012-04-29 17:08:39 -0300615 edac_dbg(2, " Intlv Shift=%d DramAddr=0x%lx maps to InputAddr=0x%lx\n",
616 intlv_shift, (unsigned long)dram_addr,
617 (unsigned long)input_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200618
619 return input_addr;
620}
621
622/*
623 * Translate the SysAddr represented by @sys_addr to an InputAddr. It is
624 * assumed that @sys_addr maps to the node given by mci.
625 */
626static u64 sys_addr_to_input_addr(struct mem_ctl_info *mci, u64 sys_addr)
627{
628 u64 input_addr;
629
630 input_addr =
631 dram_addr_to_input_addr(mci, sys_addr_to_dram_addr(mci, sys_addr));
632
Joe Perches956b9ba12012-04-29 17:08:39 -0300633 edac_dbg(2, "SysAdddr 0x%lx translates to InputAddr 0x%lx\n",
634 (unsigned long)sys_addr, (unsigned long)input_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200635
636 return input_addr;
637}
638
Doug Thompson93c2df52009-05-04 20:46:50 +0200639/* Map the Error address to a PAGE and PAGE OFFSET. */
640static inline void error_address_to_page_and_offset(u64 error_address,
Borislav Petkov33ca0642012-08-30 18:01:36 +0200641 struct err_info *err)
Doug Thompson93c2df52009-05-04 20:46:50 +0200642{
Borislav Petkov33ca0642012-08-30 18:01:36 +0200643 err->page = (u32) (error_address >> PAGE_SHIFT);
644 err->offset = ((u32) error_address) & ~PAGE_MASK;
Doug Thompson93c2df52009-05-04 20:46:50 +0200645}
646
647/*
648 * @sys_addr is an error address (a SysAddr) extracted from the MCA NB Address
649 * Low (section 3.6.4.5) and MCA NB Address High (section 3.6.4.6) registers
650 * of a node that detected an ECC memory error. mci represents the node that
651 * the error address maps to (possibly different from the node that detected
652 * the error). Return the number of the csrow that sys_addr maps to, or -1 on
653 * error.
654 */
655static int sys_addr_to_csrow(struct mem_ctl_info *mci, u64 sys_addr)
656{
657 int csrow;
658
659 csrow = input_addr_to_csrow(mci, sys_addr_to_input_addr(mci, sys_addr));
660
661 if (csrow == -1)
Borislav Petkov24f9a7f2010-10-07 18:29:15 +0200662 amd64_mc_err(mci, "Failed to translate InputAddr to csrow for "
663 "address 0x%lx\n", (unsigned long)sys_addr);
Doug Thompson93c2df52009-05-04 20:46:50 +0200664 return csrow;
665}
Doug Thompsone2ce7252009-04-27 15:57:12 +0200666
Borislav Petkovbfc04ae2009-11-12 19:05:07 +0100667static int get_channel_from_ecc_syndrome(struct mem_ctl_info *, u16);
Doug Thompson2da11652009-04-27 16:09:09 +0200668
Doug Thompson2da11652009-04-27 16:09:09 +0200669/*
670 * Determine if the DIMMs have ECC enabled. ECC is enabled ONLY if all the DIMMs
671 * are ECC capable.
672 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100673static unsigned long determine_edac_cap(struct amd64_pvt *pvt)
Doug Thompson2da11652009-04-27 16:09:09 +0200674{
Borislav Petkovcb328502010-12-22 14:28:24 +0100675 u8 bit;
Dan Carpenter1f6189e2011-10-06 02:30:25 -0400676 unsigned long edac_cap = EDAC_FLAG_NONE;
Doug Thompson2da11652009-04-27 16:09:09 +0200677
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200678 bit = (pvt->fam > 0xf || pvt->ext_model >= K8_REV_F)
Doug Thompson2da11652009-04-27 16:09:09 +0200679 ? 19
680 : 17;
681
Borislav Petkov584fcff2009-06-10 18:29:54 +0200682 if (pvt->dclr0 & BIT(bit))
Doug Thompson2da11652009-04-27 16:09:09 +0200683 edac_cap = EDAC_FLAG_SECDED;
684
685 return edac_cap;
686}
687
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100688static void debug_display_dimm_sizes(struct amd64_pvt *, u8);
Doug Thompson2da11652009-04-27 16:09:09 +0200689
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100690static void debug_dump_dramcfg_low(struct amd64_pvt *pvt, u32 dclr, int chan)
Borislav Petkov68798e12009-11-03 16:18:33 +0100691{
Joe Perches956b9ba12012-04-29 17:08:39 -0300692 edac_dbg(1, "F2x%d90 (DRAM Cfg Low): 0x%08x\n", chan, dclr);
Borislav Petkov68798e12009-11-03 16:18:33 +0100693
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100694 if (pvt->dram_type == MEM_LRDDR3) {
695 u32 dcsm = pvt->csels[chan].csmasks[0];
696 /*
697 * It's assumed all LRDIMMs in a DCT are going to be of
698 * same 'type' until proven otherwise. So, use a cs
699 * value of '0' here to get dcsm value.
700 */
701 edac_dbg(1, " LRDIMM %dx rank multiply\n", (dcsm & 0x3));
702 }
703
704 edac_dbg(1, "All DIMMs support ECC:%s\n",
705 (dclr & BIT(19)) ? "yes" : "no");
706
Borislav Petkov68798e12009-11-03 16:18:33 +0100707
Joe Perches956b9ba12012-04-29 17:08:39 -0300708 edac_dbg(1, " PAR/ERR parity: %s\n",
709 (dclr & BIT(8)) ? "enabled" : "disabled");
Borislav Petkov68798e12009-11-03 16:18:33 +0100710
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200711 if (pvt->fam == 0x10)
Joe Perches956b9ba12012-04-29 17:08:39 -0300712 edac_dbg(1, " DCT 128bit mode width: %s\n",
713 (dclr & BIT(11)) ? "128b" : "64b");
Borislav Petkov68798e12009-11-03 16:18:33 +0100714
Joe Perches956b9ba12012-04-29 17:08:39 -0300715 edac_dbg(1, " x4 logical DIMMs present: L0: %s L1: %s L2: %s L3: %s\n",
716 (dclr & BIT(12)) ? "yes" : "no",
717 (dclr & BIT(13)) ? "yes" : "no",
718 (dclr & BIT(14)) ? "yes" : "no",
719 (dclr & BIT(15)) ? "yes" : "no");
Borislav Petkov68798e12009-11-03 16:18:33 +0100720}
721
Doug Thompson2da11652009-04-27 16:09:09 +0200722/* Display and decode various NB registers for debug purposes. */
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200723static void dump_misc_regs(struct amd64_pvt *pvt)
Doug Thompson2da11652009-04-27 16:09:09 +0200724{
Joe Perches956b9ba12012-04-29 17:08:39 -0300725 edac_dbg(1, "F3xE8 (NB Cap): 0x%08x\n", pvt->nbcap);
Doug Thompson2da11652009-04-27 16:09:09 +0200726
Joe Perches956b9ba12012-04-29 17:08:39 -0300727 edac_dbg(1, " NB two channel DRAM capable: %s\n",
728 (pvt->nbcap & NBCAP_DCT_DUAL) ? "yes" : "no");
Borislav Petkov68798e12009-11-03 16:18:33 +0100729
Joe Perches956b9ba12012-04-29 17:08:39 -0300730 edac_dbg(1, " ECC capable: %s, ChipKill ECC capable: %s\n",
731 (pvt->nbcap & NBCAP_SECDED) ? "yes" : "no",
732 (pvt->nbcap & NBCAP_CHIPKILL) ? "yes" : "no");
Borislav Petkov68798e12009-11-03 16:18:33 +0100733
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100734 debug_dump_dramcfg_low(pvt, pvt->dclr0, 0);
Doug Thompson2da11652009-04-27 16:09:09 +0200735
Joe Perches956b9ba12012-04-29 17:08:39 -0300736 edac_dbg(1, "F3xB0 (Online Spare): 0x%08x\n", pvt->online_spare);
Doug Thompson2da11652009-04-27 16:09:09 +0200737
Joe Perches956b9ba12012-04-29 17:08:39 -0300738 edac_dbg(1, "F1xF0 (DRAM Hole Address): 0x%08x, base: 0x%08x, offset: 0x%08x\n",
739 pvt->dhar, dhar_base(pvt),
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200740 (pvt->fam == 0xf) ? k8_dhar_offset(pvt)
741 : f10_dhar_offset(pvt));
Doug Thompson2da11652009-04-27 16:09:09 +0200742
Joe Perches956b9ba12012-04-29 17:08:39 -0300743 edac_dbg(1, " DramHoleValid: %s\n", dhar_valid(pvt) ? "yes" : "no");
Doug Thompson2da11652009-04-27 16:09:09 +0200744
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100745 debug_display_dimm_sizes(pvt, 0);
Borislav Petkov4d796362011-02-03 15:59:57 +0100746
Borislav Petkov8de1d912009-10-16 13:39:30 +0200747 /* everything below this point is Fam10h and above */
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200748 if (pvt->fam == 0xf)
Doug Thompson2da11652009-04-27 16:09:09 +0200749 return;
Borislav Petkov4d796362011-02-03 15:59:57 +0100750
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100751 debug_display_dimm_sizes(pvt, 1);
Doug Thompson2da11652009-04-27 16:09:09 +0200752
Borislav Petkova3b7db02011-01-19 20:35:12 +0100753 amd64_info("using %s syndromes.\n", ((pvt->ecc_sym_sz == 8) ? "x8" : "x4"));
Borislav Petkovad6a32e2010-03-09 12:46:00 +0100754
Borislav Petkov8de1d912009-10-16 13:39:30 +0200755 /* Only if NOT ganged does dclr1 have valid info */
Borislav Petkov68798e12009-11-03 16:18:33 +0100756 if (!dct_ganging_enabled(pvt))
Borislav Petkovd1ea71c2013-12-15 17:54:27 +0100757 debug_dump_dramcfg_low(pvt, pvt->dclr1, 1);
Doug Thompson2da11652009-04-27 16:09:09 +0200758}
759
Doug Thompson94be4bf2009-04-27 16:12:00 +0200760/*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500761 * See BKDG, F2x[1,0][5C:40], F2[1,0][6C:60]
Doug Thompson94be4bf2009-04-27 16:12:00 +0200762 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100763static void prep_chip_selects(struct amd64_pvt *pvt)
Doug Thompson94be4bf2009-04-27 16:12:00 +0200764{
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500765 if (pvt->fam == 0xf && pvt->ext_model < K8_REV_F) {
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100766 pvt->csels[0].b_cnt = pvt->csels[1].b_cnt = 8;
767 pvt->csels[0].m_cnt = pvt->csels[1].m_cnt = 8;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100768 } else if (pvt->fam == 0x15 && pvt->model == 0x30) {
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500769 pvt->csels[0].b_cnt = pvt->csels[1].b_cnt = 4;
770 pvt->csels[0].m_cnt = pvt->csels[1].m_cnt = 2;
Borislav Petkov9d858bb2009-09-21 14:35:51 +0200771 } else {
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100772 pvt->csels[0].b_cnt = pvt->csels[1].b_cnt = 8;
773 pvt->csels[0].m_cnt = pvt->csels[1].m_cnt = 4;
Doug Thompson94be4bf2009-04-27 16:12:00 +0200774 }
775}
776
777/*
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100778 * Function 2 Offset F10_DCSB0; read in the DCS Base and DCS Mask registers
Doug Thompson94be4bf2009-04-27 16:12:00 +0200779 */
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200780static void read_dct_base_mask(struct amd64_pvt *pvt)
Doug Thompson94be4bf2009-04-27 16:12:00 +0200781{
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100782 int cs;
Doug Thompson94be4bf2009-04-27 16:12:00 +0200783
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100784 prep_chip_selects(pvt);
Doug Thompson94be4bf2009-04-27 16:12:00 +0200785
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100786 for_each_chip_select(cs, 0, pvt) {
Borislav Petkov71d2a322011-02-21 19:37:24 +0100787 int reg0 = DCSB0 + (cs * 4);
788 int reg1 = DCSB1 + (cs * 4);
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100789 u32 *base0 = &pvt->csels[0].csbases[cs];
790 u32 *base1 = &pvt->csels[1].csbases[cs];
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200791
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500792 if (!amd64_read_dct_pci_cfg(pvt, 0, reg0, base0))
Joe Perches956b9ba12012-04-29 17:08:39 -0300793 edac_dbg(0, " DCSB0[%d]=0x%08x reg: F2x%x\n",
794 cs, *base0, reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +0200795
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500796 if (pvt->fam == 0xf)
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100797 continue;
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200798
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500799 if (!amd64_read_dct_pci_cfg(pvt, 1, reg0, base1))
Joe Perches956b9ba12012-04-29 17:08:39 -0300800 edac_dbg(0, " DCSB1[%d]=0x%08x reg: F2x%x\n",
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500801 cs, *base1, (pvt->fam == 0x10) ? reg1
802 : reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +0200803 }
804
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100805 for_each_chip_select_mask(cs, 0, pvt) {
Borislav Petkov71d2a322011-02-21 19:37:24 +0100806 int reg0 = DCSM0 + (cs * 4);
807 int reg1 = DCSM1 + (cs * 4);
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100808 u32 *mask0 = &pvt->csels[0].csmasks[cs];
809 u32 *mask1 = &pvt->csels[1].csmasks[cs];
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200810
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500811 if (!amd64_read_dct_pci_cfg(pvt, 0, reg0, mask0))
Joe Perches956b9ba12012-04-29 17:08:39 -0300812 edac_dbg(0, " DCSM0[%d]=0x%08x reg: F2x%x\n",
813 cs, *mask0, reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +0200814
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500815 if (pvt->fam == 0xf)
Borislav Petkov11c75ea2010-11-29 19:49:02 +0100816 continue;
Borislav Petkovb2b0c602010-10-08 18:32:29 +0200817
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500818 if (!amd64_read_dct_pci_cfg(pvt, 1, reg0, mask1))
Joe Perches956b9ba12012-04-29 17:08:39 -0300819 edac_dbg(0, " DCSM1[%d]=0x%08x reg: F2x%x\n",
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -0500820 cs, *mask1, (pvt->fam == 0x10) ? reg1
821 : reg0);
Doug Thompson94be4bf2009-04-27 16:12:00 +0200822 }
823}
824
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100825static void determine_memory_type(struct amd64_pvt *pvt)
Doug Thompson94be4bf2009-04-27 16:12:00 +0200826{
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100827 u32 dram_ctrl, dcsm;
Doug Thompson94be4bf2009-04-27 16:12:00 +0200828
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100829 switch (pvt->fam) {
830 case 0xf:
831 if (pvt->ext_model >= K8_REV_F)
832 goto ddr3;
833
834 pvt->dram_type = (pvt->dclr0 & BIT(18)) ? MEM_DDR : MEM_RDDR;
835 return;
836
837 case 0x10:
Borislav Petkov6b4c0bd2009-11-12 15:37:57 +0100838 if (pvt->dchr0 & DDR3_MODE)
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100839 goto ddr3;
840
841 pvt->dram_type = (pvt->dclr0 & BIT(16)) ? MEM_DDR2 : MEM_RDDR2;
842 return;
843
844 case 0x15:
845 if (pvt->model < 0x60)
846 goto ddr3;
847
848 /*
849 * Model 0x60h needs special handling:
850 *
851 * We use a Chip Select value of '0' to obtain dcsm.
852 * Theoretically, it is possible to populate LRDIMMs of different
853 * 'Rank' value on a DCT. But this is not the common case. So,
854 * it's reasonable to assume all DIMMs are going to be of same
855 * 'type' until proven otherwise.
856 */
857 amd64_read_dct_pci_cfg(pvt, 0, DRAM_CONTROL, &dram_ctrl);
858 dcsm = pvt->csels[0].csmasks[0];
859
860 if (((dram_ctrl >> 8) & 0x7) == 0x2)
861 pvt->dram_type = MEM_DDR4;
862 else if (pvt->dclr0 & BIT(16))
863 pvt->dram_type = MEM_DDR3;
864 else if (dcsm & 0x3)
865 pvt->dram_type = MEM_LRDDR3;
Borislav Petkov6b4c0bd2009-11-12 15:37:57 +0100866 else
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100867 pvt->dram_type = MEM_RDDR3;
868
869 return;
870
871 case 0x16:
872 goto ddr3;
873
874 default:
875 WARN(1, KERN_ERR "%s: Family??? 0x%x\n", __func__, pvt->fam);
876 pvt->dram_type = MEM_EMPTY;
Doug Thompson94be4bf2009-04-27 16:12:00 +0200877 }
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100878 return;
Doug Thompson94be4bf2009-04-27 16:12:00 +0200879
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +0100880ddr3:
881 pvt->dram_type = (pvt->dclr0 & BIT(16)) ? MEM_DDR3 : MEM_RDDR3;
Doug Thompson94be4bf2009-04-27 16:12:00 +0200882}
883
Borislav Petkovcb328502010-12-22 14:28:24 +0100884/* Get the number of DCT channels the memory controller is using. */
Doug Thompsonddff8762009-04-27 16:14:52 +0200885static int k8_early_channel_count(struct amd64_pvt *pvt)
886{
Borislav Petkovcb328502010-12-22 14:28:24 +0100887 int flag;
Doug Thompsonddff8762009-04-27 16:14:52 +0200888
Borislav Petkov9f56da02010-10-01 19:44:53 +0200889 if (pvt->ext_model >= K8_REV_F)
Doug Thompsonddff8762009-04-27 16:14:52 +0200890 /* RevF (NPT) and later */
Borislav Petkov41d8bfa2011-01-18 19:16:08 +0100891 flag = pvt->dclr0 & WIDTH_128;
Borislav Petkov9f56da02010-10-01 19:44:53 +0200892 else
Doug Thompsonddff8762009-04-27 16:14:52 +0200893 /* RevE and earlier */
894 flag = pvt->dclr0 & REVE_WIDTH_128;
Doug Thompsonddff8762009-04-27 16:14:52 +0200895
896 /* not used */
897 pvt->dclr1 = 0;
898
899 return (flag) ? 2 : 1;
900}
901
Borislav Petkov70046622011-01-10 14:37:27 +0100902/* On F10h and later ErrAddr is MC4_ADDR[47:1] */
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200903static u64 get_error_address(struct amd64_pvt *pvt, struct mce *m)
Doug Thompsonddff8762009-04-27 16:14:52 +0200904{
Borislav Petkov2ec591a2015-02-17 10:58:34 +0100905 u16 mce_nid = amd_get_nb_id(m->extcpu);
906 struct mem_ctl_info *mci;
Borislav Petkov70046622011-01-10 14:37:27 +0100907 u8 start_bit = 1;
908 u8 end_bit = 47;
Borislav Petkov2ec591a2015-02-17 10:58:34 +0100909 u64 addr;
910
911 mci = edac_mc_find(mce_nid);
912 if (!mci)
913 return 0;
914
915 pvt = mci->pvt_info;
Borislav Petkov70046622011-01-10 14:37:27 +0100916
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200917 if (pvt->fam == 0xf) {
Borislav Petkov70046622011-01-10 14:37:27 +0100918 start_bit = 3;
919 end_bit = 39;
920 }
921
Chen, Gong10ef6b02013-10-18 14:29:07 -0700922 addr = m->addr & GENMASK_ULL(end_bit, start_bit);
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200923
924 /*
925 * Erratum 637 workaround
926 */
Borislav Petkova4b4bed2013-08-10 13:54:48 +0200927 if (pvt->fam == 0x15) {
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200928 u64 cc6_base, tmp_addr;
929 u32 tmp;
Daniel J Blueman8b84c8d2012-11-27 14:32:10 +0800930 u8 intlv_en;
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200931
Chen, Gong10ef6b02013-10-18 14:29:07 -0700932 if ((addr & GENMASK_ULL(47, 24)) >> 24 != 0x00fdf7)
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200933 return addr;
934
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200935
936 amd64_read_pci_cfg(pvt->F1, DRAM_LOCAL_NODE_LIM, &tmp);
937 intlv_en = tmp >> 21 & 0x7;
938
939 /* add [47:27] + 3 trailing bits */
Chen, Gong10ef6b02013-10-18 14:29:07 -0700940 cc6_base = (tmp & GENMASK_ULL(20, 0)) << 3;
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200941
942 /* reverse and add DramIntlvEn */
943 cc6_base |= intlv_en ^ 0x7;
944
945 /* pin at [47:24] */
946 cc6_base <<= 24;
947
948 if (!intlv_en)
Chen, Gong10ef6b02013-10-18 14:29:07 -0700949 return cc6_base | (addr & GENMASK_ULL(23, 0));
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200950
951 amd64_read_pci_cfg(pvt->F1, DRAM_LOCAL_NODE_BASE, &tmp);
952
953 /* faster log2 */
Chen, Gong10ef6b02013-10-18 14:29:07 -0700954 tmp_addr = (addr & GENMASK_ULL(23, 12)) << __fls(intlv_en + 1);
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200955
956 /* OR DramIntlvSel into bits [14:12] */
Chen, Gong10ef6b02013-10-18 14:29:07 -0700957 tmp_addr |= (tmp & GENMASK_ULL(23, 21)) >> 9;
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200958
959 /* add remaining [11:0] bits from original MC4_ADDR */
Chen, Gong10ef6b02013-10-18 14:29:07 -0700960 tmp_addr |= addr & GENMASK_ULL(11, 0);
Borislav Petkovc1ae6832011-03-30 15:42:10 +0200961
962 return cc6_base | tmp_addr;
963 }
964
965 return addr;
Doug Thompsonddff8762009-04-27 16:14:52 +0200966}
967
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +0800968static struct pci_dev *pci_get_related_function(unsigned int vendor,
969 unsigned int device,
970 struct pci_dev *related)
971{
972 struct pci_dev *dev = NULL;
973
974 while ((dev = pci_get_device(vendor, device, dev))) {
975 if (pci_domain_nr(dev->bus) == pci_domain_nr(related->bus) &&
976 (dev->bus->number == related->bus->number) &&
977 (PCI_SLOT(dev->devfn) == PCI_SLOT(related->devfn)))
978 break;
979 }
980
981 return dev;
982}
983
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200984static void read_dram_base_limit_regs(struct amd64_pvt *pvt, unsigned range)
Doug Thompsonddff8762009-04-27 16:14:52 +0200985{
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +0800986 struct amd_northbridge *nb;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500987 struct pci_dev *f1 = NULL;
988 unsigned int pci_func;
Borislav Petkov71d2a322011-02-21 19:37:24 +0100989 int off = range << 3;
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +0800990 u32 llim;
Doug Thompsonddff8762009-04-27 16:14:52 +0200991
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200992 amd64_read_pci_cfg(pvt->F1, DRAM_BASE_LO + off, &pvt->ranges[range].base.lo);
993 amd64_read_pci_cfg(pvt->F1, DRAM_LIMIT_LO + off, &pvt->ranges[range].lim.lo);
Doug Thompsonddff8762009-04-27 16:14:52 +0200994
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -0500995 if (pvt->fam == 0xf)
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200996 return;
Doug Thompsonddff8762009-04-27 16:14:52 +0200997
Borislav Petkov7f19bf72010-10-21 18:52:53 +0200998 if (!dram_rw(pvt, range))
999 return;
Doug Thompsonddff8762009-04-27 16:14:52 +02001000
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001001 amd64_read_pci_cfg(pvt->F1, DRAM_BASE_HI + off, &pvt->ranges[range].base.hi);
1002 amd64_read_pci_cfg(pvt->F1, DRAM_LIMIT_HI + off, &pvt->ranges[range].lim.hi);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001003
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001004 /* F15h: factor in CC6 save area by reading dst node's limit reg */
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001005 if (pvt->fam != 0x15)
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001006 return;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001007
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001008 nb = node_to_amd_nb(dram_dst_node(pvt, range));
1009 if (WARN_ON(!nb))
1010 return;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001011
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001012 if (pvt->model == 0x60)
1013 pci_func = PCI_DEVICE_ID_AMD_15H_M60H_NB_F1;
1014 else if (pvt->model == 0x30)
1015 pci_func = PCI_DEVICE_ID_AMD_15H_M30H_NB_F1;
1016 else
1017 pci_func = PCI_DEVICE_ID_AMD_15H_NB_F1;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001018
1019 f1 = pci_get_related_function(nb->misc->vendor, pci_func, nb->misc);
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001020 if (WARN_ON(!f1))
1021 return;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001022
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001023 amd64_read_pci_cfg(f1, DRAM_LOCAL_NODE_LIM, &llim);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001024
Chen, Gong10ef6b02013-10-18 14:29:07 -07001025 pvt->ranges[range].lim.lo &= GENMASK_ULL(15, 0);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001026
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001027 /* {[39:27],111b} */
1028 pvt->ranges[range].lim.lo |= ((llim & 0x1fff) << 3 | 0x7) << 16;
Borislav Petkovf08e4572011-03-21 20:45:06 +01001029
Chen, Gong10ef6b02013-10-18 14:29:07 -07001030 pvt->ranges[range].lim.hi &= GENMASK_ULL(7, 0);
Borislav Petkovf08e4572011-03-21 20:45:06 +01001031
Daniel J Bluemane2c0bff2012-11-30 16:44:19 +08001032 /* [47:40] */
1033 pvt->ranges[range].lim.hi |= llim >> 13;
1034
1035 pci_dev_put(f1);
Doug Thompsonddff8762009-04-27 16:14:52 +02001036}
1037
Borislav Petkovf192c7b2011-01-10 14:24:32 +01001038static void k8_map_sysaddr_to_csrow(struct mem_ctl_info *mci, u64 sys_addr,
Borislav Petkov33ca0642012-08-30 18:01:36 +02001039 struct err_info *err)
Doug Thompsonddff8762009-04-27 16:14:52 +02001040{
Borislav Petkovf192c7b2011-01-10 14:24:32 +01001041 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompsonddff8762009-04-27 16:14:52 +02001042
Borislav Petkov33ca0642012-08-30 18:01:36 +02001043 error_address_to_page_and_offset(sys_addr, err);
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001044
1045 /*
1046 * Find out which node the error address belongs to. This may be
1047 * different from the node that detected the error.
1048 */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001049 err->src_mci = find_mc_by_sys_addr(mci, sys_addr);
1050 if (!err->src_mci) {
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001051 amd64_mc_err(mci, "failed to map error addr 0x%lx to a node\n",
1052 (unsigned long)sys_addr);
Borislav Petkov33ca0642012-08-30 18:01:36 +02001053 err->err_code = ERR_NODE;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001054 return;
1055 }
1056
1057 /* Now map the sys_addr to a CSROW */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001058 err->csrow = sys_addr_to_csrow(err->src_mci, sys_addr);
1059 if (err->csrow < 0) {
1060 err->err_code = ERR_CSROW;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001061 return;
1062 }
1063
Doug Thompsonddff8762009-04-27 16:14:52 +02001064 /* CHIPKILL enabled */
Borislav Petkovf192c7b2011-01-10 14:24:32 +01001065 if (pvt->nbcfg & NBCFG_CHIPKILL) {
Borislav Petkov33ca0642012-08-30 18:01:36 +02001066 err->channel = get_channel_from_ecc_syndrome(mci, err->syndrome);
1067 if (err->channel < 0) {
Doug Thompsonddff8762009-04-27 16:14:52 +02001068 /*
1069 * Syndrome didn't map, so we don't know which of the
1070 * 2 DIMMs is in error. So we need to ID 'both' of them
1071 * as suspect.
1072 */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001073 amd64_mc_warn(err->src_mci, "unknown syndrome 0x%04x - "
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001074 "possible error reporting race\n",
Borislav Petkov33ca0642012-08-30 18:01:36 +02001075 err->syndrome);
1076 err->err_code = ERR_CHANNEL;
Doug Thompsonddff8762009-04-27 16:14:52 +02001077 return;
1078 }
1079 } else {
1080 /*
1081 * non-chipkill ecc mode
1082 *
1083 * The k8 documentation is unclear about how to determine the
1084 * channel number when using non-chipkill memory. This method
1085 * was obtained from email communication with someone at AMD.
1086 * (Wish the email was placed in this comment - norsk)
1087 */
Borislav Petkov33ca0642012-08-30 18:01:36 +02001088 err->channel = ((sys_addr & BIT(3)) != 0);
Doug Thompsonddff8762009-04-27 16:14:52 +02001089 }
Doug Thompsonddff8762009-04-27 16:14:52 +02001090}
1091
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001092static int ddr2_cs_size(unsigned i, bool dct_width)
Doug Thompsonddff8762009-04-27 16:14:52 +02001093{
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001094 unsigned shift = 0;
Doug Thompsonddff8762009-04-27 16:14:52 +02001095
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001096 if (i <= 2)
1097 shift = i;
1098 else if (!(i & 0x1))
1099 shift = i >> 1;
Borislav Petkov1433eb92009-10-21 13:44:36 +02001100 else
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001101 shift = (i + 1) >> 1;
Doug Thompsonddff8762009-04-27 16:14:52 +02001102
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001103 return 128 << (shift + !!dct_width);
1104}
1105
1106static int k8_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001107 unsigned cs_mode, int cs_mask_nr)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001108{
1109 u32 dclr = dct ? pvt->dclr1 : pvt->dclr0;
1110
1111 if (pvt->ext_model >= K8_REV_F) {
1112 WARN_ON(cs_mode > 11);
1113 return ddr2_cs_size(cs_mode, dclr & WIDTH_128);
1114 }
1115 else if (pvt->ext_model >= K8_REV_D) {
Borislav Petkov11b0a312011-11-09 21:28:43 +01001116 unsigned diff;
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001117 WARN_ON(cs_mode > 10);
1118
Borislav Petkov11b0a312011-11-09 21:28:43 +01001119 /*
1120 * the below calculation, besides trying to win an obfuscated C
1121 * contest, maps cs_mode values to DIMM chip select sizes. The
1122 * mappings are:
1123 *
1124 * cs_mode CS size (mb)
1125 * ======= ============
1126 * 0 32
1127 * 1 64
1128 * 2 128
1129 * 3 128
1130 * 4 256
1131 * 5 512
1132 * 6 256
1133 * 7 512
1134 * 8 1024
1135 * 9 1024
1136 * 10 2048
1137 *
1138 * Basically, it calculates a value with which to shift the
1139 * smallest CS size of 32MB.
1140 *
1141 * ddr[23]_cs_size have a similar purpose.
1142 */
1143 diff = cs_mode/3 + (unsigned)(cs_mode > 5);
1144
1145 return 32 << (cs_mode - diff);
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001146 }
1147 else {
1148 WARN_ON(cs_mode > 6);
1149 return 32 << cs_mode;
1150 }
Doug Thompsonddff8762009-04-27 16:14:52 +02001151}
1152
Doug Thompson1afd3c92009-04-27 16:16:50 +02001153/*
1154 * Get the number of DCT channels in use.
1155 *
1156 * Return:
1157 * number of Memory Channels in operation
1158 * Pass back:
1159 * contents of the DCL0_LOW register
1160 */
Borislav Petkov7d20d142011-01-07 17:58:04 +01001161static int f1x_early_channel_count(struct amd64_pvt *pvt)
Doug Thompson1afd3c92009-04-27 16:16:50 +02001162{
Borislav Petkov6ba5dcd2009-10-13 19:26:55 +02001163 int i, j, channels = 0;
Doug Thompsonddff8762009-04-27 16:14:52 +02001164
Borislav Petkov7d20d142011-01-07 17:58:04 +01001165 /* On F10h, if we are in 128 bit mode, then we are using 2 channels */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001166 if (pvt->fam == 0x10 && (pvt->dclr0 & WIDTH_128))
Borislav Petkov7d20d142011-01-07 17:58:04 +01001167 return 2;
Doug Thompson1afd3c92009-04-27 16:16:50 +02001168
1169 /*
Borislav Petkovd16149e2009-10-16 19:55:49 +02001170 * Need to check if in unganged mode: In such, there are 2 channels,
1171 * but they are not in 128 bit mode and thus the above 'dclr0' status
1172 * bit will be OFF.
Doug Thompson1afd3c92009-04-27 16:16:50 +02001173 *
1174 * Need to check DCT0[0] and DCT1[0] to see if only one of them has
1175 * their CSEnable bit on. If so, then SINGLE DIMM case.
1176 */
Joe Perches956b9ba12012-04-29 17:08:39 -03001177 edac_dbg(0, "Data width is not 128 bits - need more decoding\n");
Doug Thompson1afd3c92009-04-27 16:16:50 +02001178
1179 /*
1180 * Check DRAM Bank Address Mapping values for each DIMM to see if there
1181 * is more than just one DIMM present in unganged mode. Need to check
1182 * both controllers since DIMMs can be placed in either one.
1183 */
Borislav Petkov525a1b22010-12-21 15:53:27 +01001184 for (i = 0; i < 2; i++) {
1185 u32 dbam = (i ? pvt->dbam1 : pvt->dbam0);
Doug Thompson1afd3c92009-04-27 16:16:50 +02001186
Wan Wei57a30852009-08-07 17:04:49 +02001187 for (j = 0; j < 4; j++) {
1188 if (DBAM_DIMM(j, dbam) > 0) {
1189 channels++;
1190 break;
1191 }
1192 }
Doug Thompson1afd3c92009-04-27 16:16:50 +02001193 }
1194
Borislav Petkovd16149e2009-10-16 19:55:49 +02001195 if (channels > 2)
1196 channels = 2;
1197
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02001198 amd64_info("MCT channel count: %d\n", channels);
Doug Thompson1afd3c92009-04-27 16:16:50 +02001199
1200 return channels;
Doug Thompson1afd3c92009-04-27 16:16:50 +02001201}
1202
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001203static int ddr3_cs_size(unsigned i, bool dct_width)
Doug Thompson1afd3c92009-04-27 16:16:50 +02001204{
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001205 unsigned shift = 0;
1206 int cs_size = 0;
1207
1208 if (i == 0 || i == 3 || i == 4)
1209 cs_size = -1;
1210 else if (i <= 2)
1211 shift = i;
1212 else if (i == 12)
1213 shift = 7;
1214 else if (!(i & 0x1))
1215 shift = i >> 1;
1216 else
1217 shift = (i + 1) >> 1;
1218
1219 if (cs_size != -1)
1220 cs_size = (128 * (1 << !!dct_width)) << shift;
1221
1222 return cs_size;
1223}
1224
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001225static int ddr3_lrdimm_cs_size(unsigned i, unsigned rank_multiply)
1226{
1227 unsigned shift = 0;
1228 int cs_size = 0;
1229
1230 if (i < 4 || i == 6)
1231 cs_size = -1;
1232 else if (i == 12)
1233 shift = 7;
1234 else if (!(i & 0x1))
1235 shift = i >> 1;
1236 else
1237 shift = (i + 1) >> 1;
1238
1239 if (cs_size != -1)
1240 cs_size = rank_multiply * (128 << shift);
1241
1242 return cs_size;
1243}
1244
1245static int ddr4_cs_size(unsigned i)
1246{
1247 int cs_size = 0;
1248
1249 if (i == 0)
1250 cs_size = -1;
1251 else if (i == 1)
1252 cs_size = 1024;
1253 else
1254 /* Min cs_size = 1G */
1255 cs_size = 1024 * (1 << (i >> 1));
1256
1257 return cs_size;
1258}
1259
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001260static int f10_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001261 unsigned cs_mode, int cs_mask_nr)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001262{
1263 u32 dclr = dct ? pvt->dclr1 : pvt->dclr0;
1264
1265 WARN_ON(cs_mode > 11);
Borislav Petkov1433eb92009-10-21 13:44:36 +02001266
1267 if (pvt->dchr0 & DDR3_MODE || pvt->dchr1 & DDR3_MODE)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001268 return ddr3_cs_size(cs_mode, dclr & WIDTH_128);
Borislav Petkov1433eb92009-10-21 13:44:36 +02001269 else
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001270 return ddr2_cs_size(cs_mode, dclr & WIDTH_128);
1271}
Borislav Petkov1433eb92009-10-21 13:44:36 +02001272
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001273/*
1274 * F15h supports only 64bit DCT interfaces
1275 */
1276static int f15_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001277 unsigned cs_mode, int cs_mask_nr)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001278{
1279 WARN_ON(cs_mode > 12);
1280
1281 return ddr3_cs_size(cs_mode, false);
Doug Thompson1afd3c92009-04-27 16:16:50 +02001282}
1283
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001284/* F15h M60h supports DDR4 mapping as well.. */
1285static int f15_m60h_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
1286 unsigned cs_mode, int cs_mask_nr)
1287{
1288 int cs_size;
1289 u32 dcsm = pvt->csels[dct].csmasks[cs_mask_nr];
1290
1291 WARN_ON(cs_mode > 12);
1292
1293 if (pvt->dram_type == MEM_DDR4) {
1294 if (cs_mode > 9)
1295 return -1;
1296
1297 cs_size = ddr4_cs_size(cs_mode);
1298 } else if (pvt->dram_type == MEM_LRDDR3) {
1299 unsigned rank_multiply = dcsm & 0xf;
1300
1301 if (rank_multiply == 3)
1302 rank_multiply = 4;
1303 cs_size = ddr3_lrdimm_cs_size(cs_mode, rank_multiply);
1304 } else {
1305 /* Minimum cs size is 512mb for F15hM60h*/
1306 if (cs_mode == 0x1)
1307 return -1;
1308
1309 cs_size = ddr3_cs_size(cs_mode, false);
1310 }
1311
1312 return cs_size;
1313}
1314
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001315/*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001316 * F16h and F15h model 30h have only limited cs_modes.
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001317 */
1318static int f16_dbam_to_chip_select(struct amd64_pvt *pvt, u8 dct,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001319 unsigned cs_mode, int cs_mask_nr)
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001320{
1321 WARN_ON(cs_mode > 12);
1322
1323 if (cs_mode == 6 || cs_mode == 8 ||
1324 cs_mode == 9 || cs_mode == 12)
1325 return -1;
1326 else
1327 return ddr3_cs_size(cs_mode, false);
1328}
1329
Borislav Petkov5a5d2372011-01-17 17:52:57 +01001330static void read_dram_ctl_register(struct amd64_pvt *pvt)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001331{
Doug Thompson6163b5d2009-04-27 16:20:17 +02001332
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001333 if (pvt->fam == 0xf)
Borislav Petkov5a5d2372011-01-17 17:52:57 +01001334 return;
1335
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001336 if (!amd64_read_pci_cfg(pvt->F2, DCT_SEL_LO, &pvt->dct_sel_lo)) {
Joe Perches956b9ba12012-04-29 17:08:39 -03001337 edac_dbg(0, "F2x110 (DCTSelLow): 0x%08x, High range addrs at: 0x%x\n",
1338 pvt->dct_sel_lo, dct_sel_baseaddr(pvt));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001339
Joe Perches956b9ba12012-04-29 17:08:39 -03001340 edac_dbg(0, " DCTs operate in %s mode\n",
1341 (dct_ganging_enabled(pvt) ? "ganged" : "unganged"));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001342
Borislav Petkov72381bd2009-10-09 19:14:43 +02001343 if (!dct_ganging_enabled(pvt))
Joe Perches956b9ba12012-04-29 17:08:39 -03001344 edac_dbg(0, " Address range split per DCT: %s\n",
1345 (dct_high_range_enabled(pvt) ? "yes" : "no"));
Borislav Petkov72381bd2009-10-09 19:14:43 +02001346
Joe Perches956b9ba12012-04-29 17:08:39 -03001347 edac_dbg(0, " data interleave for ECC: %s, DRAM cleared since last warm reset: %s\n",
1348 (dct_data_intlv_enabled(pvt) ? "enabled" : "disabled"),
1349 (dct_memory_cleared(pvt) ? "yes" : "no"));
Borislav Petkov72381bd2009-10-09 19:14:43 +02001350
Joe Perches956b9ba12012-04-29 17:08:39 -03001351 edac_dbg(0, " channel interleave: %s, "
1352 "interleave bits selector: 0x%x\n",
1353 (dct_interleave_enabled(pvt) ? "enabled" : "disabled"),
1354 dct_sel_interleave_addr(pvt));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001355 }
1356
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001357 amd64_read_pci_cfg(pvt->F2, DCT_SEL_HI, &pvt->dct_sel_hi);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001358}
1359
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001360/*
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001361 * Determine channel (DCT) based on the interleaving mode (see F15h M30h BKDG,
1362 * 2.10.12 Memory Interleaving Modes).
1363 */
1364static u8 f15_m30h_determine_channel(struct amd64_pvt *pvt, u64 sys_addr,
1365 u8 intlv_en, int num_dcts_intlv,
1366 u32 dct_sel)
1367{
1368 u8 channel = 0;
1369 u8 select;
1370
1371 if (!(intlv_en))
1372 return (u8)(dct_sel);
1373
1374 if (num_dcts_intlv == 2) {
1375 select = (sys_addr >> 8) & 0x3;
1376 channel = select ? 0x3 : 0;
Aravind Gopalakrishnan9d0e8d82014-01-21 15:03:36 -06001377 } else if (num_dcts_intlv == 4) {
1378 u8 intlv_addr = dct_sel_interleave_addr(pvt);
1379 switch (intlv_addr) {
1380 case 0x4:
1381 channel = (sys_addr >> 8) & 0x3;
1382 break;
1383 case 0x5:
1384 channel = (sys_addr >> 9) & 0x3;
1385 break;
1386 }
1387 }
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001388 return channel;
1389}
1390
1391/*
Borislav Petkov229a7a12010-12-09 18:57:54 +01001392 * Determine channel (DCT) based on the interleaving mode: F10h BKDG, 2.8.9 Memory
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001393 * Interleaving Modes.
1394 */
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001395static u8 f1x_determine_channel(struct amd64_pvt *pvt, u64 sys_addr,
Borislav Petkov229a7a12010-12-09 18:57:54 +01001396 bool hi_range_sel, u8 intlv_en)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001397{
Borislav Petkov151fa712011-02-21 19:33:10 +01001398 u8 dct_sel_high = (pvt->dct_sel_lo >> 1) & 1;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001399
1400 if (dct_ganging_enabled(pvt))
Borislav Petkov229a7a12010-12-09 18:57:54 +01001401 return 0;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001402
Borislav Petkov229a7a12010-12-09 18:57:54 +01001403 if (hi_range_sel)
1404 return dct_sel_high;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001405
Borislav Petkov229a7a12010-12-09 18:57:54 +01001406 /*
1407 * see F2x110[DctSelIntLvAddr] - channel interleave mode
1408 */
1409 if (dct_interleave_enabled(pvt)) {
1410 u8 intlv_addr = dct_sel_interleave_addr(pvt);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001411
Borislav Petkov229a7a12010-12-09 18:57:54 +01001412 /* return DCT select function: 0=DCT0, 1=DCT1 */
1413 if (!intlv_addr)
1414 return sys_addr >> 6 & 1;
1415
1416 if (intlv_addr & 0x2) {
1417 u8 shift = intlv_addr & 0x1 ? 9 : 6;
1418 u32 temp = hweight_long((u32) ((sys_addr >> 16) & 0x1F)) % 2;
1419
1420 return ((sys_addr >> shift) & 1) ^ temp;
1421 }
1422
1423 return (sys_addr >> (12 + hweight8(intlv_en))) & 1;
1424 }
1425
1426 if (dct_high_range_enabled(pvt))
1427 return ~dct_sel_high & 1;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001428
1429 return 0;
1430}
1431
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001432/* Convert the sys_addr to the normalized DCT address */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08001433static u64 f1x_get_norm_dct_addr(struct amd64_pvt *pvt, u8 range,
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001434 u64 sys_addr, bool hi_rng,
1435 u32 dct_sel_base_addr)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001436{
1437 u64 chan_off;
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001438 u64 dram_base = get_dram_base(pvt, range);
1439 u64 hole_off = f10_dhar_offset(pvt);
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001440 u64 dct_sel_base_off = (pvt->dct_sel_hi & 0xFFFFFC00) << 16;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001441
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001442 if (hi_rng) {
1443 /*
1444 * if
1445 * base address of high range is below 4Gb
1446 * (bits [47:27] at [31:11])
1447 * DRAM address space on this DCT is hoisted above 4Gb &&
1448 * sys_addr > 4Gb
1449 *
1450 * remove hole offset from sys_addr
1451 * else
1452 * remove high range offset from sys_addr
1453 */
1454 if ((!(dct_sel_base_addr >> 16) ||
1455 dct_sel_base_addr < dhar_base(pvt)) &&
Borislav Petkov972ea172011-02-21 19:43:02 +01001456 dhar_valid(pvt) &&
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001457 (sys_addr >= BIT_64(32)))
Borislav Petkovbc21fa52010-11-11 17:29:13 +01001458 chan_off = hole_off;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001459 else
1460 chan_off = dct_sel_base_off;
1461 } else {
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001462 /*
1463 * if
1464 * we have a valid hole &&
1465 * sys_addr > 4Gb
1466 *
1467 * remove hole
1468 * else
1469 * remove dram base to normalize to DCT address
1470 */
Borislav Petkov972ea172011-02-21 19:43:02 +01001471 if (dhar_valid(pvt) && (sys_addr >= BIT_64(32)))
Borislav Petkovbc21fa52010-11-11 17:29:13 +01001472 chan_off = hole_off;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001473 else
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001474 chan_off = dram_base;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001475 }
1476
Chen, Gong10ef6b02013-10-18 14:29:07 -07001477 return (sys_addr & GENMASK_ULL(47,6)) - (chan_off & GENMASK_ULL(47,23));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001478}
1479
Doug Thompson6163b5d2009-04-27 16:20:17 +02001480/*
1481 * checks if the csrow passed in is marked as SPARED, if so returns the new
1482 * spare row
1483 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001484static int f10_process_possible_spare(struct amd64_pvt *pvt, u8 dct, int csrow)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001485{
Borislav Petkov614ec9d2011-01-13 18:02:22 +01001486 int tmp_cs;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001487
Borislav Petkov614ec9d2011-01-13 18:02:22 +01001488 if (online_spare_swap_done(pvt, dct) &&
1489 csrow == online_spare_bad_dramcs(pvt, dct)) {
1490
1491 for_each_chip_select(tmp_cs, dct, pvt) {
1492 if (chip_select_base(tmp_cs, dct, pvt) & 0x2) {
1493 csrow = tmp_cs;
1494 break;
1495 }
1496 }
Doug Thompson6163b5d2009-04-27 16:20:17 +02001497 }
1498 return csrow;
1499}
1500
1501/*
1502 * Iterate over the DRAM DCT "base" and "mask" registers looking for a
1503 * SystemAddr match on the specified 'ChannelSelect' and 'NodeID'
1504 *
1505 * Return:
1506 * -EINVAL: NOT FOUND
1507 * 0..csrow = Chip-Select Row
1508 */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08001509static int f1x_lookup_addr_in_dct(u64 in_addr, u8 nid, u8 dct)
Doug Thompson6163b5d2009-04-27 16:20:17 +02001510{
1511 struct mem_ctl_info *mci;
1512 struct amd64_pvt *pvt;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001513 u64 cs_base, cs_mask;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001514 int cs_found = -EINVAL;
1515 int csrow;
1516
Borislav Petkov2ec591a2015-02-17 10:58:34 +01001517 mci = edac_mc_find(nid);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001518 if (!mci)
1519 return cs_found;
1520
1521 pvt = mci->pvt_info;
1522
Joe Perches956b9ba12012-04-29 17:08:39 -03001523 edac_dbg(1, "input addr: 0x%llx, DCT: %d\n", in_addr, dct);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001524
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001525 for_each_chip_select(csrow, dct, pvt) {
1526 if (!csrow_enabled(csrow, dct, pvt))
Doug Thompson6163b5d2009-04-27 16:20:17 +02001527 continue;
1528
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001529 get_cs_base_and_mask(pvt, csrow, dct, &cs_base, &cs_mask);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001530
Joe Perches956b9ba12012-04-29 17:08:39 -03001531 edac_dbg(1, " CSROW=%d CSBase=0x%llx CSMask=0x%llx\n",
1532 csrow, cs_base, cs_mask);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001533
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001534 cs_mask = ~cs_mask;
Doug Thompson6163b5d2009-04-27 16:20:17 +02001535
Joe Perches956b9ba12012-04-29 17:08:39 -03001536 edac_dbg(1, " (InputAddr & ~CSMask)=0x%llx (CSBase & ~CSMask)=0x%llx\n",
1537 (in_addr & cs_mask), (cs_base & cs_mask));
Doug Thompson6163b5d2009-04-27 16:20:17 +02001538
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001539 if ((in_addr & cs_mask) == (cs_base & cs_mask)) {
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001540 if (pvt->fam == 0x15 && pvt->model >= 0x30) {
1541 cs_found = csrow;
1542 break;
1543 }
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001544 cs_found = f10_process_possible_spare(pvt, dct, csrow);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001545
Joe Perches956b9ba12012-04-29 17:08:39 -03001546 edac_dbg(1, " MATCH csrow=%d\n", cs_found);
Doug Thompson6163b5d2009-04-27 16:20:17 +02001547 break;
1548 }
1549 }
1550 return cs_found;
1551}
1552
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001553/*
1554 * See F2x10C. Non-interleaved graphics framebuffer memory under the 16G is
1555 * swapped with a region located at the bottom of memory so that the GPU can use
1556 * the interleaved region and thus two channels.
1557 */
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001558static u64 f1x_swap_interleaved_region(struct amd64_pvt *pvt, u64 sys_addr)
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001559{
1560 u32 swap_reg, swap_base, swap_limit, rgn_size, tmp_addr;
1561
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001562 if (pvt->fam == 0x10) {
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001563 /* only revC3 and revE have that feature */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001564 if (pvt->model < 4 || (pvt->model < 0xa && pvt->stepping < 3))
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001565 return sys_addr;
1566 }
1567
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001568 amd64_read_pci_cfg(pvt->F2, SWAP_INTLV_REG, &swap_reg);
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001569
1570 if (!(swap_reg & 0x1))
1571 return sys_addr;
1572
1573 swap_base = (swap_reg >> 3) & 0x7f;
1574 swap_limit = (swap_reg >> 11) & 0x7f;
1575 rgn_size = (swap_reg >> 20) & 0x7f;
1576 tmp_addr = sys_addr >> 27;
1577
1578 if (!(sys_addr >> 34) &&
1579 (((tmp_addr >= swap_base) &&
1580 (tmp_addr <= swap_limit)) ||
1581 (tmp_addr < rgn_size)))
1582 return sys_addr ^ (u64)swap_base << 27;
1583
1584 return sys_addr;
1585}
1586
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001587/* For a given @dram_range, check if @sys_addr falls within it. */
Borislav Petkove761359a2011-02-21 19:49:01 +01001588static int f1x_match_to_this_node(struct amd64_pvt *pvt, unsigned range,
Borislav Petkov33ca0642012-08-30 18:01:36 +02001589 u64 sys_addr, int *chan_sel)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001590{
Borislav Petkov229a7a12010-12-09 18:57:54 +01001591 int cs_found = -EINVAL;
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001592 u64 chan_addr;
Borislav Petkov5d4b58e2011-01-13 16:01:13 +01001593 u32 dct_sel_base;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001594 u8 channel;
Borislav Petkov229a7a12010-12-09 18:57:54 +01001595 bool high_range = false;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001596
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001597 u8 node_id = dram_dst_node(pvt, range);
Borislav Petkov229a7a12010-12-09 18:57:54 +01001598 u8 intlv_en = dram_intlv_en(pvt, range);
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001599 u32 intlv_sel = dram_intlv_sel(pvt, range);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001600
Joe Perches956b9ba12012-04-29 17:08:39 -03001601 edac_dbg(1, "(range %d) SystemAddr= 0x%llx Limit=0x%llx\n",
1602 range, sys_addr, get_dram_limit(pvt, range));
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001603
Borislav Petkov355fba62011-01-17 13:03:26 +01001604 if (dhar_valid(pvt) &&
1605 dhar_base(pvt) <= sys_addr &&
1606 sys_addr < BIT_64(32)) {
1607 amd64_warn("Huh? Address is in the MMIO hole: 0x%016llx\n",
1608 sys_addr);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001609 return -EINVAL;
Borislav Petkov355fba62011-01-17 13:03:26 +01001610 }
1611
Borislav Petkovf030ddf2011-04-08 15:05:21 +02001612 if (intlv_en && (intlv_sel != ((sys_addr >> 12) & intlv_en)))
Borislav Petkov355fba62011-01-17 13:03:26 +01001613 return -EINVAL;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001614
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001615 sys_addr = f1x_swap_interleaved_region(pvt, sys_addr);
Borislav Petkov95b0ef52011-01-11 22:08:07 +01001616
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001617 dct_sel_base = dct_sel_baseaddr(pvt);
1618
1619 /*
1620 * check whether addresses >= DctSelBaseAddr[47:27] are to be used to
1621 * select between DCT0 and DCT1.
1622 */
1623 if (dct_high_range_enabled(pvt) &&
1624 !dct_ganging_enabled(pvt) &&
1625 ((sys_addr >> 27) >= (dct_sel_base >> 11)))
Borislav Petkov229a7a12010-12-09 18:57:54 +01001626 high_range = true;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001627
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001628 channel = f1x_determine_channel(pvt, sys_addr, high_range, intlv_en);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001629
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001630 chan_addr = f1x_get_norm_dct_addr(pvt, range, sys_addr,
Borislav Petkovc8e518d2010-12-10 19:49:19 +01001631 high_range, dct_sel_base);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001632
Borislav Petkove2f79db2011-01-13 14:57:34 +01001633 /* Remove node interleaving, see F1x120 */
1634 if (intlv_en)
1635 chan_addr = ((chan_addr >> (12 + hweight8(intlv_en))) << 12) |
1636 (chan_addr & 0xfff);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001637
Borislav Petkov5d4b58e2011-01-13 16:01:13 +01001638 /* remove channel interleave */
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001639 if (dct_interleave_enabled(pvt) &&
1640 !dct_high_range_enabled(pvt) &&
1641 !dct_ganging_enabled(pvt)) {
Borislav Petkov5d4b58e2011-01-13 16:01:13 +01001642
1643 if (dct_sel_interleave_addr(pvt) != 1) {
1644 if (dct_sel_interleave_addr(pvt) == 0x3)
1645 /* hash 9 */
1646 chan_addr = ((chan_addr >> 10) << 9) |
1647 (chan_addr & 0x1ff);
1648 else
1649 /* A[6] or hash 6 */
1650 chan_addr = ((chan_addr >> 7) << 6) |
1651 (chan_addr & 0x3f);
1652 } else
1653 /* A[12] */
1654 chan_addr = ((chan_addr >> 13) << 12) |
1655 (chan_addr & 0xfff);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001656 }
1657
Joe Perches956b9ba12012-04-29 17:08:39 -03001658 edac_dbg(1, " Normalized DCT addr: 0x%llx\n", chan_addr);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001659
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001660 cs_found = f1x_lookup_addr_in_dct(chan_addr, node_id, channel);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001661
Borislav Petkov33ca0642012-08-30 18:01:36 +02001662 if (cs_found >= 0)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001663 *chan_sel = channel;
Borislav Petkov33ca0642012-08-30 18:01:36 +02001664
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001665 return cs_found;
1666}
1667
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001668static int f15_m30h_match_to_this_node(struct amd64_pvt *pvt, unsigned range,
1669 u64 sys_addr, int *chan_sel)
1670{
1671 int cs_found = -EINVAL;
1672 int num_dcts_intlv = 0;
1673 u64 chan_addr, chan_offset;
1674 u64 dct_base, dct_limit;
1675 u32 dct_cont_base_reg, dct_cont_limit_reg, tmp;
1676 u8 channel, alias_channel, leg_mmio_hole, dct_sel, dct_offset_en;
1677
1678 u64 dhar_offset = f10_dhar_offset(pvt);
1679 u8 intlv_addr = dct_sel_interleave_addr(pvt);
1680 u8 node_id = dram_dst_node(pvt, range);
1681 u8 intlv_en = dram_intlv_en(pvt, range);
1682
1683 amd64_read_pci_cfg(pvt->F1, DRAM_CONT_BASE, &dct_cont_base_reg);
1684 amd64_read_pci_cfg(pvt->F1, DRAM_CONT_LIMIT, &dct_cont_limit_reg);
1685
1686 dct_offset_en = (u8) ((dct_cont_base_reg >> 3) & BIT(0));
1687 dct_sel = (u8) ((dct_cont_base_reg >> 4) & 0x7);
1688
1689 edac_dbg(1, "(range %d) SystemAddr= 0x%llx Limit=0x%llx\n",
1690 range, sys_addr, get_dram_limit(pvt, range));
1691
1692 if (!(get_dram_base(pvt, range) <= sys_addr) &&
1693 !(get_dram_limit(pvt, range) >= sys_addr))
1694 return -EINVAL;
1695
1696 if (dhar_valid(pvt) &&
1697 dhar_base(pvt) <= sys_addr &&
1698 sys_addr < BIT_64(32)) {
1699 amd64_warn("Huh? Address is in the MMIO hole: 0x%016llx\n",
1700 sys_addr);
1701 return -EINVAL;
1702 }
1703
1704 /* Verify sys_addr is within DCT Range. */
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05001705 dct_base = (u64) dct_sel_baseaddr(pvt);
1706 dct_limit = (dct_cont_limit_reg >> 11) & 0x1FFF;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001707
1708 if (!(dct_cont_base_reg & BIT(0)) &&
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05001709 !(dct_base <= (sys_addr >> 27) &&
1710 dct_limit >= (sys_addr >> 27)))
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001711 return -EINVAL;
1712
1713 /* Verify number of dct's that participate in channel interleaving. */
1714 num_dcts_intlv = (int) hweight8(intlv_en);
1715
1716 if (!(num_dcts_intlv % 2 == 0) || (num_dcts_intlv > 4))
1717 return -EINVAL;
1718
1719 channel = f15_m30h_determine_channel(pvt, sys_addr, intlv_en,
1720 num_dcts_intlv, dct_sel);
1721
1722 /* Verify we stay within the MAX number of channels allowed */
Aravind Gopalakrishnan7f3f5242013-12-04 11:40:11 -06001723 if (channel > 3)
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001724 return -EINVAL;
1725
1726 leg_mmio_hole = (u8) (dct_cont_base_reg >> 1 & BIT(0));
1727
1728 /* Get normalized DCT addr */
1729 if (leg_mmio_hole && (sys_addr >= BIT_64(32)))
1730 chan_offset = dhar_offset;
1731 else
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05001732 chan_offset = dct_base << 27;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001733
1734 chan_addr = sys_addr - chan_offset;
1735
1736 /* remove channel interleave */
1737 if (num_dcts_intlv == 2) {
1738 if (intlv_addr == 0x4)
1739 chan_addr = ((chan_addr >> 9) << 8) |
1740 (chan_addr & 0xff);
1741 else if (intlv_addr == 0x5)
1742 chan_addr = ((chan_addr >> 10) << 9) |
1743 (chan_addr & 0x1ff);
1744 else
1745 return -EINVAL;
1746
1747 } else if (num_dcts_intlv == 4) {
1748 if (intlv_addr == 0x4)
1749 chan_addr = ((chan_addr >> 10) << 8) |
1750 (chan_addr & 0xff);
1751 else if (intlv_addr == 0x5)
1752 chan_addr = ((chan_addr >> 11) << 9) |
1753 (chan_addr & 0x1ff);
1754 else
1755 return -EINVAL;
1756 }
1757
1758 if (dct_offset_en) {
1759 amd64_read_pci_cfg(pvt->F1,
1760 DRAM_CONT_HIGH_OFF + (int) channel * 4,
1761 &tmp);
Aravind Gopalakrishnan4fc06b32013-08-24 10:47:48 -05001762 chan_addr += (u64) ((tmp >> 11) & 0xfff) << 27;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001763 }
1764
1765 f15h_select_dct(pvt, channel);
1766
1767 edac_dbg(1, " Normalized DCT addr: 0x%llx\n", chan_addr);
1768
1769 /*
1770 * Find Chip select:
1771 * if channel = 3, then alias it to 1. This is because, in F15 M30h,
1772 * there is support for 4 DCT's, but only 2 are currently functional.
1773 * They are DCT0 and DCT3. But we have read all registers of DCT3 into
1774 * pvt->csels[1]. So we need to use '1' here to get correct info.
1775 * Refer F15 M30h BKDG Section 2.10 and 2.10.3 for clarifications.
1776 */
1777 alias_channel = (channel == 3) ? 1 : channel;
1778
1779 cs_found = f1x_lookup_addr_in_dct(chan_addr, node_id, alias_channel);
1780
1781 if (cs_found >= 0)
1782 *chan_sel = alias_channel;
1783
1784 return cs_found;
1785}
1786
1787static int f1x_translate_sysaddr_to_cs(struct amd64_pvt *pvt,
1788 u64 sys_addr,
1789 int *chan_sel)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001790{
Borislav Petkove761359a2011-02-21 19:49:01 +01001791 int cs_found = -EINVAL;
1792 unsigned range;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001793
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001794 for (range = 0; range < DRAM_RANGES; range++) {
Borislav Petkov7f19bf72010-10-21 18:52:53 +02001795 if (!dram_rw(pvt, range))
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001796 continue;
1797
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001798 if (pvt->fam == 0x15 && pvt->model >= 0x30)
1799 cs_found = f15_m30h_match_to_this_node(pvt, range,
1800 sys_addr,
1801 chan_sel);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001802
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001803 else if ((get_dram_base(pvt, range) <= sys_addr) &&
1804 (get_dram_limit(pvt, range) >= sys_addr)) {
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001805 cs_found = f1x_match_to_this_node(pvt, range,
Borislav Petkov33ca0642012-08-30 18:01:36 +02001806 sys_addr, chan_sel);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001807 if (cs_found >= 0)
1808 break;
1809 }
1810 }
1811 return cs_found;
1812}
1813
1814/*
Borislav Petkovbdc30a02009-11-13 15:10:43 +01001815 * For reference see "2.8.5 Routing DRAM Requests" in F10 BKDG. This code maps
1816 * a @sys_addr to NodeID, DCT (channel) and chip select (CSROW).
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001817 *
Borislav Petkovbdc30a02009-11-13 15:10:43 +01001818 * The @sys_addr is usually an error address received from the hardware
1819 * (MCX_ADDR).
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001820 */
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001821static void f1x_map_sysaddr_to_csrow(struct mem_ctl_info *mci, u64 sys_addr,
Borislav Petkov33ca0642012-08-30 18:01:36 +02001822 struct err_info *err)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001823{
1824 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001825
Borislav Petkov33ca0642012-08-30 18:01:36 +02001826 error_address_to_page_and_offset(sys_addr, err);
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03001827
Borislav Petkov33ca0642012-08-30 18:01:36 +02001828 err->csrow = f1x_translate_sysaddr_to_cs(pvt, sys_addr, &err->channel);
1829 if (err->csrow < 0) {
1830 err->err_code = ERR_CSROW;
Borislav Petkovbdc30a02009-11-13 15:10:43 +01001831 return;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001832 }
Borislav Petkovbdc30a02009-11-13 15:10:43 +01001833
Borislav Petkovbdc30a02009-11-13 15:10:43 +01001834 /*
1835 * We need the syndromes for channel detection only when we're
1836 * ganged. Otherwise @chan should already contain the channel at
1837 * this point.
1838 */
Borislav Petkova97fa682010-12-23 14:07:18 +01001839 if (dct_ganging_enabled(pvt))
Borislav Petkov33ca0642012-08-30 18:01:36 +02001840 err->channel = get_channel_from_ecc_syndrome(mci, err->syndrome);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001841}
1842
1843/*
Borislav Petkov8566c4d2009-10-16 13:48:28 +02001844 * debug routine to display the memory sizes of all logical DIMMs and its
Borislav Petkovcb328502010-12-22 14:28:24 +01001845 * CSROWs
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001846 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01001847static void debug_display_dimm_sizes(struct amd64_pvt *pvt, u8 ctrl)
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001848{
Borislav Petkovbb89f5a2012-09-12 18:06:00 +02001849 int dimm, size0, size1;
Borislav Petkov525a1b22010-12-21 15:53:27 +01001850 u32 *dcsb = ctrl ? pvt->csels[1].csbases : pvt->csels[0].csbases;
1851 u32 dbam = ctrl ? pvt->dbam1 : pvt->dbam0;
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001852
Borislav Petkova4b4bed2013-08-10 13:54:48 +02001853 if (pvt->fam == 0xf) {
Borislav Petkov8566c4d2009-10-16 13:48:28 +02001854 /* K8 families < revF not supported yet */
Borislav Petkov1433eb92009-10-21 13:44:36 +02001855 if (pvt->ext_model < K8_REV_F)
Borislav Petkov8566c4d2009-10-16 13:48:28 +02001856 return;
1857 else
1858 WARN_ON(ctrl != 0);
1859 }
1860
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05001861 if (pvt->fam == 0x10) {
1862 dbam = (ctrl && !dct_ganging_enabled(pvt)) ? pvt->dbam1
1863 : pvt->dbam0;
1864 dcsb = (ctrl && !dct_ganging_enabled(pvt)) ?
1865 pvt->csels[1].csbases :
1866 pvt->csels[0].csbases;
1867 } else if (ctrl) {
1868 dbam = pvt->dbam0;
1869 dcsb = pvt->csels[1].csbases;
1870 }
Joe Perches956b9ba12012-04-29 17:08:39 -03001871 edac_dbg(1, "F2x%d80 (DRAM Bank Address Mapping): 0x%08x\n",
1872 ctrl, dbam);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001873
Borislav Petkov8566c4d2009-10-16 13:48:28 +02001874 edac_printk(KERN_DEBUG, EDAC_MC, "DCT%d chip selects:\n", ctrl);
1875
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001876 /* Dump memory sizes for DIMM and its CSROWs */
1877 for (dimm = 0; dimm < 4; dimm++) {
1878
1879 size0 = 0;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001880 if (dcsb[dimm*2] & DCSB_CS_ENABLE)
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001881 /* For f15m60h, need multiplier for LRDIMM cs_size
1882 * calculation. We pass 'dimm' value to the dbam_to_cs
1883 * mapper so we can find the multiplier from the
1884 * corresponding DCSM.
1885 */
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001886 size0 = pvt->ops->dbam_to_cs(pvt, ctrl,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001887 DBAM_DIMM(dimm, dbam),
1888 dimm);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001889
1890 size1 = 0;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01001891 if (dcsb[dimm*2 + 1] & DCSB_CS_ENABLE)
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001892 size1 = pvt->ops->dbam_to_cs(pvt, ctrl,
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001893 DBAM_DIMM(dimm, dbam),
1894 dimm);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001895
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02001896 amd64_info(EDAC_MC ": %d: %5dMB %d: %5dMB\n",
Borislav Petkovbb89f5a2012-09-12 18:06:00 +02001897 dimm * 2, size0,
1898 dimm * 2 + 1, size1);
Doug Thompsonf71d0a02009-04-27 16:22:43 +02001899 }
1900}
1901
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01001902static struct amd64_family_type family_types[] = {
Doug Thompson4d376072009-04-27 16:25:05 +02001903 [K8_CPUS] = {
Borislav Petkov0092b202010-10-01 19:20:05 +02001904 .ctl_name = "K8",
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02001905 .f1_id = PCI_DEVICE_ID_AMD_K8_NB_ADDRMAP,
1906 .f3_id = PCI_DEVICE_ID_AMD_K8_NB_MISC,
Doug Thompson4d376072009-04-27 16:25:05 +02001907 .ops = {
Borislav Petkov1433eb92009-10-21 13:44:36 +02001908 .early_channel_count = k8_early_channel_count,
Borislav Petkov1433eb92009-10-21 13:44:36 +02001909 .map_sysaddr_to_csrow = k8_map_sysaddr_to_csrow,
1910 .dbam_to_cs = k8_dbam_to_chip_select,
Doug Thompson4d376072009-04-27 16:25:05 +02001911 }
1912 },
1913 [F10_CPUS] = {
Borislav Petkov0092b202010-10-01 19:20:05 +02001914 .ctl_name = "F10h",
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02001915 .f1_id = PCI_DEVICE_ID_AMD_10H_NB_MAP,
1916 .f3_id = PCI_DEVICE_ID_AMD_10H_NB_MISC,
Doug Thompson4d376072009-04-27 16:25:05 +02001917 .ops = {
Borislav Petkov7d20d142011-01-07 17:58:04 +01001918 .early_channel_count = f1x_early_channel_count,
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001919 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
Borislav Petkov1433eb92009-10-21 13:44:36 +02001920 .dbam_to_cs = f10_dbam_to_chip_select,
Borislav Petkovb2b0c602010-10-08 18:32:29 +02001921 }
1922 },
1923 [F15_CPUS] = {
1924 .ctl_name = "F15h",
Borislav Petkovdf71a052011-01-19 18:15:10 +01001925 .f1_id = PCI_DEVICE_ID_AMD_15H_NB_F1,
1926 .f3_id = PCI_DEVICE_ID_AMD_15H_NB_F3,
Borislav Petkovb2b0c602010-10-08 18:32:29 +02001927 .ops = {
Borislav Petkov7d20d142011-01-07 17:58:04 +01001928 .early_channel_count = f1x_early_channel_count,
Borislav Petkovb15f0fc2011-01-17 15:59:58 +01001929 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
Borislav Petkov41d8bfa2011-01-18 19:16:08 +01001930 .dbam_to_cs = f15_dbam_to_chip_select,
Doug Thompson4d376072009-04-27 16:25:05 +02001931 }
1932 },
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001933 [F15_M30H_CPUS] = {
1934 .ctl_name = "F15h_M30h",
1935 .f1_id = PCI_DEVICE_ID_AMD_15H_M30H_NB_F1,
1936 .f3_id = PCI_DEVICE_ID_AMD_15H_M30H_NB_F3,
1937 .ops = {
1938 .early_channel_count = f1x_early_channel_count,
1939 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
1940 .dbam_to_cs = f16_dbam_to_chip_select,
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05001941 }
1942 },
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01001943 [F15_M60H_CPUS] = {
1944 .ctl_name = "F15h_M60h",
1945 .f1_id = PCI_DEVICE_ID_AMD_15H_M60H_NB_F1,
1946 .f3_id = PCI_DEVICE_ID_AMD_15H_M60H_NB_F3,
1947 .ops = {
1948 .early_channel_count = f1x_early_channel_count,
1949 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
1950 .dbam_to_cs = f15_m60h_dbam_to_chip_select,
1951 }
1952 },
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001953 [F16_CPUS] = {
1954 .ctl_name = "F16h",
1955 .f1_id = PCI_DEVICE_ID_AMD_16H_NB_F1,
1956 .f3_id = PCI_DEVICE_ID_AMD_16H_NB_F3,
1957 .ops = {
1958 .early_channel_count = f1x_early_channel_count,
1959 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
1960 .dbam_to_cs = f16_dbam_to_chip_select,
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05001961 }
1962 },
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06001963 [F16_M30H_CPUS] = {
1964 .ctl_name = "F16h_M30h",
1965 .f1_id = PCI_DEVICE_ID_AMD_16H_M30H_NB_F1,
1966 .f3_id = PCI_DEVICE_ID_AMD_16H_M30H_NB_F3,
1967 .ops = {
1968 .early_channel_count = f1x_early_channel_count,
1969 .map_sysaddr_to_csrow = f1x_map_sysaddr_to_csrow,
1970 .dbam_to_cs = f16_dbam_to_chip_select,
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06001971 }
1972 },
Doug Thompson4d376072009-04-27 16:25:05 +02001973};
1974
Doug Thompsonb1289d62009-04-27 16:37:05 +02001975/*
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01001976 * These are tables of eigenvectors (one per line) which can be used for the
1977 * construction of the syndrome tables. The modified syndrome search algorithm
1978 * uses those to find the symbol in error and thus the DIMM.
Doug Thompsonb1289d62009-04-27 16:37:05 +02001979 *
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01001980 * Algorithm courtesy of Ross LaFetra from AMD.
Doug Thompsonb1289d62009-04-27 16:37:05 +02001981 */
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08001982static const u16 x4_vectors[] = {
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01001983 0x2f57, 0x1afe, 0x66cc, 0xdd88,
1984 0x11eb, 0x3396, 0x7f4c, 0xeac8,
1985 0x0001, 0x0002, 0x0004, 0x0008,
1986 0x1013, 0x3032, 0x4044, 0x8088,
1987 0x106b, 0x30d6, 0x70fc, 0xe0a8,
1988 0x4857, 0xc4fe, 0x13cc, 0x3288,
1989 0x1ac5, 0x2f4a, 0x5394, 0xa1e8,
1990 0x1f39, 0x251e, 0xbd6c, 0x6bd8,
1991 0x15c1, 0x2a42, 0x89ac, 0x4758,
1992 0x2b03, 0x1602, 0x4f0c, 0xca08,
1993 0x1f07, 0x3a0e, 0x6b04, 0xbd08,
1994 0x8ba7, 0x465e, 0x244c, 0x1cc8,
1995 0x2b87, 0x164e, 0x642c, 0xdc18,
1996 0x40b9, 0x80de, 0x1094, 0x20e8,
1997 0x27db, 0x1eb6, 0x9dac, 0x7b58,
1998 0x11c1, 0x2242, 0x84ac, 0x4c58,
1999 0x1be5, 0x2d7a, 0x5e34, 0xa718,
2000 0x4b39, 0x8d1e, 0x14b4, 0x28d8,
2001 0x4c97, 0xc87e, 0x11fc, 0x33a8,
2002 0x8e97, 0x497e, 0x2ffc, 0x1aa8,
2003 0x16b3, 0x3d62, 0x4f34, 0x8518,
2004 0x1e2f, 0x391a, 0x5cac, 0xf858,
2005 0x1d9f, 0x3b7a, 0x572c, 0xfe18,
2006 0x15f5, 0x2a5a, 0x5264, 0xa3b8,
2007 0x1dbb, 0x3b66, 0x715c, 0xe3f8,
2008 0x4397, 0xc27e, 0x17fc, 0x3ea8,
2009 0x1617, 0x3d3e, 0x6464, 0xb8b8,
2010 0x23ff, 0x12aa, 0xab6c, 0x56d8,
2011 0x2dfb, 0x1ba6, 0x913c, 0x7328,
2012 0x185d, 0x2ca6, 0x7914, 0x9e28,
2013 0x171b, 0x3e36, 0x7d7c, 0xebe8,
2014 0x4199, 0x82ee, 0x19f4, 0x2e58,
2015 0x4807, 0xc40e, 0x130c, 0x3208,
2016 0x1905, 0x2e0a, 0x5804, 0xac08,
2017 0x213f, 0x132a, 0xadfc, 0x5ba8,
2018 0x19a9, 0x2efe, 0xb5cc, 0x6f88,
Doug Thompsonb1289d62009-04-27 16:37:05 +02002019};
2020
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002021static const u16 x8_vectors[] = {
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002022 0x0145, 0x028a, 0x2374, 0x43c8, 0xa1f0, 0x0520, 0x0a40, 0x1480,
2023 0x0211, 0x0422, 0x0844, 0x1088, 0x01b0, 0x44e0, 0x23c0, 0xed80,
2024 0x1011, 0x0116, 0x022c, 0x0458, 0x08b0, 0x8c60, 0x2740, 0x4e80,
2025 0x0411, 0x0822, 0x1044, 0x0158, 0x02b0, 0x2360, 0x46c0, 0xab80,
2026 0x0811, 0x1022, 0x012c, 0x0258, 0x04b0, 0x4660, 0x8cc0, 0x2780,
2027 0x2071, 0x40e2, 0xa0c4, 0x0108, 0x0210, 0x0420, 0x0840, 0x1080,
2028 0x4071, 0x80e2, 0x0104, 0x0208, 0x0410, 0x0820, 0x1040, 0x2080,
2029 0x8071, 0x0102, 0x0204, 0x0408, 0x0810, 0x1020, 0x2040, 0x4080,
2030 0x019d, 0x03d6, 0x136c, 0x2198, 0x50b0, 0xb2e0, 0x0740, 0x0e80,
2031 0x0189, 0x03ea, 0x072c, 0x0e58, 0x1cb0, 0x56e0, 0x37c0, 0xf580,
2032 0x01fd, 0x0376, 0x06ec, 0x0bb8, 0x1110, 0x2220, 0x4440, 0x8880,
2033 0x0163, 0x02c6, 0x1104, 0x0758, 0x0eb0, 0x2be0, 0x6140, 0xc280,
2034 0x02fd, 0x01c6, 0x0b5c, 0x1108, 0x07b0, 0x25a0, 0x8840, 0x6180,
2035 0x0801, 0x012e, 0x025c, 0x04b8, 0x1370, 0x26e0, 0x57c0, 0xb580,
2036 0x0401, 0x0802, 0x015c, 0x02b8, 0x22b0, 0x13e0, 0x7140, 0xe280,
2037 0x0201, 0x0402, 0x0804, 0x01b8, 0x11b0, 0x31a0, 0x8040, 0x7180,
2038 0x0101, 0x0202, 0x0404, 0x0808, 0x1010, 0x2020, 0x4040, 0x8080,
2039 0x0001, 0x0002, 0x0004, 0x0008, 0x0010, 0x0020, 0x0040, 0x0080,
2040 0x0100, 0x0200, 0x0400, 0x0800, 0x1000, 0x2000, 0x4000, 0x8000,
2041};
2042
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002043static int decode_syndrome(u16 syndrome, const u16 *vectors, unsigned num_vecs,
Borislav Petkovd34a6ec2011-02-23 17:41:50 +01002044 unsigned v_dim)
Doug Thompsonb1289d62009-04-27 16:37:05 +02002045{
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002046 unsigned int i, err_sym;
Doug Thompsonb1289d62009-04-27 16:37:05 +02002047
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002048 for (err_sym = 0; err_sym < num_vecs / v_dim; err_sym++) {
2049 u16 s = syndrome;
Borislav Petkovd34a6ec2011-02-23 17:41:50 +01002050 unsigned v_idx = err_sym * v_dim;
2051 unsigned v_end = (err_sym + 1) * v_dim;
Doug Thompsonb1289d62009-04-27 16:37:05 +02002052
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002053 /* walk over all 16 bits of the syndrome */
2054 for (i = 1; i < (1U << 16); i <<= 1) {
2055
2056 /* if bit is set in that eigenvector... */
2057 if (v_idx < v_end && vectors[v_idx] & i) {
2058 u16 ev_comp = vectors[v_idx++];
2059
2060 /* ... and bit set in the modified syndrome, */
2061 if (s & i) {
2062 /* remove it. */
2063 s ^= ev_comp;
2064
2065 if (!s)
2066 return err_sym;
2067 }
2068
2069 } else if (s & i)
2070 /* can't get to zero, move to next symbol */
2071 break;
2072 }
Doug Thompsonb1289d62009-04-27 16:37:05 +02002073 }
2074
Joe Perches956b9ba12012-04-29 17:08:39 -03002075 edac_dbg(0, "syndrome(%x) not found\n", syndrome);
Doug Thompsonb1289d62009-04-27 16:37:05 +02002076 return -1;
2077}
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002078
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002079static int map_err_sym_to_channel(int err_sym, int sym_size)
2080{
2081 if (sym_size == 4)
2082 switch (err_sym) {
2083 case 0x20:
2084 case 0x21:
2085 return 0;
2086 break;
2087 case 0x22:
2088 case 0x23:
2089 return 1;
2090 break;
2091 default:
2092 return err_sym >> 4;
2093 break;
2094 }
2095 /* x8 symbols */
2096 else
2097 switch (err_sym) {
2098 /* imaginary bits not in a DIMM */
2099 case 0x10:
2100 WARN(1, KERN_ERR "Invalid error symbol: 0x%x\n",
2101 err_sym);
2102 return -1;
2103 break;
2104
2105 case 0x11:
2106 return 0;
2107 break;
2108 case 0x12:
2109 return 1;
2110 break;
2111 default:
2112 return err_sym >> 3;
2113 break;
2114 }
2115 return -1;
2116}
2117
2118static int get_channel_from_ecc_syndrome(struct mem_ctl_info *mci, u16 syndrome)
2119{
2120 struct amd64_pvt *pvt = mci->pvt_info;
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002121 int err_sym = -1;
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002122
Borislav Petkova3b7db02011-01-19 20:35:12 +01002123 if (pvt->ecc_sym_sz == 8)
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002124 err_sym = decode_syndrome(syndrome, x8_vectors,
2125 ARRAY_SIZE(x8_vectors),
Borislav Petkova3b7db02011-01-19 20:35:12 +01002126 pvt->ecc_sym_sz);
2127 else if (pvt->ecc_sym_sz == 4)
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002128 err_sym = decode_syndrome(syndrome, x4_vectors,
2129 ARRAY_SIZE(x4_vectors),
Borislav Petkova3b7db02011-01-19 20:35:12 +01002130 pvt->ecc_sym_sz);
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002131 else {
Borislav Petkova3b7db02011-01-19 20:35:12 +01002132 amd64_warn("Illegal syndrome type: %u\n", pvt->ecc_sym_sz);
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002133 return err_sym;
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002134 }
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002135
Borislav Petkova3b7db02011-01-19 20:35:12 +01002136 return map_err_sym_to_channel(err_sym, pvt->ecc_sym_sz);
Borislav Petkovbfc04ae2009-11-12 19:05:07 +01002137}
2138
Borislav Petkov33ca0642012-08-30 18:01:36 +02002139static void __log_bus_error(struct mem_ctl_info *mci, struct err_info *err,
2140 u8 ecc_type)
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002141{
Borislav Petkov33ca0642012-08-30 18:01:36 +02002142 enum hw_event_mc_err_type err_type;
2143 const char *string;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002144
Borislav Petkov33ca0642012-08-30 18:01:36 +02002145 if (ecc_type == 2)
2146 err_type = HW_EVENT_ERR_CORRECTED;
2147 else if (ecc_type == 1)
2148 err_type = HW_EVENT_ERR_UNCORRECTED;
2149 else {
2150 WARN(1, "Something is rotten in the state of Denmark.\n");
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002151 return;
2152 }
2153
Borislav Petkov33ca0642012-08-30 18:01:36 +02002154 switch (err->err_code) {
2155 case DECODE_OK:
2156 string = "";
2157 break;
2158 case ERR_NODE:
2159 string = "Failed to map error addr to a node";
2160 break;
2161 case ERR_CSROW:
2162 string = "Failed to map error addr to a csrow";
2163 break;
2164 case ERR_CHANNEL:
2165 string = "unknown syndrome - possible error reporting race";
2166 break;
2167 default:
2168 string = "WTF error";
2169 break;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002170 }
Borislav Petkov33ca0642012-08-30 18:01:36 +02002171
2172 edac_mc_handle_error(err_type, mci, 1,
2173 err->page, err->offset, err->syndrome,
2174 err->csrow, err->channel, -1,
2175 string, "");
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002176}
2177
Borislav Petkovdf781d02013-12-15 17:29:44 +01002178static inline void decode_bus_error(int node_id, struct mce *m)
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002179{
Daniel J Blueman0c510cc2015-02-17 11:34:38 +08002180 struct mem_ctl_info *mci;
2181 struct amd64_pvt *pvt;
Borislav Petkovf192c7b2011-01-10 14:24:32 +01002182 u8 ecc_type = (m->status >> 45) & 0x3;
Borislav Petkov66fed2d2012-08-09 18:41:07 +02002183 u8 xec = XEC(m->status, 0x1f);
2184 u16 ec = EC(m->status);
Borislav Petkov33ca0642012-08-30 18:01:36 +02002185 u64 sys_addr;
2186 struct err_info err;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002187
Daniel J Blueman0c510cc2015-02-17 11:34:38 +08002188 mci = edac_mc_find(node_id);
2189 if (!mci)
2190 return;
2191
2192 pvt = mci->pvt_info;
2193
Borislav Petkov66fed2d2012-08-09 18:41:07 +02002194 /* Bail out early if this was an 'observed' error */
Borislav Petkov5980bb92011-01-07 16:26:49 +01002195 if (PP(ec) == NBSL_PP_OBS)
Borislav Petkovb70ef012009-06-25 19:32:38 +02002196 return;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002197
Borislav Petkovecaf5602009-07-23 16:32:01 +02002198 /* Do only ECC errors */
2199 if (xec && xec != F10_NBSL_EXT_ERR_ECC)
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002200 return;
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002201
Borislav Petkov33ca0642012-08-30 18:01:36 +02002202 memset(&err, 0, sizeof(err));
2203
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002204 sys_addr = get_error_address(pvt, m);
Borislav Petkov33ca0642012-08-30 18:01:36 +02002205
Borislav Petkovecaf5602009-07-23 16:32:01 +02002206 if (ecc_type == 2)
Borislav Petkov33ca0642012-08-30 18:01:36 +02002207 err.syndrome = extract_syndrome(m->status);
2208
2209 pvt->ops->map_sysaddr_to_csrow(mci, sys_addr, &err);
2210
2211 __log_bus_error(mci, &err, ecc_type);
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002212}
2213
Doug Thompson0ec449e2009-04-27 19:41:25 +02002214/*
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002215 * Use pvt->F2 which contains the F2 CPU PCI device to get the related
Borislav Petkovbbd0c1f62010-10-01 19:27:58 +02002216 * F1 (AddrMap) and F3 (Misc) devices. Return negative value on error.
Doug Thompson0ec449e2009-04-27 19:41:25 +02002217 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02002218static int reserve_mc_sibling_devs(struct amd64_pvt *pvt, u16 f1_id, u16 f3_id)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002219{
Doug Thompson0ec449e2009-04-27 19:41:25 +02002220 /* Reserve the ADDRESS MAP Device */
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002221 pvt->F1 = pci_get_related_function(pvt->F2->vendor, f1_id, pvt->F2);
2222 if (!pvt->F1) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002223 amd64_err("error address map device not found: "
2224 "vendor %x device 0x%x (broken BIOS?)\n",
2225 PCI_VENDOR_ID_AMD, f1_id);
Borislav Petkovbbd0c1f62010-10-01 19:27:58 +02002226 return -ENODEV;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002227 }
2228
2229 /* Reserve the MISC Device */
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002230 pvt->F3 = pci_get_related_function(pvt->F2->vendor, f3_id, pvt->F2);
2231 if (!pvt->F3) {
2232 pci_dev_put(pvt->F1);
2233 pvt->F1 = NULL;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002234
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002235 amd64_err("error F3 device not found: "
2236 "vendor %x device 0x%x (broken BIOS?)\n",
2237 PCI_VENDOR_ID_AMD, f3_id);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002238
Borislav Petkovbbd0c1f62010-10-01 19:27:58 +02002239 return -ENODEV;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002240 }
Joe Perches956b9ba12012-04-29 17:08:39 -03002241 edac_dbg(1, "F1: %s\n", pci_name(pvt->F1));
2242 edac_dbg(1, "F2: %s\n", pci_name(pvt->F2));
2243 edac_dbg(1, "F3: %s\n", pci_name(pvt->F3));
Doug Thompson0ec449e2009-04-27 19:41:25 +02002244
2245 return 0;
2246}
2247
Borislav Petkov360b7f32010-10-15 19:25:38 +02002248static void free_mc_sibling_devs(struct amd64_pvt *pvt)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002249{
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002250 pci_dev_put(pvt->F1);
2251 pci_dev_put(pvt->F3);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002252}
2253
2254/*
2255 * Retrieve the hardware registers of the memory controller (this includes the
2256 * 'Address Map' and 'Misc' device regs)
2257 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02002258static void read_mc_regs(struct amd64_pvt *pvt)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002259{
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002260 unsigned range;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002261 u64 msr_val;
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002262 u32 tmp;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002263
2264 /*
2265 * Retrieve TOP_MEM and TOP_MEM2; no masking off of reserved bits since
2266 * those are Read-As-Zero
2267 */
Borislav Petkove97f8bb2009-10-12 15:27:45 +02002268 rdmsrl(MSR_K8_TOP_MEM1, pvt->top_mem);
Joe Perches956b9ba12012-04-29 17:08:39 -03002269 edac_dbg(0, " TOP_MEM: 0x%016llx\n", pvt->top_mem);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002270
2271 /* check first whether TOP_MEM2 is enabled */
2272 rdmsrl(MSR_K8_SYSCFG, msr_val);
2273 if (msr_val & (1U << 21)) {
Borislav Petkove97f8bb2009-10-12 15:27:45 +02002274 rdmsrl(MSR_K8_TOP_MEM2, pvt->top_mem2);
Joe Perches956b9ba12012-04-29 17:08:39 -03002275 edac_dbg(0, " TOP_MEM2: 0x%016llx\n", pvt->top_mem2);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002276 } else
Joe Perches956b9ba12012-04-29 17:08:39 -03002277 edac_dbg(0, " TOP_MEM2 disabled\n");
Doug Thompson0ec449e2009-04-27 19:41:25 +02002278
Borislav Petkov5980bb92011-01-07 16:26:49 +01002279 amd64_read_pci_cfg(pvt->F3, NBCAP, &pvt->nbcap);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002280
Borislav Petkov5a5d2372011-01-17 17:52:57 +01002281 read_dram_ctl_register(pvt);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002282
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002283 for (range = 0; range < DRAM_RANGES; range++) {
2284 u8 rw;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002285
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002286 /* read settings for this DRAM range */
2287 read_dram_base_limit_regs(pvt, range);
Borislav Petkove97f8bb2009-10-12 15:27:45 +02002288
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002289 rw = dram_rw(pvt, range);
2290 if (!rw)
2291 continue;
2292
Joe Perches956b9ba12012-04-29 17:08:39 -03002293 edac_dbg(1, " DRAM range[%d], base: 0x%016llx; limit: 0x%016llx\n",
2294 range,
2295 get_dram_base(pvt, range),
2296 get_dram_limit(pvt, range));
Borislav Petkov7f19bf72010-10-21 18:52:53 +02002297
Joe Perches956b9ba12012-04-29 17:08:39 -03002298 edac_dbg(1, " IntlvEn=%s; Range access: %s%s IntlvSel=%d DstNode=%d\n",
2299 dram_intlv_en(pvt, range) ? "Enabled" : "Disabled",
2300 (rw & 0x1) ? "R" : "-",
2301 (rw & 0x2) ? "W" : "-",
2302 dram_intlv_sel(pvt, range),
2303 dram_dst_node(pvt, range));
Doug Thompson0ec449e2009-04-27 19:41:25 +02002304 }
2305
Borislav Petkovb2b0c602010-10-08 18:32:29 +02002306 read_dct_base_mask(pvt);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002307
Borislav Petkovbc21fa52010-11-11 17:29:13 +01002308 amd64_read_pci_cfg(pvt->F1, DHAR, &pvt->dhar);
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002309 amd64_read_dct_pci_cfg(pvt, 0, DBAM0, &pvt->dbam0);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002310
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002311 amd64_read_pci_cfg(pvt->F3, F10_ONLINE_SPARE, &pvt->online_spare);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002312
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002313 amd64_read_dct_pci_cfg(pvt, 0, DCLR0, &pvt->dclr0);
2314 amd64_read_dct_pci_cfg(pvt, 0, DCHR0, &pvt->dchr0);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002315
Borislav Petkov78da1212010-12-22 19:31:45 +01002316 if (!dct_ganging_enabled(pvt)) {
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002317 amd64_read_dct_pci_cfg(pvt, 1, DCLR0, &pvt->dclr1);
2318 amd64_read_dct_pci_cfg(pvt, 1, DCHR0, &pvt->dchr1);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002319 }
Borislav Petkovad6a32e2010-03-09 12:46:00 +01002320
Borislav Petkova3b7db02011-01-19 20:35:12 +01002321 pvt->ecc_sym_sz = 4;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002322 determine_memory_type(pvt);
2323 edac_dbg(1, " DIMM type: %s\n", edac_mem_types[pvt->dram_type]);
Borislav Petkova3b7db02011-01-19 20:35:12 +01002324
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002325 if (pvt->fam >= 0x10) {
Borislav Petkovb2b0c602010-10-08 18:32:29 +02002326 amd64_read_pci_cfg(pvt->F3, EXT_NB_MCA_CFG, &tmp);
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002327 /* F16h has only DCT0, so no need to read dbam1 */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002328 if (pvt->fam != 0x16)
Aravind Gopalakrishnan7981a282014-09-15 11:37:38 -05002329 amd64_read_dct_pci_cfg(pvt, 1, DBAM0, &pvt->dbam1);
Borislav Petkova3b7db02011-01-19 20:35:12 +01002330
2331 /* F10h, revD and later can do x8 ECC too */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002332 if ((pvt->fam > 0x10 || pvt->model > 7) && tmp & BIT(25))
Borislav Petkova3b7db02011-01-19 20:35:12 +01002333 pvt->ecc_sym_sz = 8;
Borislav Petkov525a1b22010-12-21 15:53:27 +01002334 }
Borislav Petkovb2b0c602010-10-08 18:32:29 +02002335 dump_misc_regs(pvt);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002336}
2337
2338/*
2339 * NOTE: CPU Revision Dependent code
2340 *
2341 * Input:
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002342 * @csrow_nr ChipSelect Row Number (0..NUM_CHIPSELECTS-1)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002343 * k8 private pointer to -->
2344 * DRAM Bank Address mapping register
2345 * node_id
2346 * DCL register where dual_channel_active is
2347 *
2348 * The DBAM register consists of 4 sets of 4 bits each definitions:
2349 *
2350 * Bits: CSROWs
2351 * 0-3 CSROWs 0 and 1
2352 * 4-7 CSROWs 2 and 3
2353 * 8-11 CSROWs 4 and 5
2354 * 12-15 CSROWs 6 and 7
2355 *
2356 * Values range from: 0 to 15
2357 * The meaning of the values depends on CPU revision and dual-channel state,
2358 * see relevant BKDG more info.
2359 *
2360 * The memory controller provides for total of only 8 CSROWs in its current
2361 * architecture. Each "pair" of CSROWs normally represents just one DIMM in
2362 * single channel or two (2) DIMMs in dual channel mode.
2363 *
2364 * The following code logic collapses the various tables for CSROW based on CPU
2365 * revision.
2366 *
2367 * Returns:
2368 * The number of PAGE_SIZE pages on the specified CSROW number it
2369 * encompasses
2370 *
2371 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002372static u32 get_csrow_nr_pages(struct amd64_pvt *pvt, u8 dct, int csrow_nr)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002373{
Borislav Petkov1433eb92009-10-21 13:44:36 +02002374 u32 cs_mode, nr_pages;
Ashish Shenoyf92cae42012-02-22 17:20:38 -08002375 u32 dbam = dct ? pvt->dbam1 : pvt->dbam0;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002376
Borislav Petkov10de6492012-09-12 19:00:38 +02002377
Doug Thompson0ec449e2009-04-27 19:41:25 +02002378 /*
2379 * The math on this doesn't look right on the surface because x/2*4 can
2380 * be simplified to x*2 but this expression makes use of the fact that
2381 * it is integral math where 1/2=0. This intermediate value becomes the
2382 * number of bits to shift the DBAM register to extract the proper CSROW
2383 * field.
2384 */
Borislav Petkov0a5dfc32012-09-12 18:16:01 +02002385 cs_mode = DBAM_DIMM(csrow_nr / 2, dbam);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002386
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002387 nr_pages = pvt->ops->dbam_to_cs(pvt, dct, cs_mode, (csrow_nr / 2))
2388 << (20 - PAGE_SHIFT);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002389
Borislav Petkov10de6492012-09-12 19:00:38 +02002390 edac_dbg(0, "csrow: %d, channel: %d, DBAM idx: %d\n",
2391 csrow_nr, dct, cs_mode);
2392 edac_dbg(0, "nr_pages/channel: %u\n", nr_pages);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002393
2394 return nr_pages;
2395}
2396
2397/*
2398 * Initialize the array of csrow attribute instances, based on the values
2399 * from pci config hardware registers.
2400 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02002401static int init_csrows(struct mem_ctl_info *mci)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002402{
Borislav Petkov10de6492012-09-12 19:00:38 +02002403 struct amd64_pvt *pvt = mci->pvt_info;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002404 struct csrow_info *csrow;
Mauro Carvalho Chehabde3910eb2012-04-24 15:05:43 -03002405 struct dimm_info *dimm;
Mauro Carvalho Chehab084a4fc2012-01-27 18:38:08 -03002406 enum edac_type edac_mode;
Borislav Petkov10de6492012-09-12 19:00:38 +02002407 int i, j, empty = 1;
Mauro Carvalho Chehaba895bf82012-01-28 09:09:38 -03002408 int nr_pages = 0;
Borislav Petkov10de6492012-09-12 19:00:38 +02002409 u32 val;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002410
Borislav Petkova97fa682010-12-23 14:07:18 +01002411 amd64_read_pci_cfg(pvt->F3, NBCFG, &val);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002412
Borislav Petkov2299ef72010-10-15 17:44:04 +02002413 pvt->nbcfg = val;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002414
Joe Perches956b9ba12012-04-29 17:08:39 -03002415 edac_dbg(0, "node %d, NBCFG=0x%08x[ChipKillEccCap: %d|DramEccEn: %d]\n",
2416 pvt->mc_node_id, val,
2417 !!(val & NBCFG_CHIPKILL), !!(val & NBCFG_ECC_ENABLE));
Doug Thompson0ec449e2009-04-27 19:41:25 +02002418
Borislav Petkov10de6492012-09-12 19:00:38 +02002419 /*
2420 * We iterate over DCT0 here but we look at DCT1 in parallel, if needed.
2421 */
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002422 for_each_chip_select(i, 0, pvt) {
Borislav Petkov10de6492012-09-12 19:00:38 +02002423 bool row_dct0 = !!csrow_enabled(i, 0, pvt);
2424 bool row_dct1 = false;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002425
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002426 if (pvt->fam != 0xf)
Borislav Petkov10de6492012-09-12 19:00:38 +02002427 row_dct1 = !!csrow_enabled(i, 1, pvt);
2428
2429 if (!row_dct0 && !row_dct1)
Doug Thompson0ec449e2009-04-27 19:41:25 +02002430 continue;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002431
Borislav Petkov10de6492012-09-12 19:00:38 +02002432 csrow = mci->csrows[i];
Doug Thompson0ec449e2009-04-27 19:41:25 +02002433 empty = 0;
Borislav Petkov11c75ea2010-11-29 19:49:02 +01002434
Borislav Petkov10de6492012-09-12 19:00:38 +02002435 edac_dbg(1, "MC node: %d, csrow: %d\n",
2436 pvt->mc_node_id, i);
2437
Mauro Carvalho Chehab1eef1282013-03-11 09:07:46 -03002438 if (row_dct0) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002439 nr_pages = get_csrow_nr_pages(pvt, 0, i);
Mauro Carvalho Chehab1eef1282013-03-11 09:07:46 -03002440 csrow->channels[0]->dimm->nr_pages = nr_pages;
2441 }
Borislav Petkov10de6492012-09-12 19:00:38 +02002442
2443 /* K8 has only one DCT */
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002444 if (pvt->fam != 0xf && row_dct1) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002445 int row_dct1_pages = get_csrow_nr_pages(pvt, 1, i);
Mauro Carvalho Chehab1eef1282013-03-11 09:07:46 -03002446
2447 csrow->channels[1]->dimm->nr_pages = row_dct1_pages;
2448 nr_pages += row_dct1_pages;
2449 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002450
Borislav Petkov10de6492012-09-12 19:00:38 +02002451 edac_dbg(1, "Total csrow%d pages: %u\n", i, nr_pages);
Doug Thompson0ec449e2009-04-27 19:41:25 +02002452
2453 /*
2454 * determine whether CHIPKILL or JUST ECC or NO ECC is operating
2455 */
Borislav Petkova97fa682010-12-23 14:07:18 +01002456 if (pvt->nbcfg & NBCFG_ECC_ENABLE)
Mauro Carvalho Chehab084a4fc2012-01-27 18:38:08 -03002457 edac_mode = (pvt->nbcfg & NBCFG_CHIPKILL) ?
2458 EDAC_S4ECD4ED : EDAC_SECDED;
Doug Thompson0ec449e2009-04-27 19:41:25 +02002459 else
Mauro Carvalho Chehab084a4fc2012-01-27 18:38:08 -03002460 edac_mode = EDAC_NONE;
2461
2462 for (j = 0; j < pvt->channel_count; j++) {
Mauro Carvalho Chehabde3910eb2012-04-24 15:05:43 -03002463 dimm = csrow->channels[j]->dimm;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002464 dimm->mtype = pvt->dram_type;
Mauro Carvalho Chehabde3910eb2012-04-24 15:05:43 -03002465 dimm->edac_mode = edac_mode;
Mauro Carvalho Chehab084a4fc2012-01-27 18:38:08 -03002466 }
Doug Thompson0ec449e2009-04-27 19:41:25 +02002467 }
2468
2469 return empty;
2470}
Doug Thompsond27bf6f2009-05-06 17:55:27 +02002471
Borislav Petkov06724532009-09-16 13:05:46 +02002472/* get all cores on this DCT */
Daniel J Blueman8b84c8d2012-11-27 14:32:10 +08002473static void get_cpus_on_this_dct_cpumask(struct cpumask *mask, u16 nid)
Doug Thompsonf9431992009-04-27 19:46:08 +02002474{
Borislav Petkov06724532009-09-16 13:05:46 +02002475 int cpu;
Doug Thompsonf9431992009-04-27 19:46:08 +02002476
Borislav Petkov06724532009-09-16 13:05:46 +02002477 for_each_online_cpu(cpu)
2478 if (amd_get_nb_id(cpu) == nid)
2479 cpumask_set_cpu(cpu, mask);
Doug Thompsonf9431992009-04-27 19:46:08 +02002480}
2481
2482/* check MCG_CTL on all the cpus on this node */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002483static bool nb_mce_bank_enabled_on_node(u16 nid)
Doug Thompsonf9431992009-04-27 19:46:08 +02002484{
Rusty Russellba578cb2009-11-03 14:56:35 +10302485 cpumask_var_t mask;
Borislav Petkov50542252009-12-11 18:14:40 +01002486 int cpu, nbe;
Borislav Petkov06724532009-09-16 13:05:46 +02002487 bool ret = false;
Doug Thompsonf9431992009-04-27 19:46:08 +02002488
Rusty Russellba578cb2009-11-03 14:56:35 +10302489 if (!zalloc_cpumask_var(&mask, GFP_KERNEL)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002490 amd64_warn("%s: Error allocating mask\n", __func__);
Rusty Russellba578cb2009-11-03 14:56:35 +10302491 return false;
2492 }
Borislav Petkov06724532009-09-16 13:05:46 +02002493
Rusty Russellba578cb2009-11-03 14:56:35 +10302494 get_cpus_on_this_dct_cpumask(mask, nid);
Borislav Petkov06724532009-09-16 13:05:46 +02002495
Rusty Russellba578cb2009-11-03 14:56:35 +10302496 rdmsr_on_cpus(mask, MSR_IA32_MCG_CTL, msrs);
Borislav Petkov06724532009-09-16 13:05:46 +02002497
Rusty Russellba578cb2009-11-03 14:56:35 +10302498 for_each_cpu(cpu, mask) {
Borislav Petkov50542252009-12-11 18:14:40 +01002499 struct msr *reg = per_cpu_ptr(msrs, cpu);
Borislav Petkov5980bb92011-01-07 16:26:49 +01002500 nbe = reg->l & MSR_MCGCTL_NBE;
Borislav Petkov06724532009-09-16 13:05:46 +02002501
Joe Perches956b9ba12012-04-29 17:08:39 -03002502 edac_dbg(0, "core: %u, MCG_CTL: 0x%llx, NB MSR is %s\n",
2503 cpu, reg->q,
2504 (nbe ? "enabled" : "disabled"));
Borislav Petkov06724532009-09-16 13:05:46 +02002505
2506 if (!nbe)
2507 goto out;
Borislav Petkov06724532009-09-16 13:05:46 +02002508 }
2509 ret = true;
2510
2511out:
Rusty Russellba578cb2009-11-03 14:56:35 +10302512 free_cpumask_var(mask);
Doug Thompsonf9431992009-04-27 19:46:08 +02002513 return ret;
2514}
2515
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002516static int toggle_ecc_err_reporting(struct ecc_settings *s, u16 nid, bool on)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002517{
2518 cpumask_var_t cmask;
Borislav Petkov50542252009-12-11 18:14:40 +01002519 int cpu;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002520
2521 if (!zalloc_cpumask_var(&cmask, GFP_KERNEL)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002522 amd64_warn("%s: error allocating mask\n", __func__);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002523 return false;
2524 }
2525
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002526 get_cpus_on_this_dct_cpumask(cmask, nid);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002527
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002528 rdmsr_on_cpus(cmask, MSR_IA32_MCG_CTL, msrs);
2529
2530 for_each_cpu(cpu, cmask) {
2531
Borislav Petkov50542252009-12-11 18:14:40 +01002532 struct msr *reg = per_cpu_ptr(msrs, cpu);
2533
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002534 if (on) {
Borislav Petkov5980bb92011-01-07 16:26:49 +01002535 if (reg->l & MSR_MCGCTL_NBE)
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002536 s->flags.nb_mce_enable = 1;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002537
Borislav Petkov5980bb92011-01-07 16:26:49 +01002538 reg->l |= MSR_MCGCTL_NBE;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002539 } else {
2540 /*
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01002541 * Turn off NB MCE reporting only when it was off before
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002542 */
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002543 if (!s->flags.nb_mce_enable)
Borislav Petkov5980bb92011-01-07 16:26:49 +01002544 reg->l &= ~MSR_MCGCTL_NBE;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002545 }
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002546 }
2547 wrmsr_on_cpus(cmask, MSR_IA32_MCG_CTL, msrs);
2548
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002549 free_cpumask_var(cmask);
2550
2551 return 0;
2552}
2553
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002554static bool enable_ecc_error_reporting(struct ecc_settings *s, u16 nid,
Borislav Petkov2299ef72010-10-15 17:44:04 +02002555 struct pci_dev *F3)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002556{
Borislav Petkov2299ef72010-10-15 17:44:04 +02002557 bool ret = true;
Borislav Petkovc9f4f262010-12-22 19:48:20 +01002558 u32 value, mask = 0x3; /* UECC/CECC enable */
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002559
Borislav Petkov2299ef72010-10-15 17:44:04 +02002560 if (toggle_ecc_err_reporting(s, nid, ON)) {
2561 amd64_warn("Error enabling ECC reporting over MCGCTL!\n");
2562 return false;
2563 }
2564
Borislav Petkovc9f4f262010-12-22 19:48:20 +01002565 amd64_read_pci_cfg(F3, NBCTL, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002566
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002567 s->old_nbctl = value & mask;
2568 s->nbctl_valid = true;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002569
2570 value |= mask;
Borislav Petkovc9f4f262010-12-22 19:48:20 +01002571 amd64_write_pci_cfg(F3, NBCTL, value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002572
Borislav Petkova97fa682010-12-23 14:07:18 +01002573 amd64_read_pci_cfg(F3, NBCFG, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002574
Joe Perches956b9ba12012-04-29 17:08:39 -03002575 edac_dbg(0, "1: node %d, NBCFG=0x%08x[DramEccEn: %d]\n",
2576 nid, value, !!(value & NBCFG_ECC_ENABLE));
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002577
Borislav Petkova97fa682010-12-23 14:07:18 +01002578 if (!(value & NBCFG_ECC_ENABLE)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002579 amd64_warn("DRAM ECC disabled on this node, enabling...\n");
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002580
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002581 s->flags.nb_ecc_prev = 0;
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01002582
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002583 /* Attempt to turn on DRAM ECC Enable */
Borislav Petkova97fa682010-12-23 14:07:18 +01002584 value |= NBCFG_ECC_ENABLE;
2585 amd64_write_pci_cfg(F3, NBCFG, value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002586
Borislav Petkova97fa682010-12-23 14:07:18 +01002587 amd64_read_pci_cfg(F3, NBCFG, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002588
Borislav Petkova97fa682010-12-23 14:07:18 +01002589 if (!(value & NBCFG_ECC_ENABLE)) {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002590 amd64_warn("Hardware rejected DRAM ECC enable,"
2591 "check memory DIMM configuration.\n");
Borislav Petkov2299ef72010-10-15 17:44:04 +02002592 ret = false;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002593 } else {
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002594 amd64_info("Hardware accepted DRAM ECC Enable\n");
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002595 }
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01002596 } else {
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002597 s->flags.nb_ecc_prev = 1;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002598 }
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01002599
Joe Perches956b9ba12012-04-29 17:08:39 -03002600 edac_dbg(0, "2: node %d, NBCFG=0x%08x[DramEccEn: %d]\n",
2601 nid, value, !!(value & NBCFG_ECC_ENABLE));
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002602
Borislav Petkov2299ef72010-10-15 17:44:04 +02002603 return ret;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002604}
2605
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002606static void restore_ecc_error_reporting(struct ecc_settings *s, u16 nid,
Borislav Petkov360b7f32010-10-15 19:25:38 +02002607 struct pci_dev *F3)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002608{
Borislav Petkovc9f4f262010-12-22 19:48:20 +01002609 u32 value, mask = 0x3; /* UECC/CECC enable */
2610
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002611
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002612 if (!s->nbctl_valid)
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002613 return;
2614
Borislav Petkovc9f4f262010-12-22 19:48:20 +01002615 amd64_read_pci_cfg(F3, NBCTL, &value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002616 value &= ~mask;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002617 value |= s->old_nbctl;
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002618
Borislav Petkovc9f4f262010-12-22 19:48:20 +01002619 amd64_write_pci_cfg(F3, NBCTL, value);
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002620
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002621 /* restore previous BIOS DRAM ECC "off" setting we force-enabled */
2622 if (!s->flags.nb_ecc_prev) {
Borislav Petkova97fa682010-12-23 14:07:18 +01002623 amd64_read_pci_cfg(F3, NBCFG, &value);
2624 value &= ~NBCFG_ECC_ENABLE;
2625 amd64_write_pci_cfg(F3, NBCFG, value);
Borislav Petkovd95cf4d2010-02-24 14:49:47 +01002626 }
2627
2628 /* restore the NB Enable MCGCTL bit */
Borislav Petkov2299ef72010-10-15 17:44:04 +02002629 if (toggle_ecc_err_reporting(s, nid, OFF))
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002630 amd64_warn("Error restoring NB MCGCTL settings!\n");
Borislav Petkovf6d6ae92009-11-03 15:29:26 +01002631}
2632
Doug Thompsonf9431992009-04-27 19:46:08 +02002633/*
Borislav Petkov2299ef72010-10-15 17:44:04 +02002634 * EDAC requires that the BIOS have ECC enabled before
2635 * taking over the processing of ECC errors. A command line
2636 * option allows to force-enable hardware ECC later in
2637 * enable_ecc_error_reporting().
Doug Thompsonf9431992009-04-27 19:46:08 +02002638 */
Borislav Petkovcab4d272010-02-11 17:15:57 +01002639static const char *ecc_msg =
2640 "ECC disabled in the BIOS or no ECC capability, module will not load.\n"
2641 " Either enable ECC checking or force module loading by setting "
2642 "'ecc_enable_override'.\n"
2643 " (Note that use of the override may cause unknown side effects.)\n";
Borislav Petkovbe3468e2009-08-05 15:47:22 +02002644
Daniel J Bluemanc7e53012012-11-30 16:44:20 +08002645static bool ecc_enabled(struct pci_dev *F3, u16 nid)
Doug Thompsonf9431992009-04-27 19:46:08 +02002646{
2647 u32 value;
Borislav Petkov2299ef72010-10-15 17:44:04 +02002648 u8 ecc_en = 0;
Borislav Petkov06724532009-09-16 13:05:46 +02002649 bool nb_mce_en = false;
Doug Thompsonf9431992009-04-27 19:46:08 +02002650
Borislav Petkova97fa682010-12-23 14:07:18 +01002651 amd64_read_pci_cfg(F3, NBCFG, &value);
Doug Thompsonf9431992009-04-27 19:46:08 +02002652
Borislav Petkova97fa682010-12-23 14:07:18 +01002653 ecc_en = !!(value & NBCFG_ECC_ENABLE);
Borislav Petkov2299ef72010-10-15 17:44:04 +02002654 amd64_info("DRAM ECC %s.\n", (ecc_en ? "enabled" : "disabled"));
Doug Thompsonf9431992009-04-27 19:46:08 +02002655
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002656 nb_mce_en = nb_mce_bank_enabled_on_node(nid);
Borislav Petkov06724532009-09-16 13:05:46 +02002657 if (!nb_mce_en)
Borislav Petkov2299ef72010-10-15 17:44:04 +02002658 amd64_notice("NB MCE bank disabled, set MSR "
2659 "0x%08x[4] on node %d to enable.\n",
2660 MSR_IA32_MCG_CTL, nid);
Doug Thompsonf9431992009-04-27 19:46:08 +02002661
Borislav Petkov2299ef72010-10-15 17:44:04 +02002662 if (!ecc_en || !nb_mce_en) {
2663 amd64_notice("%s", ecc_msg);
2664 return false;
Borislav Petkov43f5e682009-12-21 18:55:18 +01002665 }
Borislav Petkov2299ef72010-10-15 17:44:04 +02002666 return true;
Doug Thompsonf9431992009-04-27 19:46:08 +02002667}
2668
Borislav Petkovdf71a052011-01-19 18:15:10 +01002669static void setup_mci_misc_attrs(struct mem_ctl_info *mci,
2670 struct amd64_family_type *fam)
Doug Thompson7d6034d2009-04-27 20:01:01 +02002671{
2672 struct amd64_pvt *pvt = mci->pvt_info;
2673
2674 mci->mtype_cap = MEM_FLAG_DDR2 | MEM_FLAG_RDDR2;
2675 mci->edac_ctl_cap = EDAC_FLAG_NONE;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002676
Borislav Petkov5980bb92011-01-07 16:26:49 +01002677 if (pvt->nbcap & NBCAP_SECDED)
Doug Thompson7d6034d2009-04-27 20:01:01 +02002678 mci->edac_ctl_cap |= EDAC_FLAG_SECDED;
2679
Borislav Petkov5980bb92011-01-07 16:26:49 +01002680 if (pvt->nbcap & NBCAP_CHIPKILL)
Doug Thompson7d6034d2009-04-27 20:01:01 +02002681 mci->edac_ctl_cap |= EDAC_FLAG_S4ECD4ED;
2682
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002683 mci->edac_cap = determine_edac_cap(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002684 mci->mod_name = EDAC_MOD_STR;
2685 mci->mod_ver = EDAC_AMD64_VERSION;
Borislav Petkovdf71a052011-01-19 18:15:10 +01002686 mci->ctl_name = fam->ctl_name;
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002687 mci->dev_name = pci_name(pvt->F2);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002688 mci->ctl_page_to_phys = NULL;
2689
Doug Thompson7d6034d2009-04-27 20:01:01 +02002690 /* memory scrubber interface */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002691 mci->set_sdram_scrub_rate = set_scrub_rate;
2692 mci->get_sdram_scrub_rate = get_scrub_rate;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002693}
2694
Borislav Petkov0092b202010-10-01 19:20:05 +02002695/*
2696 * returns a pointer to the family descriptor on success, NULL otherwise.
2697 */
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002698static struct amd64_family_type *per_family_init(struct amd64_pvt *pvt)
Borislav Petkov395ae782010-10-01 18:38:19 +02002699{
Borislav Petkov0092b202010-10-01 19:20:05 +02002700 struct amd64_family_type *fam_type = NULL;
2701
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002702 pvt->ext_model = boot_cpu_data.x86_model >> 4;
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002703 pvt->stepping = boot_cpu_data.x86_mask;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002704 pvt->model = boot_cpu_data.x86_model;
2705 pvt->fam = boot_cpu_data.x86;
2706
2707 switch (pvt->fam) {
Borislav Petkov395ae782010-10-01 18:38:19 +02002708 case 0xf:
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002709 fam_type = &family_types[K8_CPUS];
2710 pvt->ops = &family_types[K8_CPUS].ops;
Borislav Petkov395ae782010-10-01 18:38:19 +02002711 break;
Borislav Petkovdf71a052011-01-19 18:15:10 +01002712
Borislav Petkov395ae782010-10-01 18:38:19 +02002713 case 0x10:
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002714 fam_type = &family_types[F10_CPUS];
2715 pvt->ops = &family_types[F10_CPUS].ops;
Borislav Petkovdf71a052011-01-19 18:15:10 +01002716 break;
2717
2718 case 0x15:
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002719 if (pvt->model == 0x30) {
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002720 fam_type = &family_types[F15_M30H_CPUS];
2721 pvt->ops = &family_types[F15_M30H_CPUS].ops;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002722 break;
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002723 } else if (pvt->model == 0x60) {
2724 fam_type = &family_types[F15_M60H_CPUS];
2725 pvt->ops = &family_types[F15_M60H_CPUS].ops;
2726 break;
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002727 }
2728
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002729 fam_type = &family_types[F15_CPUS];
2730 pvt->ops = &family_types[F15_CPUS].ops;
Borislav Petkov395ae782010-10-01 18:38:19 +02002731 break;
2732
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05002733 case 0x16:
Aravind Gopalakrishnan85a88852014-02-20 10:28:46 -06002734 if (pvt->model == 0x30) {
2735 fam_type = &family_types[F16_M30H_CPUS];
2736 pvt->ops = &family_types[F16_M30H_CPUS].ops;
2737 break;
2738 }
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002739 fam_type = &family_types[F16_CPUS];
2740 pvt->ops = &family_types[F16_CPUS].ops;
Aravind Gopalakrishnan94c1acf2013-04-17 14:57:13 -05002741 break;
2742
Borislav Petkov395ae782010-10-01 18:38:19 +02002743 default:
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002744 amd64_err("Unsupported family!\n");
Borislav Petkov0092b202010-10-01 19:20:05 +02002745 return NULL;
Borislav Petkov395ae782010-10-01 18:38:19 +02002746 }
Borislav Petkov0092b202010-10-01 19:20:05 +02002747
Borislav Petkovdf71a052011-01-19 18:15:10 +01002748 amd64_info("%s %sdetected (node %d).\n", fam_type->ctl_name,
Aravind Gopalakrishnan18b94f62013-08-09 11:54:49 -05002749 (pvt->fam == 0xf ?
Borislav Petkov24f9a7f2010-10-07 18:29:15 +02002750 (pvt->ext_model >= K8_REV_F ? "revF or later "
2751 : "revE or earlier ")
2752 : ""), pvt->mc_node_id);
Borislav Petkov0092b202010-10-01 19:20:05 +02002753 return fam_type;
Borislav Petkov395ae782010-10-01 18:38:19 +02002754}
2755
Takashi Iwaie339f1e2015-02-04 11:48:53 +01002756static const struct attribute_group *amd64_edac_attr_groups[] = {
2757#ifdef CONFIG_EDAC_DEBUG
2758 &amd64_edac_dbg_group,
2759#endif
2760#ifdef CONFIG_EDAC_AMD64_ERROR_INJECTION
2761 &amd64_edac_inj_group,
2762#endif
2763 NULL
2764};
2765
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002766static int init_one_instance(struct pci_dev *F2)
Doug Thompson7d6034d2009-04-27 20:01:01 +02002767{
2768 struct amd64_pvt *pvt = NULL;
Borislav Petkov0092b202010-10-01 19:20:05 +02002769 struct amd64_family_type *fam_type = NULL;
Borislav Petkov360b7f32010-10-15 19:25:38 +02002770 struct mem_ctl_info *mci = NULL;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03002771 struct edac_mc_layer layers[2];
Doug Thompson7d6034d2009-04-27 20:01:01 +02002772 int err = 0, ret;
Daniel J Blueman772c3ff2012-11-27 14:32:09 +08002773 u16 nid = amd_get_node_id(F2);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002774
2775 ret = -ENOMEM;
2776 pvt = kzalloc(sizeof(struct amd64_pvt), GFP_KERNEL);
2777 if (!pvt)
Borislav Petkov360b7f32010-10-15 19:25:38 +02002778 goto err_ret;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002779
Borislav Petkov360b7f32010-10-15 19:25:38 +02002780 pvt->mc_node_id = nid;
Borislav Petkov8d5b5d92010-10-01 20:11:07 +02002781 pvt->F2 = F2;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002782
Borislav Petkov395ae782010-10-01 18:38:19 +02002783 ret = -EINVAL;
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002784 fam_type = per_family_init(pvt);
Borislav Petkov0092b202010-10-01 19:20:05 +02002785 if (!fam_type)
Borislav Petkov395ae782010-10-01 18:38:19 +02002786 goto err_free;
2787
Doug Thompson7d6034d2009-04-27 20:01:01 +02002788 ret = -ENODEV;
Borislav Petkov360b7f32010-10-15 19:25:38 +02002789 err = reserve_mc_sibling_devs(pvt, fam_type->f1_id, fam_type->f3_id);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002790 if (err)
2791 goto err_free;
2792
Borislav Petkov360b7f32010-10-15 19:25:38 +02002793 read_mc_regs(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002794
Doug Thompson7d6034d2009-04-27 20:01:01 +02002795 /*
2796 * We need to determine how many memory channels there are. Then use
2797 * that information for calculating the size of the dynamic instance
Borislav Petkov360b7f32010-10-15 19:25:38 +02002798 * tables in the 'mci' structure.
Doug Thompson7d6034d2009-04-27 20:01:01 +02002799 */
Borislav Petkov360b7f32010-10-15 19:25:38 +02002800 ret = -EINVAL;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002801 pvt->channel_count = pvt->ops->early_channel_count(pvt);
2802 if (pvt->channel_count < 0)
Borislav Petkov360b7f32010-10-15 19:25:38 +02002803 goto err_siblings;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002804
2805 ret = -ENOMEM;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03002806 layers[0].type = EDAC_MC_LAYER_CHIP_SELECT;
2807 layers[0].size = pvt->csels[0].b_cnt;
2808 layers[0].is_virt_csrow = true;
2809 layers[1].type = EDAC_MC_LAYER_CHANNEL;
Borislav Petkovf0a56c42013-07-23 20:01:23 +02002810
2811 /*
2812 * Always allocate two channels since we can have setups with DIMMs on
2813 * only one channel. Also, this simplifies handling later for the price
2814 * of a couple of KBs tops.
2815 */
2816 layers[1].size = 2;
Mauro Carvalho Chehabab5a5032012-04-16 15:03:50 -03002817 layers[1].is_virt_csrow = false;
Borislav Petkovf0a56c42013-07-23 20:01:23 +02002818
Mauro Carvalho Chehabca0907b2012-05-02 14:37:00 -03002819 mci = edac_mc_alloc(nid, ARRAY_SIZE(layers), layers, 0);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002820 if (!mci)
Borislav Petkov360b7f32010-10-15 19:25:38 +02002821 goto err_siblings;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002822
2823 mci->pvt_info = pvt;
Mauro Carvalho Chehabfd687502012-03-16 07:44:18 -03002824 mci->pdev = &pvt->F2->dev;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002825
Borislav Petkovdf71a052011-01-19 18:15:10 +01002826 setup_mci_misc_attrs(mci, fam_type);
Borislav Petkov360b7f32010-10-15 19:25:38 +02002827
2828 if (init_csrows(mci))
Doug Thompson7d6034d2009-04-27 20:01:01 +02002829 mci->edac_cap = EDAC_FLAG_NONE;
2830
Doug Thompson7d6034d2009-04-27 20:01:01 +02002831 ret = -ENODEV;
Takashi Iwaie339f1e2015-02-04 11:48:53 +01002832 if (edac_mc_add_mc_with_groups(mci, amd64_edac_attr_groups)) {
Joe Perches956b9ba12012-04-29 17:08:39 -03002833 edac_dbg(1, "failed edac_mc_add_mc()\n");
Doug Thompson7d6034d2009-04-27 20:01:01 +02002834 goto err_add_mc;
2835 }
2836
Borislav Petkov549d0422009-07-24 13:51:42 +02002837 /* register stuff with EDAC MCE */
2838 if (report_gart_errors)
2839 amd_report_gart_errors(true);
2840
Borislav Petkovdf781d02013-12-15 17:29:44 +01002841 amd_register_ecc_decoder(decode_bus_error);
Borislav Petkov549d0422009-07-24 13:51:42 +02002842
Borislav Petkov360b7f32010-10-15 19:25:38 +02002843 atomic_inc(&drv_instances);
2844
Doug Thompson7d6034d2009-04-27 20:01:01 +02002845 return 0;
2846
2847err_add_mc:
2848 edac_mc_free(mci);
2849
Borislav Petkov360b7f32010-10-15 19:25:38 +02002850err_siblings:
2851 free_mc_sibling_devs(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002852
Borislav Petkov360b7f32010-10-15 19:25:38 +02002853err_free:
2854 kfree(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002855
Borislav Petkov360b7f32010-10-15 19:25:38 +02002856err_ret:
Doug Thompson7d6034d2009-04-27 20:01:01 +02002857 return ret;
2858}
2859
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002860static int probe_one_instance(struct pci_dev *pdev,
2861 const struct pci_device_id *mc_type)
Doug Thompson7d6034d2009-04-27 20:01:01 +02002862{
Daniel J Blueman772c3ff2012-11-27 14:32:09 +08002863 u16 nid = amd_get_node_id(pdev);
Borislav Petkov2299ef72010-10-15 17:44:04 +02002864 struct pci_dev *F3 = node_to_amd_nb(nid)->misc;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002865 struct ecc_settings *s;
Borislav Petkov2299ef72010-10-15 17:44:04 +02002866 int ret = 0;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002867
Doug Thompson7d6034d2009-04-27 20:01:01 +02002868 ret = pci_enable_device(pdev);
Borislav Petkovb8cfa022010-10-01 19:35:38 +02002869 if (ret < 0) {
Joe Perches956b9ba12012-04-29 17:08:39 -03002870 edac_dbg(0, "ret=%d\n", ret);
Borislav Petkovb8cfa022010-10-01 19:35:38 +02002871 return -EIO;
2872 }
2873
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002874 ret = -ENOMEM;
2875 s = kzalloc(sizeof(struct ecc_settings), GFP_KERNEL);
2876 if (!s)
Borislav Petkov2299ef72010-10-15 17:44:04 +02002877 goto err_out;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002878
2879 ecc_stngs[nid] = s;
2880
Borislav Petkov2299ef72010-10-15 17:44:04 +02002881 if (!ecc_enabled(F3, nid)) {
2882 ret = -ENODEV;
2883
2884 if (!ecc_enable_override)
2885 goto err_enable;
2886
2887 amd64_warn("Forcing ECC on!\n");
2888
2889 if (!enable_ecc_error_reporting(s, nid, F3))
2890 goto err_enable;
2891 }
2892
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002893 ret = init_one_instance(pdev);
Borislav Petkov360b7f32010-10-15 19:25:38 +02002894 if (ret < 0) {
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002895 amd64_err("Error probing instance: %d\n", nid);
Borislav Petkov360b7f32010-10-15 19:25:38 +02002896 restore_ecc_error_reporting(s, nid, F3);
2897 }
Doug Thompson7d6034d2009-04-27 20:01:01 +02002898
2899 return ret;
Borislav Petkov2299ef72010-10-15 17:44:04 +02002900
2901err_enable:
2902 kfree(s);
2903 ecc_stngs[nid] = NULL;
2904
2905err_out:
2906 return ret;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002907}
2908
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002909static void remove_one_instance(struct pci_dev *pdev)
Doug Thompson7d6034d2009-04-27 20:01:01 +02002910{
2911 struct mem_ctl_info *mci;
2912 struct amd64_pvt *pvt;
Daniel J Blueman772c3ff2012-11-27 14:32:09 +08002913 u16 nid = amd_get_node_id(pdev);
Borislav Petkov360b7f32010-10-15 19:25:38 +02002914 struct pci_dev *F3 = node_to_amd_nb(nid)->misc;
2915 struct ecc_settings *s = ecc_stngs[nid];
Doug Thompson7d6034d2009-04-27 20:01:01 +02002916
Mauro Carvalho Chehabc5608752012-03-21 14:00:44 -03002917 mci = find_mci_by_dev(&pdev->dev);
Borislav Petkova4b4bed2013-08-10 13:54:48 +02002918 WARN_ON(!mci);
2919
Doug Thompson7d6034d2009-04-27 20:01:01 +02002920 /* Remove from EDAC CORE tracking list */
2921 mci = edac_mc_del_mc(&pdev->dev);
2922 if (!mci)
2923 return;
2924
2925 pvt = mci->pvt_info;
2926
Borislav Petkov360b7f32010-10-15 19:25:38 +02002927 restore_ecc_error_reporting(s, nid, F3);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002928
Borislav Petkov360b7f32010-10-15 19:25:38 +02002929 free_mc_sibling_devs(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002930
Borislav Petkov549d0422009-07-24 13:51:42 +02002931 /* unregister from EDAC MCE */
2932 amd_report_gart_errors(false);
Borislav Petkovdf781d02013-12-15 17:29:44 +01002933 amd_unregister_ecc_decoder(decode_bus_error);
Borislav Petkov549d0422009-07-24 13:51:42 +02002934
Borislav Petkov360b7f32010-10-15 19:25:38 +02002935 kfree(ecc_stngs[nid]);
2936 ecc_stngs[nid] = NULL;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02002937
Doug Thompson7d6034d2009-04-27 20:01:01 +02002938 /* Free the EDAC CORE resources */
Borislav Petkov8f68ed92009-12-21 15:15:59 +01002939 mci->pvt_info = NULL;
Borislav Petkov8f68ed92009-12-21 15:15:59 +01002940
2941 kfree(pvt);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002942 edac_mc_free(mci);
2943}
2944
2945/*
2946 * This table is part of the interface for loading drivers for PCI devices. The
2947 * PCI core identifies what devices are on a system during boot, and then
2948 * inquiry this table to see if this driver is for a given device found.
2949 */
Jingoo Hanba935f42013-12-06 10:23:08 +01002950static const struct pci_device_id amd64_pci_table[] = {
Aravind Gopalakrishnana597d2a2014-10-30 12:16:09 +01002951 { PCI_VDEVICE(AMD, PCI_DEVICE_ID_AMD_K8_NB_MEMCTL) },
2952 { PCI_VDEVICE(AMD, PCI_DEVICE_ID_AMD_10H_NB_DRAM) },
2953 { PCI_VDEVICE(AMD, PCI_DEVICE_ID_AMD_15H_NB_F2) },
2954 { PCI_VDEVICE(AMD, PCI_DEVICE_ID_AMD_15H_M30H_NB_F2) },
2955 { PCI_VDEVICE(AMD, PCI_DEVICE_ID_AMD_15H_M60H_NB_F2) },
2956 { PCI_VDEVICE(AMD, PCI_DEVICE_ID_AMD_16H_NB_F2) },
2957 { PCI_VDEVICE(AMD, PCI_DEVICE_ID_AMD_16H_M30H_NB_F2) },
Doug Thompson7d6034d2009-04-27 20:01:01 +02002958 {0, }
2959};
2960MODULE_DEVICE_TABLE(pci, amd64_pci_table);
2961
2962static struct pci_driver amd64_pci_driver = {
2963 .name = EDAC_MOD_STR,
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002964 .probe = probe_one_instance,
2965 .remove = remove_one_instance,
Doug Thompson7d6034d2009-04-27 20:01:01 +02002966 .id_table = amd64_pci_table,
Luis R. Rodriguez735c0f82015-03-30 16:20:08 -07002967 .driver.probe_type = PROBE_FORCE_SYNCHRONOUS,
Doug Thompson7d6034d2009-04-27 20:01:01 +02002968};
2969
Borislav Petkov360b7f32010-10-15 19:25:38 +02002970static void setup_pci_device(void)
Doug Thompson7d6034d2009-04-27 20:01:01 +02002971{
2972 struct mem_ctl_info *mci;
2973 struct amd64_pvt *pvt;
2974
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002975 if (pci_ctl)
Doug Thompson7d6034d2009-04-27 20:01:01 +02002976 return;
2977
Borislav Petkov2ec591a2015-02-17 10:58:34 +01002978 mci = edac_mc_find(0);
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002979 if (!mci)
2980 return;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002981
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01002982 pvt = mci->pvt_info;
2983 pci_ctl = edac_pci_create_generic_ctl(&pvt->F2->dev, EDAC_MOD_STR);
2984 if (!pci_ctl) {
2985 pr_warn("%s(): Unable to create PCI control\n", __func__);
2986 pr_warn("%s(): PCI error report via EDAC not set\n", __func__);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002987 }
2988}
2989
2990static int __init amd64_edac_init(void)
2991{
Borislav Petkov360b7f32010-10-15 19:25:38 +02002992 int err = -ENODEV;
Doug Thompson7d6034d2009-04-27 20:01:01 +02002993
Borislav Petkovdf71a052011-01-19 18:15:10 +01002994 printk(KERN_INFO "AMD64 EDAC driver v%s\n", EDAC_AMD64_VERSION);
Doug Thompson7d6034d2009-04-27 20:01:01 +02002995
2996 opstate_init();
2997
Hans Rosenfeld9653a5c2010-10-29 17:14:31 +02002998 if (amd_cache_northbridges() < 0)
Borislav Petkov56b34b92009-12-21 18:13:01 +01002999 goto err_ret;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003000
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003001 err = -ENOMEM;
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003002 ecc_stngs = kzalloc(amd_nb_num() * sizeof(ecc_stngs[0]), GFP_KERNEL);
Borislav Petkov2ec591a2015-02-17 10:58:34 +01003003 if (!ecc_stngs)
Borislav Petkova9f0fbe2011-03-29 18:10:53 +02003004 goto err_free;
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003005
Borislav Petkov50542252009-12-11 18:14:40 +01003006 msrs = msrs_alloc();
Borislav Petkov56b34b92009-12-21 18:13:01 +01003007 if (!msrs)
Borislav Petkov360b7f32010-10-15 19:25:38 +02003008 goto err_free;
Borislav Petkov50542252009-12-11 18:14:40 +01003009
Doug Thompson7d6034d2009-04-27 20:01:01 +02003010 err = pci_register_driver(&amd64_pci_driver);
3011 if (err)
Borislav Petkov56b34b92009-12-21 18:13:01 +01003012 goto err_pci;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003013
Borislav Petkov56b34b92009-12-21 18:13:01 +01003014 err = -ENODEV;
Borislav Petkov360b7f32010-10-15 19:25:38 +02003015 if (!atomic_read(&drv_instances))
3016 goto err_no_instances;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003017
Borislav Petkov360b7f32010-10-15 19:25:38 +02003018 setup_pci_device();
Tomasz Palaf5b10c42014-11-02 11:22:12 +01003019
3020#ifdef CONFIG_X86_32
3021 amd64_err("%s on 32-bit is unsupported. USE AT YOUR OWN RISK!\n", EDAC_MOD_STR);
3022#endif
3023
Borislav Petkov360b7f32010-10-15 19:25:38 +02003024 return 0;
Borislav Petkov56b34b92009-12-21 18:13:01 +01003025
Borislav Petkov360b7f32010-10-15 19:25:38 +02003026err_no_instances:
Doug Thompson7d6034d2009-04-27 20:01:01 +02003027 pci_unregister_driver(&amd64_pci_driver);
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003028
Borislav Petkov56b34b92009-12-21 18:13:01 +01003029err_pci:
3030 msrs_free(msrs);
3031 msrs = NULL;
Borislav Petkovcc4d8862010-10-13 16:11:59 +02003032
Borislav Petkov360b7f32010-10-15 19:25:38 +02003033err_free:
Borislav Petkov360b7f32010-10-15 19:25:38 +02003034 kfree(ecc_stngs);
3035 ecc_stngs = NULL;
3036
Borislav Petkov56b34b92009-12-21 18:13:01 +01003037err_ret:
Doug Thompson7d6034d2009-04-27 20:01:01 +02003038 return err;
3039}
3040
3041static void __exit amd64_edac_exit(void)
3042{
Borislav Petkovd1ea71c2013-12-15 17:54:27 +01003043 if (pci_ctl)
3044 edac_pci_release_generic_ctl(pci_ctl);
Doug Thompson7d6034d2009-04-27 20:01:01 +02003045
3046 pci_unregister_driver(&amd64_pci_driver);
Borislav Petkov50542252009-12-11 18:14:40 +01003047
Borislav Petkovae7bb7c2010-10-14 16:01:30 +02003048 kfree(ecc_stngs);
3049 ecc_stngs = NULL;
3050
Borislav Petkov50542252009-12-11 18:14:40 +01003051 msrs_free(msrs);
3052 msrs = NULL;
Doug Thompson7d6034d2009-04-27 20:01:01 +02003053}
3054
3055module_init(amd64_edac_init);
3056module_exit(amd64_edac_exit);
3057
3058MODULE_LICENSE("GPL");
3059MODULE_AUTHOR("SoftwareBitMaker: Doug Thompson, "
3060 "Dave Peterson, Thayne Harbaugh");
3061MODULE_DESCRIPTION("MC support for AMD64 memory controllers - "
3062 EDAC_AMD64_VERSION);
3063
3064module_param(edac_op_state, int, 0444);
3065MODULE_PARM_DESC(edac_op_state, "EDAC Error Reporting state: 0=Poll,1=NMI");