blob: 04f1e7ce02b196868d42253243d79d1044d9b7cc [file] [log] [blame]
Mauro Carvalho Chehab52707f92010-05-18 20:43:52 -03001/* Intel i7 core/Nehalem Memory Controller kernel module
2 *
David Sterbae7bf0682010-12-27 16:51:15 +01003 * This driver supports the memory controllers found on the Intel
Mauro Carvalho Chehab52707f92010-05-18 20:43:52 -03004 * processor families i7core, i7core 7xx/8xx, i5core, Xeon 35xx,
5 * Xeon 55xx and Xeon 56xx also known as Nehalem, Nehalem-EP, Lynnfield
6 * and Westmere-EP.
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03007 *
8 * This file may be distributed under the terms of the
9 * GNU General Public License version 2 only.
10 *
Mauro Carvalho Chehab52707f92010-05-18 20:43:52 -030011 * Copyright (c) 2009-2010 by:
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -030012 * Mauro Carvalho Chehab <mchehab@redhat.com>
13 *
14 * Red Hat Inc. http://www.redhat.com
15 *
16 * Forked and adapted from the i5400_edac driver
17 *
18 * Based on the following public Intel datasheets:
19 * Intel Core i7 Processor Extreme Edition and Intel Core i7 Processor
20 * Datasheet, Volume 2:
21 * http://download.intel.com/design/processor/datashts/320835.pdf
22 * Intel Xeon Processor 5500 Series Datasheet Volume 2
23 * http://www.intel.com/Assets/PDF/datasheet/321322.pdf
24 * also available at:
25 * http://www.arrownac.com/manufacturers/intel/s/nehalem/5500-datasheet-v2.pdf
26 */
27
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -030028#include <linux/module.h>
29#include <linux/init.h>
30#include <linux/pci.h>
31#include <linux/pci_ids.h>
32#include <linux/slab.h>
Randy Dunlap3b918c12009-11-08 01:36:40 -020033#include <linux/delay.h>
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -030034#include <linux/edac.h>
35#include <linux/mmzone.h>
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -030036#include <linux/edac_mce.h>
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -030037#include <linux/smp.h>
Mauro Carvalho Chehab14d2c082009-09-02 23:52:36 -030038#include <asm/processor.h>
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -030039
40#include "edac_core.h"
41
Mauro Carvalho Chehab18c29002010-08-10 18:33:27 -030042/* Static vars */
43static LIST_HEAD(i7core_edac_list);
44static DEFINE_MUTEX(i7core_edac_lock);
45static int probed;
46
Mauro Carvalho Chehab54a08ab2010-08-19 15:51:00 -030047static int use_pci_fixup;
48module_param(use_pci_fixup, int, 0444);
49MODULE_PARM_DESC(use_pci_fixup, "Enable PCI fixup to seek for hidden devices");
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -030050/*
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -030051 * This is used for Nehalem-EP and Nehalem-EX devices, where the non-core
52 * registers start at bus 255, and are not reported by BIOS.
53 * We currently find devices with only 2 sockets. In order to support more QPI
54 * Quick Path Interconnect, just increment this number.
55 */
56#define MAX_SOCKET_BUSES 2
57
58
59/*
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -030060 * Alter this version for the module when modifications are made
61 */
Michal Marek152ba392011-04-01 12:41:20 +020062#define I7CORE_REVISION " Ver: 1.0.0"
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -030063#define EDAC_MOD_STR "i7core_edac"
64
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -030065/*
66 * Debug macros
67 */
68#define i7core_printk(level, fmt, arg...) \
69 edac_printk(level, "i7core", fmt, ##arg)
70
71#define i7core_mc_printk(mci, level, fmt, arg...) \
72 edac_mc_chipset_printk(mci, level, "i7core", fmt, ##arg)
73
74/*
75 * i7core Memory Controller Registers
76 */
77
Mauro Carvalho Chehabe9bd2e72009-07-09 22:14:35 -030078 /* OFFSETS for Device 0 Function 0 */
79
80#define MC_CFG_CONTROL 0x90
81
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -030082 /* OFFSETS for Device 3 Function 0 */
83
84#define MC_CONTROL 0x48
85#define MC_STATUS 0x4c
86#define MC_MAX_DOD 0x64
87
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -030088/*
89 * OFFSETS for Device 3 Function 4, as inicated on Xeon 5500 datasheet:
90 * http://www.arrownac.com/manufacturers/intel/s/nehalem/5500-datasheet-v2.pdf
91 */
92
93#define MC_TEST_ERR_RCV1 0x60
94 #define DIMM2_COR_ERR(r) ((r) & 0x7fff)
95
96#define MC_TEST_ERR_RCV0 0x64
97 #define DIMM1_COR_ERR(r) (((r) >> 16) & 0x7fff)
98 #define DIMM0_COR_ERR(r) ((r) & 0x7fff)
99
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -0300100/* OFFSETS for Device 3 Function 2, as inicated on Xeon 5500 datasheet */
101#define MC_COR_ECC_CNT_0 0x80
102#define MC_COR_ECC_CNT_1 0x84
103#define MC_COR_ECC_CNT_2 0x88
104#define MC_COR_ECC_CNT_3 0x8c
105#define MC_COR_ECC_CNT_4 0x90
106#define MC_COR_ECC_CNT_5 0x94
107
108#define DIMM_TOP_COR_ERR(r) (((r) >> 16) & 0x7fff)
109#define DIMM_BOT_COR_ERR(r) ((r) & 0x7fff)
110
111
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300112 /* OFFSETS for Devices 4,5 and 6 Function 0 */
113
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300114#define MC_CHANNEL_DIMM_INIT_PARAMS 0x58
115 #define THREE_DIMMS_PRESENT (1 << 24)
116 #define SINGLE_QUAD_RANK_PRESENT (1 << 23)
117 #define QUAD_RANK_PRESENT (1 << 22)
118 #define REGISTERED_DIMM (1 << 15)
119
Mauro Carvalho Chehabf122a892009-06-22 22:48:29 -0300120#define MC_CHANNEL_MAPPER 0x60
121 #define RDLCH(r, ch) ((((r) >> (3 + (ch * 6))) & 0x07) - 1)
122 #define WRLCH(r, ch) ((((r) >> (ch * 6)) & 0x07) - 1)
123
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300124#define MC_CHANNEL_RANK_PRESENT 0x7c
125 #define RANK_PRESENT_MASK 0xffff
126
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300127#define MC_CHANNEL_ADDR_MATCH 0xf0
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300128#define MC_CHANNEL_ERROR_MASK 0xf8
129#define MC_CHANNEL_ERROR_INJECT 0xfc
130 #define INJECT_ADDR_PARITY 0x10
131 #define INJECT_ECC 0x08
132 #define MASK_CACHELINE 0x06
133 #define MASK_FULL_CACHELINE 0x06
134 #define MASK_MSB32_CACHELINE 0x04
135 #define MASK_LSB32_CACHELINE 0x02
136 #define NO_MASK_CACHELINE 0x00
137 #define REPEAT_EN 0x01
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300138
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300139 /* OFFSETS for Devices 4,5 and 6 Function 1 */
Mauro Carvalho Chehabb9905382009-08-05 21:36:35 -0300140
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300141#define MC_DOD_CH_DIMM0 0x48
142#define MC_DOD_CH_DIMM1 0x4c
143#define MC_DOD_CH_DIMM2 0x50
144 #define RANKOFFSET_MASK ((1 << 12) | (1 << 11) | (1 << 10))
145 #define RANKOFFSET(x) ((x & RANKOFFSET_MASK) >> 10)
146 #define DIMM_PRESENT_MASK (1 << 9)
147 #define DIMM_PRESENT(x) (((x) & DIMM_PRESENT_MASK) >> 9)
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300148 #define MC_DOD_NUMBANK_MASK ((1 << 8) | (1 << 7))
149 #define MC_DOD_NUMBANK(x) (((x) & MC_DOD_NUMBANK_MASK) >> 7)
150 #define MC_DOD_NUMRANK_MASK ((1 << 6) | (1 << 5))
151 #define MC_DOD_NUMRANK(x) (((x) & MC_DOD_NUMRANK_MASK) >> 5)
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -0300152 #define MC_DOD_NUMROW_MASK ((1 << 4) | (1 << 3) | (1 << 2))
Mauro Carvalho Chehab5566cb72009-06-22 22:48:31 -0300153 #define MC_DOD_NUMROW(x) (((x) & MC_DOD_NUMROW_MASK) >> 2)
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300154 #define MC_DOD_NUMCOL_MASK 3
155 #define MC_DOD_NUMCOL(x) ((x) & MC_DOD_NUMCOL_MASK)
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300156
Mauro Carvalho Chehabf122a892009-06-22 22:48:29 -0300157#define MC_RANK_PRESENT 0x7c
158
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300159#define MC_SAG_CH_0 0x80
160#define MC_SAG_CH_1 0x84
161#define MC_SAG_CH_2 0x88
162#define MC_SAG_CH_3 0x8c
163#define MC_SAG_CH_4 0x90
164#define MC_SAG_CH_5 0x94
165#define MC_SAG_CH_6 0x98
166#define MC_SAG_CH_7 0x9c
167
168#define MC_RIR_LIMIT_CH_0 0x40
169#define MC_RIR_LIMIT_CH_1 0x44
170#define MC_RIR_LIMIT_CH_2 0x48
171#define MC_RIR_LIMIT_CH_3 0x4C
172#define MC_RIR_LIMIT_CH_4 0x50
173#define MC_RIR_LIMIT_CH_5 0x54
174#define MC_RIR_LIMIT_CH_6 0x58
175#define MC_RIR_LIMIT_CH_7 0x5C
176#define MC_RIR_LIMIT_MASK ((1 << 10) - 1)
177
178#define MC_RIR_WAY_CH 0x80
179 #define MC_RIR_WAY_OFFSET_MASK (((1 << 14) - 1) & ~0x7)
180 #define MC_RIR_WAY_RANK_MASK 0x7
181
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300182/*
183 * i7core structs
184 */
185
186#define NUM_CHANS 3
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -0300187#define MAX_DIMMS 3 /* Max DIMMS per channel */
188#define MAX_MCR_FUNC 4
189#define MAX_CHAN_FUNC 3
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300190
191struct i7core_info {
192 u32 mc_control;
193 u32 mc_status;
194 u32 max_dod;
Mauro Carvalho Chehabf122a892009-06-22 22:48:29 -0300195 u32 ch_map;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300196};
197
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300198
199struct i7core_inject {
200 int enable;
201
202 u32 section;
203 u32 type;
204 u32 eccmask;
205
206 /* Error address mask */
207 int channel, dimm, rank, bank, page, col;
208};
209
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300210struct i7core_channel {
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -0300211 u32 ranks;
212 u32 dimms;
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300213};
214
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300215struct pci_id_descr {
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -0300216 int dev;
217 int func;
218 int dev_id;
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -0300219 int optional;
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300220};
221
Vernon Mauerybd9e19c2010-05-18 19:02:50 -0300222struct pci_id_table {
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -0300223 const struct pci_id_descr *descr;
224 int n_devs;
Vernon Mauerybd9e19c2010-05-18 19:02:50 -0300225};
226
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300227struct i7core_dev {
228 struct list_head list;
229 u8 socket;
230 struct pci_dev **pdev;
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -0300231 int n_devs;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300232 struct mem_ctl_info *mci;
233};
234
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300235struct i7core_pvt {
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300236 struct pci_dev *pci_noncore;
237 struct pci_dev *pci_mcr[MAX_MCR_FUNC + 1];
238 struct pci_dev *pci_ch[NUM_CHANS][MAX_CHAN_FUNC + 1];
239
240 struct i7core_dev *i7core_dev;
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300241
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300242 struct i7core_info info;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300243 struct i7core_inject inject;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300244 struct i7core_channel channel[NUM_CHANS];
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300245
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300246 int ce_count_available;
247 int csrow_map[NUM_CHANS][MAX_DIMMS];
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -0300248
249 /* ECC corrected errors counts per udimm */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300250 unsigned long udimm_ce_count[MAX_DIMMS];
251 int udimm_last_ce_count[MAX_DIMMS];
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -0300252 /* ECC corrected errors counts per rdimm */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300253 unsigned long rdimm_ce_count[NUM_CHANS][MAX_DIMMS];
254 int rdimm_last_ce_count[NUM_CHANS][MAX_DIMMS];
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -0300255
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300256 unsigned int is_registered;
Mauro Carvalho Chehab14d2c082009-09-02 23:52:36 -0300257
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -0300258 /* mcelog glue */
259 struct edac_mce edac_mce;
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -0300260
261 /* Fifo double buffers */
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -0300262 struct mce mce_entry[MCE_LOG_LEN];
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -0300263 struct mce mce_outentry[MCE_LOG_LEN];
264
265 /* Fifo in/out counters */
266 unsigned mce_in, mce_out;
267
268 /* Count indicator to show errors not got */
269 unsigned mce_overrun;
Mauro Carvalho Chehab939747bd2010-08-10 11:22:01 -0300270
271 /* Struct to control EDAC polling */
272 struct edac_pci_ctl_info *i7core_pci;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300273};
274
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300275#define PCI_DESCR(device, function, device_id) \
276 .dev = (device), \
277 .func = (function), \
278 .dev_id = (device_id)
279
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -0300280static const struct pci_id_descr pci_dev_descr_i7core_nehalem[] = {
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300281 /* Memory controller */
282 { PCI_DESCR(3, 0, PCI_DEVICE_ID_INTEL_I7_MCR) },
283 { PCI_DESCR(3, 1, PCI_DEVICE_ID_INTEL_I7_MC_TAD) },
Mauro Carvalho Chehab79daef22010-08-21 01:03:52 -0300284
285 /* Exists only for RDIMM */
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -0300286 { PCI_DESCR(3, 2, PCI_DEVICE_ID_INTEL_I7_MC_RAS), .optional = 1 },
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300287 { PCI_DESCR(3, 4, PCI_DEVICE_ID_INTEL_I7_MC_TEST) },
288
289 /* Channel 0 */
290 { PCI_DESCR(4, 0, PCI_DEVICE_ID_INTEL_I7_MC_CH0_CTRL) },
291 { PCI_DESCR(4, 1, PCI_DEVICE_ID_INTEL_I7_MC_CH0_ADDR) },
292 { PCI_DESCR(4, 2, PCI_DEVICE_ID_INTEL_I7_MC_CH0_RANK) },
293 { PCI_DESCR(4, 3, PCI_DEVICE_ID_INTEL_I7_MC_CH0_TC) },
294
295 /* Channel 1 */
296 { PCI_DESCR(5, 0, PCI_DEVICE_ID_INTEL_I7_MC_CH1_CTRL) },
297 { PCI_DESCR(5, 1, PCI_DEVICE_ID_INTEL_I7_MC_CH1_ADDR) },
298 { PCI_DESCR(5, 2, PCI_DEVICE_ID_INTEL_I7_MC_CH1_RANK) },
299 { PCI_DESCR(5, 3, PCI_DEVICE_ID_INTEL_I7_MC_CH1_TC) },
300
301 /* Channel 2 */
302 { PCI_DESCR(6, 0, PCI_DEVICE_ID_INTEL_I7_MC_CH2_CTRL) },
303 { PCI_DESCR(6, 1, PCI_DEVICE_ID_INTEL_I7_MC_CH2_ADDR) },
304 { PCI_DESCR(6, 2, PCI_DEVICE_ID_INTEL_I7_MC_CH2_RANK) },
305 { PCI_DESCR(6, 3, PCI_DEVICE_ID_INTEL_I7_MC_CH2_TC) },
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300306};
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300307
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -0300308static const struct pci_id_descr pci_dev_descr_lynnfield[] = {
Mauro Carvalho Chehab52a2e4fc2009-10-14 11:21:58 -0300309 { PCI_DESCR( 3, 0, PCI_DEVICE_ID_INTEL_LYNNFIELD_MCR) },
310 { PCI_DESCR( 3, 1, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_TAD) },
311 { PCI_DESCR( 3, 4, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_TEST) },
312
313 { PCI_DESCR( 4, 0, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH0_CTRL) },
314 { PCI_DESCR( 4, 1, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH0_ADDR) },
315 { PCI_DESCR( 4, 2, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH0_RANK) },
316 { PCI_DESCR( 4, 3, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH0_TC) },
317
Mauro Carvalho Chehab508fa172009-10-14 13:44:37 -0300318 { PCI_DESCR( 5, 0, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH1_CTRL) },
319 { PCI_DESCR( 5, 1, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH1_ADDR) },
320 { PCI_DESCR( 5, 2, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH1_RANK) },
321 { PCI_DESCR( 5, 3, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH1_TC) },
Mauro Carvalho Chehab52a2e4fc2009-10-14 11:21:58 -0300322};
323
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -0300324static const struct pci_id_descr pci_dev_descr_i7core_westmere[] = {
Vernon Mauerybd9e19c2010-05-18 19:02:50 -0300325 /* Memory controller */
326 { PCI_DESCR(3, 0, PCI_DEVICE_ID_INTEL_LYNNFIELD_MCR_REV2) },
327 { PCI_DESCR(3, 1, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_TAD_REV2) },
328 /* Exists only for RDIMM */
329 { PCI_DESCR(3, 2, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_RAS_REV2), .optional = 1 },
330 { PCI_DESCR(3, 4, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_TEST_REV2) },
331
332 /* Channel 0 */
333 { PCI_DESCR(4, 0, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH0_CTRL_REV2) },
334 { PCI_DESCR(4, 1, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH0_ADDR_REV2) },
335 { PCI_DESCR(4, 2, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH0_RANK_REV2) },
336 { PCI_DESCR(4, 3, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH0_TC_REV2) },
337
338 /* Channel 1 */
339 { PCI_DESCR(5, 0, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH1_CTRL_REV2) },
340 { PCI_DESCR(5, 1, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH1_ADDR_REV2) },
341 { PCI_DESCR(5, 2, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH1_RANK_REV2) },
342 { PCI_DESCR(5, 3, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH1_TC_REV2) },
343
344 /* Channel 2 */
345 { PCI_DESCR(6, 0, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH2_CTRL_REV2) },
346 { PCI_DESCR(6, 1, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH2_ADDR_REV2) },
347 { PCI_DESCR(6, 2, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH2_RANK_REV2) },
348 { PCI_DESCR(6, 3, PCI_DEVICE_ID_INTEL_LYNNFIELD_MC_CH2_TC_REV2) },
Vernon Mauerybd9e19c2010-05-18 19:02:50 -0300349};
350
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -0300351#define PCI_ID_TABLE_ENTRY(A) { .descr=A, .n_devs = ARRAY_SIZE(A) }
352static const struct pci_id_table pci_dev_table[] = {
Vernon Mauerybd9e19c2010-05-18 19:02:50 -0300353 PCI_ID_TABLE_ENTRY(pci_dev_descr_i7core_nehalem),
354 PCI_ID_TABLE_ENTRY(pci_dev_descr_lynnfield),
355 PCI_ID_TABLE_ENTRY(pci_dev_descr_i7core_westmere),
Mauro Carvalho Chehab3c52cc52010-10-24 11:12:28 -0200356 {0,} /* 0 terminated list. */
Vernon Mauerybd9e19c2010-05-18 19:02:50 -0300357};
358
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300359/*
360 * pci_device_id table for which devices we are looking for
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300361 */
362static const struct pci_device_id i7core_pci_tbl[] __devinitdata = {
Mauro Carvalho Chehabd1fd4fb2009-07-10 18:39:53 -0300363 {PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_X58_HUB_MGMT)},
Mauro Carvalho Chehabf05da2f2009-10-14 13:31:06 -0300364 {PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_LYNNFIELD_QPI_LINK0)},
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300365 {0,} /* 0 terminated list. */
366};
367
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300368/****************************************************************************
369 Anciliary status routines
370 ****************************************************************************/
371
372 /* MC_CONTROL bits */
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300373#define CH_ACTIVE(pvt, ch) ((pvt)->info.mc_control & (1 << (8 + ch)))
374#define ECCx8(pvt) ((pvt)->info.mc_control & (1 << 1))
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300375
376 /* MC_STATUS bits */
Keith Mannthey61053fd2009-09-02 23:46:59 -0300377#define ECC_ENABLED(pvt) ((pvt)->info.mc_status & (1 << 4))
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300378#define CH_DISABLED(pvt, ch) ((pvt)->info.mc_status & (1 << ch))
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300379
380 /* MC_MAX_DOD read functions */
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300381static inline int numdimms(u32 dimms)
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300382{
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300383 return (dimms & 0x3) + 1;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300384}
385
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300386static inline int numrank(u32 rank)
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300387{
388 static int ranks[4] = { 1, 2, 4, -EINVAL };
389
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300390 return ranks[rank & 0x3];
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300391}
392
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300393static inline int numbank(u32 bank)
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300394{
395 static int banks[4] = { 4, 8, 16, -EINVAL };
396
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300397 return banks[bank & 0x3];
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300398}
399
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300400static inline int numrow(u32 row)
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300401{
402 static int rows[8] = {
403 1 << 12, 1 << 13, 1 << 14, 1 << 15,
404 1 << 16, -EINVAL, -EINVAL, -EINVAL,
405 };
406
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300407 return rows[row & 0x7];
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300408}
409
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300410static inline int numcol(u32 col)
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300411{
412 static int cols[8] = {
413 1 << 10, 1 << 11, 1 << 12, -EINVAL,
414 };
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300415 return cols[col & 0x3];
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300416}
417
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300418static struct i7core_dev *get_i7core_dev(u8 socket)
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -0300419{
420 struct i7core_dev *i7core_dev;
421
422 list_for_each_entry(i7core_dev, &i7core_edac_list, list) {
423 if (i7core_dev->socket == socket)
424 return i7core_dev;
425 }
426
427 return NULL;
428}
429
Hidetoshi Seto848b2f72010-08-20 04:24:44 -0300430static struct i7core_dev *alloc_i7core_dev(u8 socket,
431 const struct pci_id_table *table)
432{
433 struct i7core_dev *i7core_dev;
434
435 i7core_dev = kzalloc(sizeof(*i7core_dev), GFP_KERNEL);
436 if (!i7core_dev)
437 return NULL;
438
439 i7core_dev->pdev = kzalloc(sizeof(*i7core_dev->pdev) * table->n_devs,
440 GFP_KERNEL);
441 if (!i7core_dev->pdev) {
442 kfree(i7core_dev);
443 return NULL;
444 }
445
446 i7core_dev->socket = socket;
447 i7core_dev->n_devs = table->n_devs;
448 list_add_tail(&i7core_dev->list, &i7core_edac_list);
449
450 return i7core_dev;
451}
452
Hidetoshi Seto2aa9be42010-08-20 04:25:00 -0300453static void free_i7core_dev(struct i7core_dev *i7core_dev)
454{
455 list_del(&i7core_dev->list);
456 kfree(i7core_dev->pdev);
457 kfree(i7core_dev);
458}
459
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300460/****************************************************************************
461 Memory check routines
462 ****************************************************************************/
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300463static struct pci_dev *get_pdev_slot_func(u8 socket, unsigned slot,
464 unsigned func)
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300465{
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -0300466 struct i7core_dev *i7core_dev = get_i7core_dev(socket);
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300467 int i;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300468
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -0300469 if (!i7core_dev)
470 return NULL;
471
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -0300472 for (i = 0; i < i7core_dev->n_devs; i++) {
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -0300473 if (!i7core_dev->pdev[i])
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300474 continue;
475
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -0300476 if (PCI_SLOT(i7core_dev->pdev[i]->devfn) == slot &&
477 PCI_FUNC(i7core_dev->pdev[i]->devfn) == func) {
478 return i7core_dev->pdev[i];
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300479 }
480 }
481
Mauro Carvalho Chehabeb94fc42009-06-22 22:48:31 -0300482 return NULL;
483}
484
Mauro Carvalho Chehabec6df242009-07-18 10:44:30 -0300485/**
486 * i7core_get_active_channels() - gets the number of channels and csrows
487 * @socket: Quick Path Interconnect socket
488 * @channels: Number of channels that will be returned
489 * @csrows: Number of csrows found
490 *
491 * Since EDAC core needs to know in advance the number of available channels
492 * and csrows, in order to allocate memory for csrows/channels, it is needed
493 * to run two similar steps. At the first step, implemented on this function,
494 * it checks the number of csrows/channels present at one socket.
495 * this is used in order to properly allocate the size of mci components.
496 *
497 * It should be noticed that none of the current available datasheets explain
498 * or even mention how csrows are seen by the memory controller. So, we need
499 * to add a fake description for csrows.
500 * So, this driver is attributing one DIMM memory for one csrow.
501 */
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -0300502static int i7core_get_active_channels(const u8 socket, unsigned *channels,
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300503 unsigned *csrows)
Mauro Carvalho Chehabeb94fc42009-06-22 22:48:31 -0300504{
505 struct pci_dev *pdev = NULL;
506 int i, j;
507 u32 status, control;
508
509 *channels = 0;
510 *csrows = 0;
511
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300512 pdev = get_pdev_slot_func(socket, 3, 0);
Mauro Carvalho Chehabb7c76152009-06-22 22:48:30 -0300513 if (!pdev) {
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300514 i7core_printk(KERN_ERR, "Couldn't find socket %d fn 3.0!!!\n",
515 socket);
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300516 return -ENODEV;
Mauro Carvalho Chehabb7c76152009-06-22 22:48:30 -0300517 }
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300518
519 /* Device 3 function 0 reads */
520 pci_read_config_dword(pdev, MC_STATUS, &status);
521 pci_read_config_dword(pdev, MC_CONTROL, &control);
522
523 for (i = 0; i < NUM_CHANS; i++) {
Mauro Carvalho Chehabeb94fc42009-06-22 22:48:31 -0300524 u32 dimm_dod[3];
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300525 /* Check if the channel is active */
526 if (!(control & (1 << (8 + i))))
527 continue;
528
529 /* Check if the channel is disabled */
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -0300530 if (status & (1 << i))
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300531 continue;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300532
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300533 pdev = get_pdev_slot_func(socket, i + 4, 1);
Mauro Carvalho Chehabeb94fc42009-06-22 22:48:31 -0300534 if (!pdev) {
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300535 i7core_printk(KERN_ERR, "Couldn't find socket %d "
536 "fn %d.%d!!!\n",
537 socket, i + 4, 1);
Mauro Carvalho Chehabeb94fc42009-06-22 22:48:31 -0300538 return -ENODEV;
539 }
540 /* Devices 4-6 function 1 */
541 pci_read_config_dword(pdev,
542 MC_DOD_CH_DIMM0, &dimm_dod[0]);
543 pci_read_config_dword(pdev,
544 MC_DOD_CH_DIMM1, &dimm_dod[1]);
545 pci_read_config_dword(pdev,
546 MC_DOD_CH_DIMM2, &dimm_dod[2]);
547
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300548 (*channels)++;
Mauro Carvalho Chehabeb94fc42009-06-22 22:48:31 -0300549
550 for (j = 0; j < 3; j++) {
551 if (!DIMM_PRESENT(dimm_dod[j]))
552 continue;
553 (*csrows)++;
554 }
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300555 }
556
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -0300557 debugf0("Number of active channels on socket %d: %d\n",
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300558 socket, *channels);
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300559
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -0300560 return 0;
561}
562
Hidetoshi Seto2e5185f2010-08-20 04:32:56 -0300563static int get_dimm_config(const struct mem_ctl_info *mci)
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300564{
565 struct i7core_pvt *pvt = mci->pvt_info;
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300566 struct csrow_info *csr;
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300567 struct pci_dev *pdev;
Mauro Carvalho Chehabba6c5c62009-07-15 09:02:32 -0300568 int i, j;
Hidetoshi Seto2e5185f2010-08-20 04:32:56 -0300569 int csrow = 0;
Mauro Carvalho Chehab5566cb72009-06-22 22:48:31 -0300570 unsigned long last_page = 0;
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300571 enum edac_type mode;
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300572 enum mem_type mtype;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300573
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300574 /* Get data from the MC register, function 0 */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300575 pdev = pvt->pci_mcr[0];
Mauro Carvalho Chehab7dd69532009-06-22 22:48:30 -0300576 if (!pdev)
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300577 return -ENODEV;
578
Mauro Carvalho Chehabf122a892009-06-22 22:48:29 -0300579 /* Device 3 function 0 reads */
Mauro Carvalho Chehab7dd69532009-06-22 22:48:30 -0300580 pci_read_config_dword(pdev, MC_CONTROL, &pvt->info.mc_control);
581 pci_read_config_dword(pdev, MC_STATUS, &pvt->info.mc_status);
582 pci_read_config_dword(pdev, MC_MAX_DOD, &pvt->info.max_dod);
583 pci_read_config_dword(pdev, MC_CHANNEL_MAPPER, &pvt->info.ch_map);
Mauro Carvalho Chehabf122a892009-06-22 22:48:29 -0300584
Mauro Carvalho Chehab17cb7b02009-07-20 18:48:18 -0300585 debugf0("QPI %d control=0x%08x status=0x%08x dod=0x%08x map=0x%08x\n",
Mauro Carvalho Chehab4af91882009-09-24 09:58:26 -0300586 pvt->i7core_dev->socket, pvt->info.mc_control, pvt->info.mc_status,
Mauro Carvalho Chehabf122a892009-06-22 22:48:29 -0300587 pvt->info.max_dod, pvt->info.ch_map);
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300588
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300589 if (ECC_ENABLED(pvt)) {
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -0300590 debugf0("ECC enabled with x%d SDCC\n", ECCx8(pvt) ? 8 : 4);
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300591 if (ECCx8(pvt))
592 mode = EDAC_S8ECD8ED;
593 else
594 mode = EDAC_S4ECD4ED;
595 } else {
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300596 debugf0("ECC disabled\n");
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300597 mode = EDAC_NONE;
598 }
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300599
600 /* FIXME: need to handle the error codes */
Mauro Carvalho Chehab17cb7b02009-07-20 18:48:18 -0300601 debugf0("DOD Max limits: DIMMS: %d, %d-ranked, %d-banked "
602 "x%x x 0x%x\n",
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300603 numdimms(pvt->info.max_dod),
604 numrank(pvt->info.max_dod >> 2),
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -0300605 numbank(pvt->info.max_dod >> 4),
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300606 numrow(pvt->info.max_dod >> 6),
607 numcol(pvt->info.max_dod >> 9));
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300608
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300609 for (i = 0; i < NUM_CHANS; i++) {
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300610 u32 data, dimm_dod[3], value[8];
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300611
Mauro Carvalho Chehab52a2e4fc2009-10-14 11:21:58 -0300612 if (!pvt->pci_ch[i][0])
613 continue;
614
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300615 if (!CH_ACTIVE(pvt, i)) {
616 debugf0("Channel %i is not active\n", i);
617 continue;
618 }
619 if (CH_DISABLED(pvt, i)) {
620 debugf0("Channel %i is disabled\n", i);
621 continue;
622 }
623
Mauro Carvalho Chehabf122a892009-06-22 22:48:29 -0300624 /* Devices 4-6 function 0 */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300625 pci_read_config_dword(pvt->pci_ch[i][0],
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300626 MC_CHANNEL_DIMM_INIT_PARAMS, &data);
627
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300628 pvt->channel[i].ranks = (data & QUAD_RANK_PRESENT) ?
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -0300629 4 : 2;
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300630
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300631 if (data & REGISTERED_DIMM)
632 mtype = MEM_RDDR3;
Mauro Carvalho Chehab14d2c082009-09-02 23:52:36 -0300633 else
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300634 mtype = MEM_DDR3;
635#if 0
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300636 if (data & THREE_DIMMS_PRESENT)
637 pvt->channel[i].dimms = 3;
638 else if (data & SINGLE_QUAD_RANK_PRESENT)
639 pvt->channel[i].dimms = 1;
640 else
641 pvt->channel[i].dimms = 2;
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300642#endif
643
644 /* Devices 4-6 function 1 */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300645 pci_read_config_dword(pvt->pci_ch[i][1],
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300646 MC_DOD_CH_DIMM0, &dimm_dod[0]);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300647 pci_read_config_dword(pvt->pci_ch[i][1],
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300648 MC_DOD_CH_DIMM1, &dimm_dod[1]);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300649 pci_read_config_dword(pvt->pci_ch[i][1],
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300650 MC_DOD_CH_DIMM2, &dimm_dod[2]);
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300651
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300652 debugf0("Ch%d phy rd%d, wr%d (0x%08x): "
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300653 "%d ranks, %cDIMMs\n",
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300654 i,
655 RDLCH(pvt->info.ch_map, i), WRLCH(pvt->info.ch_map, i),
656 data,
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300657 pvt->channel[i].ranks,
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -0300658 (data & REGISTERED_DIMM) ? 'R' : 'U');
Mauro Carvalho Chehab7dd69532009-06-22 22:48:30 -0300659
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300660 for (j = 0; j < 3; j++) {
661 u32 banks, ranks, rows, cols;
Mauro Carvalho Chehab5566cb72009-06-22 22:48:31 -0300662 u32 size, npages;
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300663
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300664 if (!DIMM_PRESENT(dimm_dod[j]))
665 continue;
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300666
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300667 banks = numbank(MC_DOD_NUMBANK(dimm_dod[j]));
668 ranks = numrank(MC_DOD_NUMRANK(dimm_dod[j]));
669 rows = numrow(MC_DOD_NUMROW(dimm_dod[j]));
670 cols = numcol(MC_DOD_NUMCOL(dimm_dod[j]));
Mauro Carvalho Chehab1c6fed82009-06-22 22:48:30 -0300671
Mauro Carvalho Chehab5566cb72009-06-22 22:48:31 -0300672 /* DDR3 has 8 I/O banks */
673 size = (rows * cols * banks * ranks) >> (20 - 3);
674
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300675 pvt->channel[i].dimms++;
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300676
Mauro Carvalho Chehab17cb7b02009-07-20 18:48:18 -0300677 debugf0("\tdimm %d %d Mb offset: %x, "
678 "bank: %d, rank: %d, row: %#x, col: %#x\n",
679 j, size,
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300680 RANKOFFSET(dimm_dod[j]),
681 banks, ranks, rows, cols);
682
Mauro Carvalho Chehabe9144602010-08-10 20:26:35 -0300683 npages = MiB_TO_PAGES(size);
Mauro Carvalho Chehab5566cb72009-06-22 22:48:31 -0300684
Hidetoshi Seto2e5185f2010-08-20 04:32:56 -0300685 csr = &mci->csrows[csrow];
Mauro Carvalho Chehab5566cb72009-06-22 22:48:31 -0300686 csr->first_page = last_page + 1;
687 last_page += npages;
688 csr->last_page = last_page;
689 csr->nr_pages = npages;
690
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300691 csr->page_mask = 0;
Mauro Carvalho Chehabeb94fc42009-06-22 22:48:31 -0300692 csr->grain = 8;
Hidetoshi Seto2e5185f2010-08-20 04:32:56 -0300693 csr->csrow_idx = csrow;
Mauro Carvalho Chehabeb94fc42009-06-22 22:48:31 -0300694 csr->nr_channels = 1;
695
696 csr->channels[0].chan_idx = i;
697 csr->channels[0].ce_count = 0;
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300698
Hidetoshi Seto2e5185f2010-08-20 04:32:56 -0300699 pvt->csrow_map[i][j] = csrow;
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -0300700
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300701 switch (banks) {
702 case 4:
703 csr->dtype = DEV_X4;
704 break;
705 case 8:
706 csr->dtype = DEV_X8;
707 break;
708 case 16:
709 csr->dtype = DEV_X16;
710 break;
711 default:
712 csr->dtype = DEV_UNKNOWN;
713 }
714
715 csr->edac_mode = mode;
716 csr->mtype = mtype;
717
Hidetoshi Seto2e5185f2010-08-20 04:32:56 -0300718 csrow++;
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300719 }
720
721 pci_read_config_dword(pdev, MC_SAG_CH_0, &value[0]);
722 pci_read_config_dword(pdev, MC_SAG_CH_1, &value[1]);
723 pci_read_config_dword(pdev, MC_SAG_CH_2, &value[2]);
724 pci_read_config_dword(pdev, MC_SAG_CH_3, &value[3]);
725 pci_read_config_dword(pdev, MC_SAG_CH_4, &value[4]);
726 pci_read_config_dword(pdev, MC_SAG_CH_5, &value[5]);
727 pci_read_config_dword(pdev, MC_SAG_CH_6, &value[6]);
728 pci_read_config_dword(pdev, MC_SAG_CH_7, &value[7]);
Mauro Carvalho Chehab17cb7b02009-07-20 18:48:18 -0300729 debugf1("\t[%i] DIVBY3\tREMOVED\tOFFSET\n", i);
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300730 for (j = 0; j < 8; j++)
Mauro Carvalho Chehab17cb7b02009-07-20 18:48:18 -0300731 debugf1("\t\t%#x\t%#x\t%#x\n",
Mauro Carvalho Chehab854d3342009-06-22 22:48:30 -0300732 (value[j] >> 27) & 0x1,
733 (value[j] >> 24) & 0x7,
734 (value[j] && ((1 << 24) - 1)));
Mauro Carvalho Chehab0b2b7b72009-06-22 22:48:29 -0300735 }
736
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -0300737 return 0;
738}
739
740/****************************************************************************
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300741 Error insertion routines
742 ****************************************************************************/
743
744/* The i7core has independent error injection features per channel.
745 However, to have a simpler code, we don't allow enabling error injection
746 on more than one channel.
747 Also, since a change at an inject parameter will be applied only at enable,
748 we're disabling error injection on all write calls to the sysfs nodes that
749 controls the error code injection.
750 */
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -0300751static int disable_inject(const struct mem_ctl_info *mci)
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300752{
753 struct i7core_pvt *pvt = mci->pvt_info;
754
755 pvt->inject.enable = 0;
756
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300757 if (!pvt->pci_ch[pvt->inject.channel][0])
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300758 return -ENODEV;
759
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300760 pci_write_config_dword(pvt->pci_ch[pvt->inject.channel][0],
Mauro Carvalho Chehab4157d9f2009-08-05 20:27:15 -0300761 MC_CHANNEL_ERROR_INJECT, 0);
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300762
763 return 0;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300764}
765
766/*
767 * i7core inject inject.section
768 *
769 * accept and store error injection inject.section value
770 * bit 0 - refers to the lower 32-byte half cacheline
771 * bit 1 - refers to the upper 32-byte half cacheline
772 */
773static ssize_t i7core_inject_section_store(struct mem_ctl_info *mci,
774 const char *data, size_t count)
775{
776 struct i7core_pvt *pvt = mci->pvt_info;
777 unsigned long value;
778 int rc;
779
780 if (pvt->inject.enable)
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -0300781 disable_inject(mci);
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300782
783 rc = strict_strtoul(data, 10, &value);
784 if ((rc < 0) || (value > 3))
Mauro Carvalho Chehab2068def2009-08-05 19:28:27 -0300785 return -EIO;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300786
787 pvt->inject.section = (u32) value;
788 return count;
789}
790
791static ssize_t i7core_inject_section_show(struct mem_ctl_info *mci,
792 char *data)
793{
794 struct i7core_pvt *pvt = mci->pvt_info;
795 return sprintf(data, "0x%08x\n", pvt->inject.section);
796}
797
798/*
799 * i7core inject.type
800 *
801 * accept and store error injection inject.section value
802 * bit 0 - repeat enable - Enable error repetition
803 * bit 1 - inject ECC error
804 * bit 2 - inject parity error
805 */
806static ssize_t i7core_inject_type_store(struct mem_ctl_info *mci,
807 const char *data, size_t count)
808{
809 struct i7core_pvt *pvt = mci->pvt_info;
810 unsigned long value;
811 int rc;
812
813 if (pvt->inject.enable)
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -0300814 disable_inject(mci);
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300815
816 rc = strict_strtoul(data, 10, &value);
817 if ((rc < 0) || (value > 7))
Mauro Carvalho Chehab2068def2009-08-05 19:28:27 -0300818 return -EIO;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300819
820 pvt->inject.type = (u32) value;
821 return count;
822}
823
824static ssize_t i7core_inject_type_show(struct mem_ctl_info *mci,
825 char *data)
826{
827 struct i7core_pvt *pvt = mci->pvt_info;
828 return sprintf(data, "0x%08x\n", pvt->inject.type);
829}
830
831/*
832 * i7core_inject_inject.eccmask_store
833 *
834 * The type of error (UE/CE) will depend on the inject.eccmask value:
835 * Any bits set to a 1 will flip the corresponding ECC bit
836 * Correctable errors can be injected by flipping 1 bit or the bits within
837 * a symbol pair (2 consecutive aligned 8-bit pairs - i.e. 7:0 and 15:8 or
838 * 23:16 and 31:24). Flipping bits in two symbol pairs will cause an
839 * uncorrectable error to be injected.
840 */
841static ssize_t i7core_inject_eccmask_store(struct mem_ctl_info *mci,
842 const char *data, size_t count)
843{
844 struct i7core_pvt *pvt = mci->pvt_info;
845 unsigned long value;
846 int rc;
847
848 if (pvt->inject.enable)
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -0300849 disable_inject(mci);
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300850
851 rc = strict_strtoul(data, 10, &value);
852 if (rc < 0)
Mauro Carvalho Chehab2068def2009-08-05 19:28:27 -0300853 return -EIO;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300854
855 pvt->inject.eccmask = (u32) value;
856 return count;
857}
858
859static ssize_t i7core_inject_eccmask_show(struct mem_ctl_info *mci,
860 char *data)
861{
862 struct i7core_pvt *pvt = mci->pvt_info;
863 return sprintf(data, "0x%08x\n", pvt->inject.eccmask);
864}
865
866/*
867 * i7core_addrmatch
868 *
869 * The type of error (UE/CE) will depend on the inject.eccmask value:
870 * Any bits set to a 1 will flip the corresponding ECC bit
871 * Correctable errors can be injected by flipping 1 bit or the bits within
872 * a symbol pair (2 consecutive aligned 8-bit pairs - i.e. 7:0 and 15:8 or
873 * 23:16 and 31:24). Flipping bits in two symbol pairs will cause an
874 * uncorrectable error to be injected.
875 */
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300876
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -0300877#define DECLARE_ADDR_MATCH(param, limit) \
878static ssize_t i7core_inject_store_##param( \
879 struct mem_ctl_info *mci, \
880 const char *data, size_t count) \
881{ \
Mauro Carvalho Chehabcc301b32009-09-24 16:23:42 -0300882 struct i7core_pvt *pvt; \
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -0300883 long value; \
884 int rc; \
885 \
Mauro Carvalho Chehabcc301b32009-09-24 16:23:42 -0300886 debugf1("%s()\n", __func__); \
887 pvt = mci->pvt_info; \
888 \
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -0300889 if (pvt->inject.enable) \
890 disable_inject(mci); \
891 \
Mauro Carvalho Chehab4f87fad2009-10-04 11:54:56 -0300892 if (!strcasecmp(data, "any") || !strcasecmp(data, "any\n"))\
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -0300893 value = -1; \
894 else { \
895 rc = strict_strtoul(data, 10, &value); \
896 if ((rc < 0) || (value >= limit)) \
897 return -EIO; \
898 } \
899 \
900 pvt->inject.param = value; \
901 \
902 return count; \
903} \
904 \
905static ssize_t i7core_inject_show_##param( \
906 struct mem_ctl_info *mci, \
907 char *data) \
908{ \
Mauro Carvalho Chehabcc301b32009-09-24 16:23:42 -0300909 struct i7core_pvt *pvt; \
910 \
911 pvt = mci->pvt_info; \
912 debugf1("%s() pvt=%p\n", __func__, pvt); \
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -0300913 if (pvt->inject.param < 0) \
914 return sprintf(data, "any\n"); \
915 else \
916 return sprintf(data, "%d\n", pvt->inject.param);\
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300917}
918
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -0300919#define ATTR_ADDR_MATCH(param) \
920 { \
921 .attr = { \
922 .name = #param, \
923 .mode = (S_IRUGO | S_IWUSR) \
924 }, \
925 .show = i7core_inject_show_##param, \
926 .store = i7core_inject_store_##param, \
927 }
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300928
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -0300929DECLARE_ADDR_MATCH(channel, 3);
930DECLARE_ADDR_MATCH(dimm, 3);
931DECLARE_ADDR_MATCH(rank, 4);
932DECLARE_ADDR_MATCH(bank, 32);
933DECLARE_ADDR_MATCH(page, 0x10000);
934DECLARE_ADDR_MATCH(col, 0x4000);
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300935
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -0300936static int write_and_test(struct pci_dev *dev, const int where, const u32 val)
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -0300937{
938 u32 read;
939 int count;
940
Mauro Carvalho Chehab4157d9f2009-08-05 20:27:15 -0300941 debugf0("setting pci %02x:%02x.%x reg=%02x value=%08x\n",
942 dev->bus->number, PCI_SLOT(dev->devfn), PCI_FUNC(dev->devfn),
943 where, val);
944
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -0300945 for (count = 0; count < 10; count++) {
946 if (count)
Mauro Carvalho Chehabb9905382009-08-05 21:36:35 -0300947 msleep(100);
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -0300948 pci_write_config_dword(dev, where, val);
949 pci_read_config_dword(dev, where, &read);
950
951 if (read == val)
952 return 0;
953 }
954
Mauro Carvalho Chehab4157d9f2009-08-05 20:27:15 -0300955 i7core_printk(KERN_ERR, "Error during set pci %02x:%02x.%x reg=%02x "
956 "write=%08x. Read=%08x\n",
957 dev->bus->number, PCI_SLOT(dev->devfn), PCI_FUNC(dev->devfn),
958 where, val, read);
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -0300959
960 return -EINVAL;
961}
962
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300963/*
964 * This routine prepares the Memory Controller for error injection.
965 * The error will be injected when some process tries to write to the
966 * memory that matches the given criteria.
967 * The criteria can be set in terms of a mask where dimm, rank, bank, page
968 * and col can be specified.
969 * A -1 value for any of the mask items will make the MCU to ignore
970 * that matching criteria for error injection.
971 *
972 * It should be noticed that the error will only happen after a write operation
973 * on a memory that matches the condition. if REPEAT_EN is not enabled at
974 * inject mask, then it will produce just one error. Otherwise, it will repeat
975 * until the injectmask would be cleaned.
976 *
977 * FIXME: This routine assumes that MAXNUMDIMMS value of MC_MAX_DOD
978 * is reliable enough to check if the MC is using the
979 * three channels. However, this is not clear at the datasheet.
980 */
981static ssize_t i7core_inject_enable_store(struct mem_ctl_info *mci,
982 const char *data, size_t count)
983{
984 struct i7core_pvt *pvt = mci->pvt_info;
985 u32 injectmask;
986 u64 mask = 0;
987 int rc;
988 long enable;
989
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -0300990 if (!pvt->pci_ch[pvt->inject.channel][0])
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -0300991 return 0;
992
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -0300993 rc = strict_strtoul(data, 10, &enable);
994 if ((rc < 0))
995 return 0;
996
997 if (enable) {
998 pvt->inject.enable = 1;
999 } else {
1000 disable_inject(mci);
1001 return count;
1002 }
1003
1004 /* Sets pvt->inject.dimm mask */
1005 if (pvt->inject.dimm < 0)
Alan Cox486dd092009-11-08 01:34:27 -02001006 mask |= 1LL << 41;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001007 else {
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001008 if (pvt->channel[pvt->inject.channel].dimms > 2)
Alan Cox486dd092009-11-08 01:34:27 -02001009 mask |= (pvt->inject.dimm & 0x3LL) << 35;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001010 else
Alan Cox486dd092009-11-08 01:34:27 -02001011 mask |= (pvt->inject.dimm & 0x1LL) << 36;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001012 }
1013
1014 /* Sets pvt->inject.rank mask */
1015 if (pvt->inject.rank < 0)
Alan Cox486dd092009-11-08 01:34:27 -02001016 mask |= 1LL << 40;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001017 else {
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001018 if (pvt->channel[pvt->inject.channel].dimms > 2)
Alan Cox486dd092009-11-08 01:34:27 -02001019 mask |= (pvt->inject.rank & 0x1LL) << 34;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001020 else
Alan Cox486dd092009-11-08 01:34:27 -02001021 mask |= (pvt->inject.rank & 0x3LL) << 34;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001022 }
1023
1024 /* Sets pvt->inject.bank mask */
1025 if (pvt->inject.bank < 0)
Alan Cox486dd092009-11-08 01:34:27 -02001026 mask |= 1LL << 39;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001027 else
Alan Cox486dd092009-11-08 01:34:27 -02001028 mask |= (pvt->inject.bank & 0x15LL) << 30;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001029
1030 /* Sets pvt->inject.page mask */
1031 if (pvt->inject.page < 0)
Alan Cox486dd092009-11-08 01:34:27 -02001032 mask |= 1LL << 38;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001033 else
Alan Cox486dd092009-11-08 01:34:27 -02001034 mask |= (pvt->inject.page & 0xffff) << 14;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001035
1036 /* Sets pvt->inject.column mask */
1037 if (pvt->inject.col < 0)
Alan Cox486dd092009-11-08 01:34:27 -02001038 mask |= 1LL << 37;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001039 else
Alan Cox486dd092009-11-08 01:34:27 -02001040 mask |= (pvt->inject.col & 0x3fff);
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001041
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001042 /*
1043 * bit 0: REPEAT_EN
1044 * bits 1-2: MASK_HALF_CACHELINE
1045 * bit 3: INJECT_ECC
1046 * bit 4: INJECT_ADDR_PARITY
1047 */
1048
Mauro Carvalho Chehab7b029d02009-06-22 22:48:29 -03001049 injectmask = (pvt->inject.type & 1) |
1050 (pvt->inject.section & 0x3) << 1 |
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001051 (pvt->inject.type & 0x6) << (3 - 1);
1052
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -03001053 /* Unlock writes to registers - this register is write only */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001054 pci_write_config_dword(pvt->pci_noncore,
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -03001055 MC_CFG_CONTROL, 0x2);
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001056
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001057 write_and_test(pvt->pci_ch[pvt->inject.channel][0],
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -03001058 MC_CHANNEL_ADDR_MATCH, mask);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001059 write_and_test(pvt->pci_ch[pvt->inject.channel][0],
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -03001060 MC_CHANNEL_ADDR_MATCH + 4, mask >> 32L);
1061
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001062 write_and_test(pvt->pci_ch[pvt->inject.channel][0],
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -03001063 MC_CHANNEL_ERROR_MASK, pvt->inject.eccmask);
1064
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001065 write_and_test(pvt->pci_ch[pvt->inject.channel][0],
Mauro Carvalho Chehab4157d9f2009-08-05 20:27:15 -03001066 MC_CHANNEL_ERROR_INJECT, injectmask);
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -03001067
1068 /*
1069 * This is something undocumented, based on my tests
1070 * Without writing 8 to this register, errors aren't injected. Not sure
1071 * why.
1072 */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001073 pci_write_config_dword(pvt->pci_noncore,
Mauro Carvalho Chehab276b8242009-07-22 21:45:50 -03001074 MC_CFG_CONTROL, 8);
1075
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -03001076 debugf0("Error inject addr match 0x%016llx, ecc 0x%08x,"
1077 " inject 0x%08x\n",
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001078 mask, pvt->inject.eccmask, injectmask);
1079
Mauro Carvalho Chehab7b029d02009-06-22 22:48:29 -03001080
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001081 return count;
1082}
1083
1084static ssize_t i7core_inject_enable_show(struct mem_ctl_info *mci,
1085 char *data)
1086{
1087 struct i7core_pvt *pvt = mci->pvt_info;
Mauro Carvalho Chehab7b029d02009-06-22 22:48:29 -03001088 u32 injectmask;
1089
Mauro Carvalho Chehab52a2e4fc2009-10-14 11:21:58 -03001090 if (!pvt->pci_ch[pvt->inject.channel][0])
1091 return 0;
1092
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001093 pci_read_config_dword(pvt->pci_ch[pvt->inject.channel][0],
Mauro Carvalho Chehab4157d9f2009-08-05 20:27:15 -03001094 MC_CHANNEL_ERROR_INJECT, &injectmask);
Mauro Carvalho Chehab7b029d02009-06-22 22:48:29 -03001095
1096 debugf0("Inject error read: 0x%018x\n", injectmask);
1097
1098 if (injectmask & 0x0c)
1099 pvt->inject.enable = 1;
1100
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001101 return sprintf(data, "%d\n", pvt->inject.enable);
1102}
1103
Mauro Carvalho Chehabf338d732009-09-24 17:25:43 -03001104#define DECLARE_COUNTER(param) \
1105static ssize_t i7core_show_counter_##param( \
1106 struct mem_ctl_info *mci, \
1107 char *data) \
1108{ \
1109 struct i7core_pvt *pvt = mci->pvt_info; \
1110 \
1111 debugf1("%s() \n", __func__); \
1112 if (!pvt->ce_count_available || (pvt->is_registered)) \
1113 return sprintf(data, "data unavailable\n"); \
1114 return sprintf(data, "%lu\n", \
1115 pvt->udimm_ce_count[param]); \
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001116}
1117
Mauro Carvalho Chehabf338d732009-09-24 17:25:43 -03001118#define ATTR_COUNTER(param) \
1119 { \
1120 .attr = { \
1121 .name = __stringify(udimm##param), \
1122 .mode = (S_IRUGO | S_IWUSR) \
1123 }, \
1124 .show = i7core_show_counter_##param \
1125 }
1126
1127DECLARE_COUNTER(0);
1128DECLARE_COUNTER(1);
1129DECLARE_COUNTER(2);
1130
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001131/*
1132 * Sysfs struct
1133 */
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -03001134
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001135static const struct mcidev_sysfs_attribute i7core_addrmatch_attrs[] = {
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -03001136 ATTR_ADDR_MATCH(channel),
1137 ATTR_ADDR_MATCH(dimm),
1138 ATTR_ADDR_MATCH(rank),
1139 ATTR_ADDR_MATCH(bank),
1140 ATTR_ADDR_MATCH(page),
1141 ATTR_ADDR_MATCH(col),
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001142 { } /* End of list */
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -03001143};
1144
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001145static const struct mcidev_sysfs_group i7core_inject_addrmatch = {
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -03001146 .name = "inject_addrmatch",
1147 .mcidev_attr = i7core_addrmatch_attrs,
1148};
1149
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001150static const struct mcidev_sysfs_attribute i7core_udimm_counters_attrs[] = {
Mauro Carvalho Chehabf338d732009-09-24 17:25:43 -03001151 ATTR_COUNTER(0),
1152 ATTR_COUNTER(1),
1153 ATTR_COUNTER(2),
Marcin Slusarz64aab722010-09-30 15:15:30 -07001154 { .attr = { .name = NULL } }
Mauro Carvalho Chehabf338d732009-09-24 17:25:43 -03001155};
1156
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001157static const struct mcidev_sysfs_group i7core_udimm_counters = {
Mauro Carvalho Chehabf338d732009-09-24 17:25:43 -03001158 .name = "all_channel_counts",
1159 .mcidev_attr = i7core_udimm_counters_attrs,
1160};
1161
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001162static const struct mcidev_sysfs_attribute i7core_sysfs_rdimm_attrs[] = {
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001163 {
1164 .attr = {
1165 .name = "inject_section",
1166 .mode = (S_IRUGO | S_IWUSR)
1167 },
1168 .show = i7core_inject_section_show,
1169 .store = i7core_inject_section_store,
1170 }, {
1171 .attr = {
1172 .name = "inject_type",
1173 .mode = (S_IRUGO | S_IWUSR)
1174 },
1175 .show = i7core_inject_type_show,
1176 .store = i7core_inject_type_store,
1177 }, {
1178 .attr = {
1179 .name = "inject_eccmask",
1180 .mode = (S_IRUGO | S_IWUSR)
1181 },
1182 .show = i7core_inject_eccmask_show,
1183 .store = i7core_inject_eccmask_store,
1184 }, {
Mauro Carvalho Chehaba5538e52009-09-23 18:56:47 -03001185 .grp = &i7core_inject_addrmatch,
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001186 }, {
1187 .attr = {
1188 .name = "inject_enable",
1189 .mode = (S_IRUGO | S_IWUSR)
1190 },
1191 .show = i7core_inject_enable_show,
1192 .store = i7core_inject_enable_store,
1193 },
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001194 { } /* End of list */
1195};
1196
1197static const struct mcidev_sysfs_attribute i7core_sysfs_udimm_attrs[] = {
1198 {
1199 .attr = {
1200 .name = "inject_section",
1201 .mode = (S_IRUGO | S_IWUSR)
1202 },
1203 .show = i7core_inject_section_show,
1204 .store = i7core_inject_section_store,
1205 }, {
1206 .attr = {
1207 .name = "inject_type",
1208 .mode = (S_IRUGO | S_IWUSR)
1209 },
1210 .show = i7core_inject_type_show,
1211 .store = i7core_inject_type_store,
1212 }, {
1213 .attr = {
1214 .name = "inject_eccmask",
1215 .mode = (S_IRUGO | S_IWUSR)
1216 },
1217 .show = i7core_inject_eccmask_show,
1218 .store = i7core_inject_eccmask_store,
1219 }, {
1220 .grp = &i7core_inject_addrmatch,
1221 }, {
1222 .attr = {
1223 .name = "inject_enable",
1224 .mode = (S_IRUGO | S_IWUSR)
1225 },
1226 .show = i7core_inject_enable_show,
1227 .store = i7core_inject_enable_store,
1228 }, {
1229 .grp = &i7core_udimm_counters,
1230 },
1231 { } /* End of list */
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001232};
1233
1234/****************************************************************************
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001235 Device initialization routines: put/get, init/exit
1236 ****************************************************************************/
1237
1238/*
Hidetoshi Seto64c10f62010-08-20 04:28:14 -03001239 * i7core_put_all_devices 'put' all the devices that we have
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001240 * reserved via 'get'
1241 */
Mauro Carvalho Chehab13d6e9b2009-09-05 12:15:20 -03001242static void i7core_put_devices(struct i7core_dev *i7core_dev)
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001243{
Mauro Carvalho Chehab13d6e9b2009-09-05 12:15:20 -03001244 int i;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001245
Mauro Carvalho Chehab22e6bcb2009-09-05 23:06:50 -03001246 debugf0(__FILE__ ": %s()\n", __func__);
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001247 for (i = 0; i < i7core_dev->n_devs; i++) {
Mauro Carvalho Chehab22e6bcb2009-09-05 23:06:50 -03001248 struct pci_dev *pdev = i7core_dev->pdev[i];
1249 if (!pdev)
1250 continue;
1251 debugf0("Removing dev %02x:%02x.%d\n",
1252 pdev->bus->number,
1253 PCI_SLOT(pdev->devfn), PCI_FUNC(pdev->devfn));
1254 pci_dev_put(pdev);
1255 }
Mauro Carvalho Chehab13d6e9b2009-09-05 12:15:20 -03001256}
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03001257
Mauro Carvalho Chehab13d6e9b2009-09-05 12:15:20 -03001258static void i7core_put_all_devices(void)
1259{
Mauro Carvalho Chehab42538682009-09-24 09:59:13 -03001260 struct i7core_dev *i7core_dev, *tmp;
Mauro Carvalho Chehab13d6e9b2009-09-05 12:15:20 -03001261
Mauro Carvalho Chehab39300e72010-08-11 23:40:15 -03001262 list_for_each_entry_safe(i7core_dev, tmp, &i7core_edac_list, list) {
Mauro Carvalho Chehab13d6e9b2009-09-05 12:15:20 -03001263 i7core_put_devices(i7core_dev);
Hidetoshi Seto2aa9be42010-08-20 04:25:00 -03001264 free_i7core_dev(i7core_dev);
Mauro Carvalho Chehab39300e72010-08-11 23:40:15 -03001265 }
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001266}
1267
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001268static void __init i7core_xeon_pci_fixup(const struct pci_id_table *table)
Keith Manntheybc2d7242009-09-03 00:05:05 -03001269{
1270 struct pci_dev *pdev = NULL;
1271 int i;
Mauro Carvalho Chehab54a08ab2010-08-19 15:51:00 -03001272
Keith Manntheybc2d7242009-09-03 00:05:05 -03001273 /*
David Sterbae7bf0682010-12-27 16:51:15 +01001274 * On Xeon 55xx, the Intel Quick Path Arch Generic Non-core pci buses
Keith Manntheybc2d7242009-09-03 00:05:05 -03001275 * aren't announced by acpi. So, we need to use a legacy scan probing
1276 * to detect them
1277 */
Vernon Mauerybd9e19c2010-05-18 19:02:50 -03001278 while (table && table->descr) {
1279 pdev = pci_get_device(PCI_VENDOR_ID_INTEL, table->descr[0].dev_id, NULL);
1280 if (unlikely(!pdev)) {
1281 for (i = 0; i < MAX_SOCKET_BUSES; i++)
1282 pcibios_scan_specific_bus(255-i);
1283 }
Mauro Carvalho Chehabbda14282010-06-30 01:41:35 -03001284 pci_dev_put(pdev);
Vernon Mauerybd9e19c2010-05-18 19:02:50 -03001285 table++;
Keith Manntheybc2d7242009-09-03 00:05:05 -03001286 }
1287}
1288
Mauro Carvalho Chehabbda14282010-06-30 01:41:35 -03001289static unsigned i7core_pci_lastbus(void)
1290{
1291 int last_bus = 0, bus;
1292 struct pci_bus *b = NULL;
1293
1294 while ((b = pci_find_next_bus(b)) != NULL) {
1295 bus = b->number;
1296 debugf0("Found bus %d\n", bus);
1297 if (bus > last_bus)
1298 last_bus = bus;
1299 }
1300
1301 debugf0("Last bus %d\n", last_bus);
1302
1303 return last_bus;
1304}
1305
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001306/*
Hidetoshi Seto64c10f62010-08-20 04:28:14 -03001307 * i7core_get_all_devices Find and perform 'get' operation on the MCH's
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001308 * device/functions we want to reference for this driver
1309 *
1310 * Need to 'get' device 16 func 1 and func 2
1311 */
Hidetoshi Setob197cba2010-08-20 04:24:31 -03001312static int i7core_get_onedevice(struct pci_dev **prev,
1313 const struct pci_id_table *table,
1314 const unsigned devno,
1315 const unsigned last_bus)
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001316{
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03001317 struct i7core_dev *i7core_dev;
Hidetoshi Setob197cba2010-08-20 04:24:31 -03001318 const struct pci_id_descr *dev_descr = &table->descr[devno];
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03001319
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -03001320 struct pci_dev *pdev = NULL;
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -03001321 u8 bus = 0;
1322 u8 socket = 0;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001323
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001324 pdev = pci_get_device(PCI_VENDOR_ID_INTEL,
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001325 dev_descr->dev_id, *prev);
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001326
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001327 if (!pdev) {
1328 if (*prev) {
1329 *prev = pdev;
1330 return 0;
Mauro Carvalho Chehabd1fd4fb2009-07-10 18:39:53 -03001331 }
1332
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001333 if (dev_descr->optional)
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001334 return 0;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001335
Vernon Mauerybd9e19c2010-05-18 19:02:50 -03001336 if (devno == 0)
1337 return -ENODEV;
1338
Daniel J Bluemanab089372010-07-23 23:16:52 +01001339 i7core_printk(KERN_INFO,
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001340 "Device not found: dev %02x.%d PCI ID %04x:%04x\n",
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001341 dev_descr->dev, dev_descr->func,
1342 PCI_VENDOR_ID_INTEL, dev_descr->dev_id);
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001343
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001344 /* End of list, leave */
1345 return -ENODEV;
1346 }
1347 bus = pdev->bus->number;
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -03001348
Mauro Carvalho Chehabbda14282010-06-30 01:41:35 -03001349 socket = last_bus - bus;
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001350
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03001351 i7core_dev = get_i7core_dev(socket);
1352 if (!i7core_dev) {
Hidetoshi Seto848b2f72010-08-20 04:24:44 -03001353 i7core_dev = alloc_i7core_dev(socket, table);
Hidetoshi Seto28966372010-08-20 04:28:51 -03001354 if (!i7core_dev) {
1355 pci_dev_put(pdev);
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03001356 return -ENOMEM;
Hidetoshi Seto28966372010-08-20 04:28:51 -03001357 }
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001358 }
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001359
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03001360 if (i7core_dev->pdev[devno]) {
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001361 i7core_printk(KERN_ERR,
1362 "Duplicated device for "
1363 "dev %02x:%02x.%d PCI ID %04x:%04x\n",
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001364 bus, dev_descr->dev, dev_descr->func,
1365 PCI_VENDOR_ID_INTEL, dev_descr->dev_id);
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001366 pci_dev_put(pdev);
1367 return -ENODEV;
1368 }
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001369
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03001370 i7core_dev->pdev[devno] = pdev;
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001371
1372 /* Sanity check */
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001373 if (unlikely(PCI_SLOT(pdev->devfn) != dev_descr->dev ||
1374 PCI_FUNC(pdev->devfn) != dev_descr->func)) {
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001375 i7core_printk(KERN_ERR,
1376 "Device PCI ID %04x:%04x "
1377 "has dev %02x:%02x.%d instead of dev %02x:%02x.%d\n",
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001378 PCI_VENDOR_ID_INTEL, dev_descr->dev_id,
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001379 bus, PCI_SLOT(pdev->devfn), PCI_FUNC(pdev->devfn),
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001380 bus, dev_descr->dev, dev_descr->func);
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001381 return -ENODEV;
1382 }
1383
1384 /* Be sure that the device is enabled */
1385 if (unlikely(pci_enable_device(pdev) < 0)) {
1386 i7core_printk(KERN_ERR,
1387 "Couldn't enable "
1388 "dev %02x:%02x.%d PCI ID %04x:%04x\n",
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001389 bus, dev_descr->dev, dev_descr->func,
1390 PCI_VENDOR_ID_INTEL, dev_descr->dev_id);
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001391 return -ENODEV;
1392 }
1393
Mauro Carvalho Chehabd4c27792009-09-05 04:12:02 -03001394 debugf0("Detected socket %d dev %02x:%02x.%d PCI ID %04x:%04x\n",
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001395 socket, bus, dev_descr->dev,
1396 dev_descr->func,
1397 PCI_VENDOR_ID_INTEL, dev_descr->dev_id);
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001398
Mauro Carvalho Chehaba3e15412010-08-21 08:52:41 -03001399 /*
1400 * As stated on drivers/pci/search.c, the reference count for
1401 * @from is always decremented if it is not %NULL. So, as we need
1402 * to get all devices up to null, we need to do a get for the device
1403 */
1404 pci_dev_get(pdev);
1405
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001406 *prev = pdev;
1407
1408 return 0;
1409}
1410
Hidetoshi Seto64c10f62010-08-20 04:28:14 -03001411static int i7core_get_all_devices(void)
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001412{
Mauro Carvalho Chehab3c52cc52010-10-24 11:12:28 -02001413 int i, rc, last_bus;
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001414 struct pci_dev *pdev = NULL;
Mauro Carvalho Chehab3c52cc52010-10-24 11:12:28 -02001415 const struct pci_id_table *table = pci_dev_table;
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001416
Mauro Carvalho Chehabbda14282010-06-30 01:41:35 -03001417 last_bus = i7core_pci_lastbus();
1418
Mauro Carvalho Chehab3c52cc52010-10-24 11:12:28 -02001419 while (table && table->descr) {
Vernon Mauerybd9e19c2010-05-18 19:02:50 -03001420 for (i = 0; i < table->n_devs; i++) {
1421 pdev = NULL;
1422 do {
Hidetoshi Setob197cba2010-08-20 04:24:31 -03001423 rc = i7core_get_onedevice(&pdev, table, i,
Mauro Carvalho Chehabbda14282010-06-30 01:41:35 -03001424 last_bus);
Vernon Mauerybd9e19c2010-05-18 19:02:50 -03001425 if (rc < 0) {
1426 if (i == 0) {
1427 i = table->n_devs;
1428 break;
1429 }
1430 i7core_put_all_devices();
1431 return -ENODEV;
1432 }
1433 } while (pdev);
1434 }
Mauro Carvalho Chehab3c52cc52010-10-24 11:12:28 -02001435 table++;
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001436 }
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03001437
Mauro Carvalho Chehabc77720b2009-07-18 10:43:08 -03001438 return 0;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001439}
1440
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001441static int mci_bind_devs(struct mem_ctl_info *mci,
1442 struct i7core_dev *i7core_dev)
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001443{
1444 struct i7core_pvt *pvt = mci->pvt_info;
1445 struct pci_dev *pdev;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001446 int i, func, slot;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001447
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001448 pvt->is_registered = 0;
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03001449 for (i = 0; i < i7core_dev->n_devs; i++) {
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001450 pdev = i7core_dev->pdev[i];
1451 if (!pdev)
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03001452 continue;
1453
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001454 func = PCI_FUNC(pdev->devfn);
1455 slot = PCI_SLOT(pdev->devfn);
1456 if (slot == 3) {
1457 if (unlikely(func > MAX_MCR_FUNC))
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001458 goto error;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001459 pvt->pci_mcr[func] = pdev;
1460 } else if (likely(slot >= 4 && slot < 4 + NUM_CHANS)) {
1461 if (unlikely(func > MAX_CHAN_FUNC))
1462 goto error;
1463 pvt->pci_ch[slot - 4][func] = pdev;
1464 } else if (!slot && !func)
1465 pvt->pci_noncore = pdev;
1466 else
1467 goto error;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001468
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001469 debugf0("Associated fn %d.%d, dev = %p, socket %d\n",
1470 PCI_SLOT(pdev->devfn), PCI_FUNC(pdev->devfn),
1471 pdev, i7core_dev->socket);
Mauro Carvalho Chehab14d2c082009-09-02 23:52:36 -03001472
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001473 if (PCI_SLOT(pdev->devfn) == 3 &&
1474 PCI_FUNC(pdev->devfn) == 2)
1475 pvt->is_registered = 1;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001476 }
Mauro Carvalho Chehabe9bd2e72009-07-09 22:14:35 -03001477
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001478 return 0;
1479
1480error:
1481 i7core_printk(KERN_ERR, "Device %d, function %d "
1482 "is out of the expected range\n",
1483 slot, func);
1484 return -EINVAL;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001485}
1486
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001487/****************************************************************************
1488 Error check routines
1489 ****************************************************************************/
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001490static void i7core_rdimm_update_csrow(struct mem_ctl_info *mci,
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001491 const int chan,
1492 const int dimm,
1493 const int add)
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001494{
1495 char *msg;
1496 struct i7core_pvt *pvt = mci->pvt_info;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001497 int row = pvt->csrow_map[chan][dimm], i;
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001498
1499 for (i = 0; i < add; i++) {
1500 msg = kasprintf(GFP_KERNEL, "Corrected error "
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001501 "(Socket=%d channel=%d dimm=%d)",
1502 pvt->i7core_dev->socket, chan, dimm);
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001503
1504 edac_mc_handle_fbd_ce(mci, row, 0, msg);
1505 kfree (msg);
1506 }
1507}
1508
1509static void i7core_rdimm_update_ce_count(struct mem_ctl_info *mci,
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001510 const int chan,
1511 const int new0,
1512 const int new1,
1513 const int new2)
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001514{
1515 struct i7core_pvt *pvt = mci->pvt_info;
1516 int add0 = 0, add1 = 0, add2 = 0;
1517 /* Updates CE counters if it is not the first time here */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001518 if (pvt->ce_count_available) {
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001519 /* Updates CE counters */
1520
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001521 add2 = new2 - pvt->rdimm_last_ce_count[chan][2];
1522 add1 = new1 - pvt->rdimm_last_ce_count[chan][1];
1523 add0 = new0 - pvt->rdimm_last_ce_count[chan][0];
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001524
1525 if (add2 < 0)
1526 add2 += 0x7fff;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001527 pvt->rdimm_ce_count[chan][2] += add2;
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001528
1529 if (add1 < 0)
1530 add1 += 0x7fff;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001531 pvt->rdimm_ce_count[chan][1] += add1;
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001532
1533 if (add0 < 0)
1534 add0 += 0x7fff;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001535 pvt->rdimm_ce_count[chan][0] += add0;
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001536 } else
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001537 pvt->ce_count_available = 1;
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001538
1539 /* Store the new values */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001540 pvt->rdimm_last_ce_count[chan][2] = new2;
1541 pvt->rdimm_last_ce_count[chan][1] = new1;
1542 pvt->rdimm_last_ce_count[chan][0] = new0;
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001543
1544 /*updated the edac core */
1545 if (add0 != 0)
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001546 i7core_rdimm_update_csrow(mci, chan, 0, add0);
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001547 if (add1 != 0)
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001548 i7core_rdimm_update_csrow(mci, chan, 1, add1);
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001549 if (add2 != 0)
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001550 i7core_rdimm_update_csrow(mci, chan, 2, add2);
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001551
1552}
1553
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001554static void i7core_rdimm_check_mc_ecc_err(struct mem_ctl_info *mci)
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001555{
1556 struct i7core_pvt *pvt = mci->pvt_info;
1557 u32 rcv[3][2];
1558 int i, new0, new1, new2;
1559
1560 /*Read DEV 3: FUN 2: MC_COR_ECC_CNT regs directly*/
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001561 pci_read_config_dword(pvt->pci_mcr[2], MC_COR_ECC_CNT_0,
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001562 &rcv[0][0]);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001563 pci_read_config_dword(pvt->pci_mcr[2], MC_COR_ECC_CNT_1,
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001564 &rcv[0][1]);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001565 pci_read_config_dword(pvt->pci_mcr[2], MC_COR_ECC_CNT_2,
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001566 &rcv[1][0]);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001567 pci_read_config_dword(pvt->pci_mcr[2], MC_COR_ECC_CNT_3,
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001568 &rcv[1][1]);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001569 pci_read_config_dword(pvt->pci_mcr[2], MC_COR_ECC_CNT_4,
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001570 &rcv[2][0]);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001571 pci_read_config_dword(pvt->pci_mcr[2], MC_COR_ECC_CNT_5,
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001572 &rcv[2][1]);
1573 for (i = 0 ; i < 3; i++) {
1574 debugf3("MC_COR_ECC_CNT%d = 0x%x; MC_COR_ECC_CNT%d = 0x%x\n",
1575 (i * 2), rcv[i][0], (i * 2) + 1, rcv[i][1]);
1576 /*if the channel has 3 dimms*/
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001577 if (pvt->channel[i].dimms > 2) {
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001578 new0 = DIMM_BOT_COR_ERR(rcv[i][0]);
1579 new1 = DIMM_TOP_COR_ERR(rcv[i][0]);
1580 new2 = DIMM_BOT_COR_ERR(rcv[i][1]);
1581 } else {
1582 new0 = DIMM_TOP_COR_ERR(rcv[i][0]) +
1583 DIMM_BOT_COR_ERR(rcv[i][0]);
1584 new1 = DIMM_TOP_COR_ERR(rcv[i][1]) +
1585 DIMM_BOT_COR_ERR(rcv[i][1]);
1586 new2 = 0;
1587 }
1588
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001589 i7core_rdimm_update_ce_count(mci, i, new0, new1, new2);
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001590 }
1591}
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001592
1593/* This function is based on the device 3 function 4 registers as described on:
1594 * Intel Xeon Processor 5500 Series Datasheet Volume 2
1595 * http://www.intel.com/Assets/PDF/datasheet/321322.pdf
1596 * also available at:
1597 * http://www.arrownac.com/manufacturers/intel/s/nehalem/5500-datasheet-v2.pdf
1598 */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001599static void i7core_udimm_check_mc_ecc_err(struct mem_ctl_info *mci)
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001600{
1601 struct i7core_pvt *pvt = mci->pvt_info;
1602 u32 rcv1, rcv0;
1603 int new0, new1, new2;
1604
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001605 if (!pvt->pci_mcr[4]) {
Mauro Carvalho Chehabb9905382009-08-05 21:36:35 -03001606 debugf0("%s MCR registers not found\n", __func__);
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001607 return;
1608 }
1609
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001610 /* Corrected test errors */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001611 pci_read_config_dword(pvt->pci_mcr[4], MC_TEST_ERR_RCV1, &rcv1);
1612 pci_read_config_dword(pvt->pci_mcr[4], MC_TEST_ERR_RCV0, &rcv0);
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001613
1614 /* Store the new values */
1615 new2 = DIMM2_COR_ERR(rcv1);
1616 new1 = DIMM1_COR_ERR(rcv0);
1617 new0 = DIMM0_COR_ERR(rcv0);
1618
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001619 /* Updates CE counters if it is not the first time here */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001620 if (pvt->ce_count_available) {
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001621 /* Updates CE counters */
1622 int add0, add1, add2;
1623
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001624 add2 = new2 - pvt->udimm_last_ce_count[2];
1625 add1 = new1 - pvt->udimm_last_ce_count[1];
1626 add0 = new0 - pvt->udimm_last_ce_count[0];
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001627
1628 if (add2 < 0)
1629 add2 += 0x7fff;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001630 pvt->udimm_ce_count[2] += add2;
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001631
1632 if (add1 < 0)
1633 add1 += 0x7fff;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001634 pvt->udimm_ce_count[1] += add1;
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001635
1636 if (add0 < 0)
1637 add0 += 0x7fff;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001638 pvt->udimm_ce_count[0] += add0;
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001639
1640 if (add0 | add1 | add2)
1641 i7core_printk(KERN_ERR, "New Corrected error(s): "
1642 "dimm0: +%d, dimm1: +%d, dimm2 +%d\n",
1643 add0, add1, add2);
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001644 } else
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001645 pvt->ce_count_available = 1;
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001646
1647 /* Store the new values */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001648 pvt->udimm_last_ce_count[2] = new2;
1649 pvt->udimm_last_ce_count[1] = new1;
1650 pvt->udimm_last_ce_count[0] = new0;
Mauro Carvalho Chehab442305b2009-06-22 22:48:29 -03001651}
1652
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001653/*
1654 * According with tables E-11 and E-12 of chapter E.3.3 of Intel 64 and IA-32
1655 * Architectures Software Developer’s Manual Volume 3B.
Mauro Carvalho Chehabf237fcf2009-07-15 19:53:24 -03001656 * Nehalem are defined as family 0x06, model 0x1a
1657 *
1658 * The MCA registers used here are the following ones:
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001659 * struct mce field MCA Register
Mauro Carvalho Chehabf237fcf2009-07-15 19:53:24 -03001660 * m->status MSR_IA32_MC8_STATUS
1661 * m->addr MSR_IA32_MC8_ADDR
1662 * m->misc MSR_IA32_MC8_MISC
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001663 * In the case of Nehalem, the error information is masked at .status and .misc
1664 * fields
1665 */
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001666static void i7core_mce_output_error(struct mem_ctl_info *mci,
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001667 const struct mce *m)
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001668{
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001669 struct i7core_pvt *pvt = mci->pvt_info;
Mauro Carvalho Chehaba6395392009-07-17 10:54:23 -03001670 char *type, *optype, *err, *msg;
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001671 unsigned long error = m->status & 0x1ff0000l;
Mauro Carvalho Chehaba6395392009-07-17 10:54:23 -03001672 u32 optypenum = (m->status >> 4) & 0x07;
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001673 u32 core_err_cnt = (m->status >> 38) && 0x7fff;
1674 u32 dimm = (m->misc >> 16) & 0x3;
1675 u32 channel = (m->misc >> 18) & 0x3;
1676 u32 syndrome = m->misc >> 32;
1677 u32 errnum = find_first_bit(&error, 32);
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001678 int csrow;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001679
Mauro Carvalho Chehabc5d34522009-07-17 10:28:15 -03001680 if (m->mcgstatus & 1)
1681 type = "FATAL";
1682 else
1683 type = "NON_FATAL";
1684
Mauro Carvalho Chehaba6395392009-07-17 10:54:23 -03001685 switch (optypenum) {
Mauro Carvalho Chehabb9905382009-08-05 21:36:35 -03001686 case 0:
1687 optype = "generic undef request";
1688 break;
1689 case 1:
1690 optype = "read error";
1691 break;
1692 case 2:
1693 optype = "write error";
1694 break;
1695 case 3:
1696 optype = "addr/cmd error";
1697 break;
1698 case 4:
1699 optype = "scrubbing error";
1700 break;
1701 default:
1702 optype = "reserved";
1703 break;
Mauro Carvalho Chehaba6395392009-07-17 10:54:23 -03001704 }
1705
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001706 switch (errnum) {
1707 case 16:
1708 err = "read ECC error";
1709 break;
1710 case 17:
1711 err = "RAS ECC error";
1712 break;
1713 case 18:
1714 err = "write parity error";
1715 break;
1716 case 19:
1717 err = "redundacy loss";
1718 break;
1719 case 20:
1720 err = "reserved";
1721 break;
1722 case 21:
1723 err = "memory range error";
1724 break;
1725 case 22:
1726 err = "RTID out of range";
1727 break;
1728 case 23:
1729 err = "address parity error";
1730 break;
1731 case 24:
1732 err = "byte enable parity error";
1733 break;
1734 default:
1735 err = "unknown";
1736 }
1737
Mauro Carvalho Chehabf237fcf2009-07-15 19:53:24 -03001738 /* FIXME: should convert addr into bank and rank information */
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001739 msg = kasprintf(GFP_ATOMIC,
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001740 "%s (addr = 0x%08llx, cpu=%d, Dimm=%d, Channel=%d, "
Mauro Carvalho Chehaba6395392009-07-17 10:54:23 -03001741 "syndrome=0x%08x, count=%d, Err=%08llx:%08llx (%s: %s))\n",
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001742 type, (long long) m->addr, m->cpu, dimm, channel,
Mauro Carvalho Chehaba6395392009-07-17 10:54:23 -03001743 syndrome, core_err_cnt, (long long)m->status,
1744 (long long)m->misc, optype, err);
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001745
1746 debugf0("%s", msg);
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001747
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001748 csrow = pvt->csrow_map[channel][dimm];
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001749
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001750 /* Call the helper to output message */
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001751 if (m->mcgstatus & 1)
1752 edac_mc_handle_fbd_ue(mci, csrow, 0,
1753 0 /* FIXME: should be channel here */, msg);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001754 else if (!pvt->is_registered)
Mauro Carvalho Chehabb4e8f0b2009-09-02 23:49:59 -03001755 edac_mc_handle_fbd_ce(mci, csrow,
1756 0 /* FIXME: should be channel here */, msg);
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001757
1758 kfree(msg);
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001759}
1760
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001761/*
Mauro Carvalho Chehab87d1d272009-06-22 22:48:29 -03001762 * i7core_check_error Retrieve and process errors reported by the
1763 * hardware. Called by the Core module.
1764 */
1765static void i7core_check_error(struct mem_ctl_info *mci)
1766{
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001767 struct i7core_pvt *pvt = mci->pvt_info;
1768 int i;
1769 unsigned count = 0;
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001770 struct mce *m;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001771
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001772 /*
1773 * MCE first step: Copy all mce errors into a temporary buffer
1774 * We use a double buffering here, to reduce the risk of
Lucas De Marchi25985ed2011-03-30 22:57:33 -03001775 * losing an error.
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001776 */
1777 smp_rmb();
Mauro Carvalho Chehab321ece42009-10-08 13:11:08 -03001778 count = (pvt->mce_out + MCE_LOG_LEN - pvt->mce_in)
1779 % MCE_LOG_LEN;
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001780 if (!count)
Vernon Mauery8a311e12010-04-16 19:40:19 -03001781 goto check_ce_error;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001782
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001783 m = pvt->mce_outentry;
Mauro Carvalho Chehab321ece42009-10-08 13:11:08 -03001784 if (pvt->mce_in + count > MCE_LOG_LEN) {
1785 unsigned l = MCE_LOG_LEN - pvt->mce_in;
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001786
1787 memcpy(m, &pvt->mce_entry[pvt->mce_in], sizeof(*m) * l);
1788 smp_wmb();
1789 pvt->mce_in = 0;
1790 count -= l;
1791 m += l;
1792 }
1793 memcpy(m, &pvt->mce_entry[pvt->mce_in], sizeof(*m) * count);
1794 smp_wmb();
1795 pvt->mce_in += count;
1796
1797 smp_rmb();
1798 if (pvt->mce_overrun) {
1799 i7core_printk(KERN_ERR, "Lost %d memory errors\n",
1800 pvt->mce_overrun);
1801 smp_wmb();
1802 pvt->mce_overrun = 0;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001803 }
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001804
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001805 /*
1806 * MCE second step: parse errors and display
1807 */
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001808 for (i = 0; i < count; i++)
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001809 i7core_mce_output_error(mci, &pvt->mce_outentry[i]);
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001810
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001811 /*
1812 * Now, let's increment CE error counts
1813 */
Vernon Mauery8a311e12010-04-16 19:40:19 -03001814check_ce_error:
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001815 if (!pvt->is_registered)
1816 i7core_udimm_check_mc_ecc_err(mci);
1817 else
1818 i7core_rdimm_check_mc_ecc_err(mci);
Mauro Carvalho Chehab87d1d272009-06-22 22:48:29 -03001819}
1820
1821/*
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001822 * i7core_mce_check_error Replicates mcelog routine to get errors
1823 * This routine simply queues mcelog errors, and
1824 * return. The error itself should be handled later
1825 * by i7core_check_error.
Mauro Carvalho Chehab6e103be2009-10-05 09:40:09 -03001826 * WARNING: As this routine should be called at NMI time, extra care should
1827 * be taken to avoid deadlocks, and to be as fast as possible.
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001828 */
1829static int i7core_mce_check_error(void *priv, struct mce *mce)
1830{
Mauro Carvalho Chehabc5d34522009-07-17 10:28:15 -03001831 struct mem_ctl_info *mci = priv;
1832 struct i7core_pvt *pvt = mci->pvt_info;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001833
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001834 /*
1835 * Just let mcelog handle it if the error is
1836 * outside the memory controller
1837 */
1838 if (((mce->status & 0xffff) >> 7) != 1)
1839 return 0;
1840
Mauro Carvalho Chehabf237fcf2009-07-15 19:53:24 -03001841 /* Bank 8 registers are the only ones that we know how to handle */
1842 if (mce->bank != 8)
1843 return 0;
1844
Randy Dunlap3b918c12009-11-08 01:36:40 -02001845#ifdef CONFIG_SMP
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001846 /* Only handle if it is the right mc controller */
Mauro Carvalho Chehab6e103be2009-10-05 09:40:09 -03001847 if (cpu_data(mce->cpu).phys_proc_id != pvt->i7core_dev->socket)
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001848 return 0;
Randy Dunlap3b918c12009-11-08 01:36:40 -02001849#endif
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001850
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001851 smp_rmb();
Mauro Carvalho Chehab321ece42009-10-08 13:11:08 -03001852 if ((pvt->mce_out + 1) % MCE_LOG_LEN == pvt->mce_in) {
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001853 smp_wmb();
1854 pvt->mce_overrun++;
1855 return 0;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001856 }
Mauro Carvalho Chehab6e103be2009-10-05 09:40:09 -03001857
1858 /* Copy memory error at the ringbuffer */
1859 memcpy(&pvt->mce_entry[pvt->mce_out], mce, sizeof(*mce));
Mauro Carvalho Chehabca9c90b2009-10-04 10:15:40 -03001860 smp_wmb();
Mauro Carvalho Chehab321ece42009-10-08 13:11:08 -03001861 pvt->mce_out = (pvt->mce_out + 1) % MCE_LOG_LEN;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001862
Mauro Carvalho Chehabc5d34522009-07-17 10:28:15 -03001863 /* Handle fatal errors immediately */
1864 if (mce->mcgstatus & 1)
1865 i7core_check_error(mci);
1866
David Sterbae7bf0682010-12-27 16:51:15 +01001867 /* Advise mcelog that the errors were handled */
Mauro Carvalho Chehab8a2f1182009-07-15 19:01:08 -03001868 return 1;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03001869}
1870
Hidetoshi Setoa3aa0a42010-08-20 04:25:18 -03001871static void i7core_pci_ctl_create(struct i7core_pvt *pvt)
1872{
1873 pvt->i7core_pci = edac_pci_create_generic_ctl(
1874 &pvt->i7core_dev->pdev[0]->dev,
1875 EDAC_MOD_STR);
1876 if (unlikely(!pvt->i7core_pci))
1877 pr_warn("Unable to setup PCI error report via EDAC\n");
1878}
1879
1880static void i7core_pci_ctl_release(struct i7core_pvt *pvt)
1881{
1882 if (likely(pvt->i7core_pci))
1883 edac_pci_release_generic_ctl(pvt->i7core_pci);
1884 else
1885 i7core_printk(KERN_ERR,
1886 "Couldn't find mem_ctl_info for socket %d\n",
1887 pvt->i7core_dev->socket);
1888 pvt->i7core_pci = NULL;
1889}
1890
Hidetoshi Seto1c6edbb2010-08-20 04:32:33 -03001891static void i7core_unregister_mci(struct i7core_dev *i7core_dev)
1892{
1893 struct mem_ctl_info *mci = i7core_dev->mci;
1894 struct i7core_pvt *pvt;
1895
1896 if (unlikely(!mci || !mci->pvt_info)) {
1897 debugf0("MC: " __FILE__ ": %s(): dev = %p\n",
1898 __func__, &i7core_dev->pdev[0]->dev);
1899
1900 i7core_printk(KERN_ERR, "Couldn't find mci handler\n");
1901 return;
1902 }
1903
1904 pvt = mci->pvt_info;
1905
1906 debugf0("MC: " __FILE__ ": %s(): mci = %p, dev = %p\n",
1907 __func__, mci, &i7core_dev->pdev[0]->dev);
1908
1909 /* Disable MCE NMI handler */
1910 edac_mce_unregister(&pvt->edac_mce);
1911
1912 /* Disable EDAC polling */
1913 i7core_pci_ctl_release(pvt);
1914
1915 /* Remove MC sysfs nodes */
1916 edac_mc_del_mc(mci->dev);
1917
1918 debugf1("%s: free mci struct\n", mci->ctl_name);
1919 kfree(mci->ctl_name);
1920 edac_mc_free(mci);
1921 i7core_dev->mci = NULL;
1922}
1923
Hidetoshi Setoaace4282010-08-20 04:32:45 -03001924static int i7core_register_mci(struct i7core_dev *i7core_dev)
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001925{
1926 struct mem_ctl_info *mci;
1927 struct i7core_pvt *pvt;
Hidetoshi Setoaace4282010-08-20 04:32:45 -03001928 int rc, channels, csrows;
1929
1930 /* Check the number of active and not disabled channels */
1931 rc = i7core_get_active_channels(i7core_dev->socket, &channels, &csrows);
1932 if (unlikely(rc < 0))
1933 return rc;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001934
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001935 /* allocate a new MC control structure */
Hidetoshi Setoaace4282010-08-20 04:32:45 -03001936 mci = edac_mc_alloc(sizeof(*pvt), csrows, channels, i7core_dev->socket);
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001937 if (unlikely(!mci))
1938 return -ENOMEM;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001939
Mauro Carvalho Chehab3cfd0142010-08-10 23:23:46 -03001940 debugf0("MC: " __FILE__ ": %s(): mci = %p, dev = %p\n",
1941 __func__, mci, &i7core_dev->pdev[0]->dev);
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001942
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001943 pvt = mci->pvt_info;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001944 memset(pvt, 0, sizeof(*pvt));
Mauro Carvalho Chehab67166af2009-07-15 06:56:23 -03001945
Mauro Carvalho Chehab6d37d242010-08-20 12:48:26 -03001946 /* Associates i7core_dev and mci for future usage */
1947 pvt->i7core_dev = i7core_dev;
1948 i7core_dev->mci = mci;
1949
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -03001950 /*
1951 * FIXME: how to handle RDDR3 at MCI level? It is possible to have
1952 * Mixed RDDR3/UDDR3 with Nehalem, provided that they are on different
1953 * memory channels
1954 */
1955 mci->mtype_cap = MEM_FLAG_DDR3;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001956 mci->edac_ctl_cap = EDAC_FLAG_NONE;
1957 mci->edac_cap = EDAC_FLAG_NONE;
1958 mci->mod_name = "i7core_edac.c";
1959 mci->mod_ver = I7CORE_REVISION;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001960 mci->ctl_name = kasprintf(GFP_KERNEL, "i7 core #%d",
1961 i7core_dev->socket);
1962 mci->dev_name = pci_name(i7core_dev->pdev[0]);
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001963 mci->ctl_page_to_phys = NULL;
Mauro Carvalho Chehab1288c182010-08-10 18:57:01 -03001964
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001965 /* Store pci devices at mci for faster access */
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03001966 rc = mci_bind_devs(mci, i7core_dev);
Mauro Carvalho Chehab41fcb7f2009-06-22 22:48:31 -03001967 if (unlikely(rc < 0))
Hidetoshi Seto628c5dd2010-08-20 04:28:40 -03001968 goto fail0;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001969
Hidetoshi Seto59398132010-08-20 04:28:25 -03001970 if (pvt->is_registered)
1971 mci->mc_driver_sysfs_attributes = i7core_sysfs_rdimm_attrs;
1972 else
1973 mci->mc_driver_sysfs_attributes = i7core_sysfs_udimm_attrs;
1974
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001975 /* Get dimm basic config */
Hidetoshi Seto2e5185f2010-08-20 04:32:56 -03001976 get_dimm_config(mci);
Hidetoshi Seto59398132010-08-20 04:28:25 -03001977 /* record ptr to the generic device */
1978 mci->dev = &i7core_dev->pdev[0]->dev;
1979 /* Set the function pointer to an actual operation function */
1980 mci->edac_check = i7core_check_error;
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001981
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001982 /* add this new MC control structure to EDAC's list of MCs */
Mauro Carvalho Chehabb7c76152009-06-22 22:48:30 -03001983 if (unlikely(edac_mc_add_mc(mci))) {
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001984 debugf0("MC: " __FILE__
1985 ": %s(): failed edac_mc_add_mc()\n", __func__);
1986 /* FIXME: perhaps some code should go here that disables error
1987 * reporting if we just enabled it
1988 */
Mauro Carvalho Chehabb7c76152009-06-22 22:48:30 -03001989
1990 rc = -EINVAL;
Hidetoshi Seto628c5dd2010-08-20 04:28:40 -03001991 goto fail0;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03001992 }
1993
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001994 /* Default error mask is any memory */
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03001995 pvt->inject.channel = 0;
Mauro Carvalho Chehab194a40f2009-06-22 22:48:28 -03001996 pvt->inject.dimm = -1;
1997 pvt->inject.rank = -1;
1998 pvt->inject.bank = -1;
1999 pvt->inject.page = -1;
2000 pvt->inject.col = -1;
2001
Hidetoshi Setoa3aa0a42010-08-20 04:25:18 -03002002 /* allocating generic PCI control info */
2003 i7core_pci_ctl_create(pvt);
2004
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03002005 /* Registers on edac_mce in order to receive memory errors */
Mauro Carvalho Chehabc5d34522009-07-17 10:28:15 -03002006 pvt->edac_mce.priv = mci;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03002007 pvt->edac_mce.check_error = i7core_mce_check_error;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03002008 rc = edac_mce_register(&pvt->edac_mce);
Mauro Carvalho Chehabb9905382009-08-05 21:36:35 -03002009 if (unlikely(rc < 0)) {
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03002010 debugf0("MC: " __FILE__
2011 ": %s(): failed edac_mce_register()\n", __func__);
Hidetoshi Seto628c5dd2010-08-20 04:28:40 -03002012 goto fail1;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03002013 }
2014
Hidetoshi Seto628c5dd2010-08-20 04:28:40 -03002015 return 0;
2016
2017fail1:
2018 i7core_pci_ctl_release(pvt);
2019 edac_mc_del_mc(mci->dev);
2020fail0:
2021 kfree(mci->ctl_name);
2022 edac_mc_free(mci);
Hidetoshi Seto1c6edbb2010-08-20 04:32:33 -03002023 i7core_dev->mci = NULL;
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03002024 return rc;
2025}
2026
2027/*
2028 * i7core_probe Probe for ONE instance of device to see if it is
2029 * present.
2030 * return:
2031 * 0 for FOUND a device
2032 * < 0 for error code
2033 */
Mauro Carvalho Chehab2d95d812010-06-30 01:42:21 -03002034
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03002035static int __devinit i7core_probe(struct pci_dev *pdev,
2036 const struct pci_device_id *id)
2037{
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03002038 int rc;
2039 struct i7core_dev *i7core_dev;
2040
Mauro Carvalho Chehab2d95d812010-06-30 01:42:21 -03002041 /* get the pci devices we want to reserve for our use */
2042 mutex_lock(&i7core_edac_lock);
2043
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03002044 /*
Mauro Carvalho Chehabd4c27792009-09-05 04:12:02 -03002045 * All memory controllers are allocated at the first pass.
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03002046 */
Mauro Carvalho Chehab2d95d812010-06-30 01:42:21 -03002047 if (unlikely(probed >= 1)) {
2048 mutex_unlock(&i7core_edac_lock);
Mauro Carvalho Chehab76a7bd82010-10-24 11:36:19 -02002049 return -ENODEV;
Mauro Carvalho Chehab2d95d812010-06-30 01:42:21 -03002050 }
2051 probed++;
Mauro Carvalho Chehabde06eee2009-10-14 08:02:40 -03002052
Hidetoshi Seto64c10f62010-08-20 04:28:14 -03002053 rc = i7core_get_all_devices();
Mauro Carvalho Chehabf4742942009-09-05 02:35:08 -03002054 if (unlikely(rc < 0))
2055 goto fail0;
2056
2057 list_for_each_entry(i7core_dev, &i7core_edac_list, list) {
Hidetoshi Setoaace4282010-08-20 04:32:45 -03002058 rc = i7core_register_mci(i7core_dev);
Mauro Carvalho Chehabd4c27792009-09-05 04:12:02 -03002059 if (unlikely(rc < 0))
2060 goto fail1;
Mauro Carvalho Chehabd5381642009-07-09 22:06:41 -03002061 }
2062
Mauro Carvalho Chehabef708b52009-06-22 22:48:30 -03002063 i7core_printk(KERN_INFO, "Driver loaded.\n");
Mauro Carvalho Chehab8f331902009-06-22 22:48:29 -03002064
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03002065 mutex_unlock(&i7core_edac_lock);
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03002066 return 0;
2067
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03002068fail1:
Mauro Carvalho Chehab88ef5ea2010-08-20 15:39:38 -03002069 list_for_each_entry(i7core_dev, &i7core_edac_list, list)
2070 i7core_unregister_mci(i7core_dev);
2071
Mauro Carvalho Chehab13d6e9b2009-09-05 12:15:20 -03002072 i7core_put_all_devices();
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03002073fail0:
2074 mutex_unlock(&i7core_edac_lock);
Mauro Carvalho Chehabb7c76152009-06-22 22:48:30 -03002075 return rc;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03002076}
2077
2078/*
2079 * i7core_remove destructor for one instance of device
2080 *
2081 */
2082static void __devexit i7core_remove(struct pci_dev *pdev)
2083{
Hidetoshi Seto64c10f62010-08-20 04:28:14 -03002084 struct i7core_dev *i7core_dev;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03002085
2086 debugf0(__FILE__ ": %s()\n", __func__);
2087
Mauro Carvalho Chehab22e6bcb2009-09-05 23:06:50 -03002088 /*
2089 * we have a trouble here: pdev value for removal will be wrong, since
2090 * it will point to the X58 register used to detect that the machine
2091 * is a Nehalem or upper design. However, due to the way several PCI
2092 * devices are grouped together to provide MC functionality, we need
2093 * to use a different method for releasing the devices
2094 */
Mauro Carvalho Chehab87d1d272009-06-22 22:48:29 -03002095
Mauro Carvalho Chehab66607702009-09-05 00:52:11 -03002096 mutex_lock(&i7core_edac_lock);
Hidetoshi Seto71fe0172010-08-20 04:29:47 -03002097
2098 if (unlikely(!probed)) {
2099 mutex_unlock(&i7core_edac_lock);
2100 return;
2101 }
2102
Mauro Carvalho Chehab88ef5ea2010-08-20 15:39:38 -03002103 list_for_each_entry(i7core_dev, &i7core_edac_list, list)
2104 i7core_unregister_mci(i7core_dev);
Hidetoshi Seto64c10f62010-08-20 04:28:14 -03002105
2106 /* Release PCI resources */
2107 i7core_put_all_devices();
2108
Mauro Carvalho Chehab2d95d812010-06-30 01:42:21 -03002109 probed--;
2110
Mauro Carvalho Chehab22e6bcb2009-09-05 23:06:50 -03002111 mutex_unlock(&i7core_edac_lock);
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03002112}
2113
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03002114MODULE_DEVICE_TABLE(pci, i7core_pci_tbl);
2115
2116/*
2117 * i7core_driver pci_driver structure for this module
2118 *
2119 */
2120static struct pci_driver i7core_driver = {
2121 .name = "i7core_edac",
2122 .probe = i7core_probe,
2123 .remove = __devexit_p(i7core_remove),
2124 .id_table = i7core_pci_tbl,
2125};
2126
2127/*
2128 * i7core_init Module entry function
2129 * Try to initialize this module for its devices
2130 */
2131static int __init i7core_init(void)
2132{
2133 int pci_rc;
2134
2135 debugf2("MC: " __FILE__ ": %s()\n", __func__);
2136
2137 /* Ensure that the OPSTATE is set correctly for POLL or NMI */
2138 opstate_init();
2139
Mauro Carvalho Chehab54a08ab2010-08-19 15:51:00 -03002140 if (use_pci_fixup)
2141 i7core_xeon_pci_fixup(pci_dev_table);
Keith Manntheybc2d7242009-09-03 00:05:05 -03002142
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03002143 pci_rc = pci_register_driver(&i7core_driver);
2144
Mauro Carvalho Chehab3ef288a2009-09-02 23:43:33 -03002145 if (pci_rc >= 0)
2146 return 0;
2147
2148 i7core_printk(KERN_ERR, "Failed to register device with error %d.\n",
2149 pci_rc);
2150
2151 return pci_rc;
Mauro Carvalho Chehaba0c36a12009-06-22 22:41:15 -03002152}
2153
2154/*
2155 * i7core_exit() Module exit function
2156 * Unregister the driver
2157 */
2158static void __exit i7core_exit(void)
2159{
2160 debugf2("MC: " __FILE__ ": %s()\n", __func__);
2161 pci_unregister_driver(&i7core_driver);
2162}
2163
2164module_init(i7core_init);
2165module_exit(i7core_exit);
2166
2167MODULE_LICENSE("GPL");
2168MODULE_AUTHOR("Mauro Carvalho Chehab <mchehab@redhat.com>");
2169MODULE_AUTHOR("Red Hat Inc. (http://www.redhat.com)");
2170MODULE_DESCRIPTION("MC Driver for Intel i7 Core memory controllers - "
2171 I7CORE_REVISION);
2172
2173module_param(edac_op_state, int, 0444);
2174MODULE_PARM_DESC(edac_op_state, "EDAC Error Reporting state: 0=Poll,1=NMI");