2009-06-26 01:32:38 +08:00
|
|
|
#include <linux/module.h>
|
2010-08-18 21:11:35 +08:00
|
|
|
#include <linux/slab.h>
|
|
|
|
|
2017-07-25 17:07:03 +08:00
|
|
|
#include <asm/cpu.h>
|
|
|
|
|
2010-09-27 21:30:39 +08:00
|
|
|
#include "mce_amd.h"
|
2009-05-06 23:57:20 +08:00
|
|
|
|
2010-08-18 21:11:35 +08:00
|
|
|
static struct amd_decoder_ops *fam_ops;
|
|
|
|
|
2010-09-18 01:11:47 +08:00
|
|
|
static u8 xec_mask = 0xf;
|
2010-09-01 00:28:08 +08:00
|
|
|
|
2009-07-24 19:51:42 +08:00
|
|
|
static bool report_gart_errors;
|
2016-11-18 06:57:29 +08:00
|
|
|
static void (*decode_dram_ecc)(int node_id, struct mce *m);
|
2009-07-24 19:51:42 +08:00
|
|
|
|
|
|
|
void amd_report_gart_errors(bool v)
|
|
|
|
{
|
|
|
|
report_gart_errors = v;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(amd_report_gart_errors);
|
|
|
|
|
2011-08-25 00:44:22 +08:00
|
|
|
void amd_register_ecc_decoder(void (*f)(int, struct mce *))
|
2009-07-24 19:51:42 +08:00
|
|
|
{
|
2016-11-18 06:57:29 +08:00
|
|
|
decode_dram_ecc = f;
|
2009-07-24 19:51:42 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(amd_register_ecc_decoder);
|
|
|
|
|
2011-08-25 00:44:22 +08:00
|
|
|
void amd_unregister_ecc_decoder(void (*f)(int, struct mce *))
|
2009-07-24 19:51:42 +08:00
|
|
|
{
|
2016-11-18 06:57:29 +08:00
|
|
|
if (decode_dram_ecc) {
|
|
|
|
WARN_ON(decode_dram_ecc != f);
|
2009-07-24 19:51:42 +08:00
|
|
|
|
2016-11-18 06:57:29 +08:00
|
|
|
decode_dram_ecc = NULL;
|
2009-07-24 19:51:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(amd_unregister_ecc_decoder);
|
|
|
|
|
2009-05-06 23:57:20 +08:00
|
|
|
/*
|
|
|
|
* string representation for the different MCA reported error types, see F3x48
|
|
|
|
* or MSR0000_0411.
|
|
|
|
*/
|
2010-09-07 00:13:39 +08:00
|
|
|
|
|
|
|
/* transaction type */
|
2012-12-23 19:40:45 +08:00
|
|
|
static const char * const tt_msgs[] = { "INSN", "DATA", "GEN", "RESV" };
|
2009-05-06 23:57:20 +08:00
|
|
|
|
2010-09-07 00:13:39 +08:00
|
|
|
/* cache level */
|
2012-12-23 19:40:45 +08:00
|
|
|
static const char * const ll_msgs[] = { "RESV", "L1", "L2", "L3/GEN" };
|
2009-05-06 23:57:20 +08:00
|
|
|
|
2010-09-07 00:13:39 +08:00
|
|
|
/* memory transaction type */
|
2012-12-23 19:40:45 +08:00
|
|
|
static const char * const rrrr_msgs[] = {
|
2010-09-07 00:13:39 +08:00
|
|
|
"GEN", "RD", "WR", "DRD", "DWR", "IRD", "PRF", "EV", "SNP"
|
2009-05-06 23:57:20 +08:00
|
|
|
};
|
|
|
|
|
2010-09-07 00:13:39 +08:00
|
|
|
/* participating processor */
|
2011-11-30 02:03:25 +08:00
|
|
|
const char * const pp_msgs[] = { "SRC", "RES", "OBS", "GEN" };
|
2009-06-26 01:32:38 +08:00
|
|
|
EXPORT_SYMBOL_GPL(pp_msgs);
|
2009-05-06 23:57:20 +08:00
|
|
|
|
2010-09-07 00:13:39 +08:00
|
|
|
/* request timeout */
|
2012-12-23 19:40:45 +08:00
|
|
|
static const char * const to_msgs[] = { "no timeout", "timed out" };
|
2009-05-06 23:57:20 +08:00
|
|
|
|
2010-09-07 00:13:39 +08:00
|
|
|
/* memory or i/o */
|
2012-12-23 19:40:45 +08:00
|
|
|
static const char * const ii_msgs[] = { "MEM", "RESV", "IO", "GEN" };
|
2009-05-06 23:57:20 +08:00
|
|
|
|
2012-12-19 05:06:11 +08:00
|
|
|
/* internal error type */
|
2012-12-23 19:40:45 +08:00
|
|
|
static const char * const uu_msgs[] = { "RESV", "RESV", "HWA", "RESV" };
|
2012-12-19 05:06:11 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static const char * const f15h_mc1_mce_desc[] = {
|
2010-11-08 22:03:35 +08:00
|
|
|
"UC during a demand linefill from L2",
|
|
|
|
"Parity error during data load from IC",
|
|
|
|
"Parity error for IC valid bit",
|
|
|
|
"Main tag parity error",
|
|
|
|
"Parity error in prediction queue",
|
|
|
|
"PFB data/address parity error",
|
|
|
|
"Parity error in the branch status reg",
|
|
|
|
"PFB promotion address error",
|
|
|
|
"Tag error during probe/victimization",
|
|
|
|
"Parity error for IC probe tag valid bit",
|
|
|
|
"PFB non-cacheable bit parity error",
|
|
|
|
"PFB valid bit parity error", /* xec = 0xd */
|
2011-11-22 02:45:34 +08:00
|
|
|
"Microcode Patch Buffer", /* xec = 010 */
|
2010-11-08 22:03:35 +08:00
|
|
|
"uop queue",
|
|
|
|
"insn buffer",
|
|
|
|
"predecode buffer",
|
2014-07-14 22:58:19 +08:00
|
|
|
"fetch address FIFO",
|
|
|
|
"dispatch uop queue"
|
2010-11-08 22:03:35 +08:00
|
|
|
};
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static const char * const f15h_mc2_mce_desc[] = {
|
2010-09-22 02:45:10 +08:00
|
|
|
"Fill ECC error on data fills", /* xec = 0x4 */
|
|
|
|
"Fill parity error on insn fills",
|
|
|
|
"Prefetcher request FIFO parity error",
|
|
|
|
"PRQ address parity error",
|
|
|
|
"PRQ data parity error",
|
|
|
|
"WCC Tag ECC error",
|
|
|
|
"WCC Data ECC error",
|
|
|
|
"WCB Data parity error",
|
2011-11-23 21:50:44 +08:00
|
|
|
"VB Data ECC or parity error",
|
2010-09-22 02:45:10 +08:00
|
|
|
"L2 Tag ECC error", /* xec = 0x10 */
|
|
|
|
"Hard L2 Tag ECC error",
|
|
|
|
"Multiple hits on L2 tag",
|
|
|
|
"XAB parity error",
|
|
|
|
"PRB address parity error"
|
|
|
|
};
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static const char * const mc4_mce_desc[] = {
|
2011-11-25 04:29:57 +08:00
|
|
|
"DRAM ECC error detected on the NB",
|
|
|
|
"CRC error detected on HT link",
|
|
|
|
"Link-defined sync error packets detected on HT link",
|
|
|
|
"HT Master abort",
|
|
|
|
"HT Target abort",
|
|
|
|
"Invalid GART PTE entry during GART table walk",
|
|
|
|
"Unsupported atomic RMW received from an IO link",
|
|
|
|
"Watchdog timeout due to lack of progress",
|
|
|
|
"DRAM ECC error detected on the NB",
|
|
|
|
"SVM DMA Exclusion Vector error",
|
|
|
|
"HT data error detected on link",
|
|
|
|
"Protocol error (link, L3, probe filter)",
|
|
|
|
"NB internal arrays parity error",
|
|
|
|
"DRAM addr/ctl signals parity error",
|
|
|
|
"IO link transmission error",
|
|
|
|
"L3 data cache ECC error", /* xec = 0x1c */
|
|
|
|
"L3 cache tag error",
|
|
|
|
"L3 LRU parity bits error",
|
|
|
|
"ECC Error in the Probe Filter directory"
|
|
|
|
};
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static const char * const mc5_mce_desc[] = {
|
2010-09-22 21:28:59 +08:00
|
|
|
"CPU Watchdog timer expire",
|
|
|
|
"Wakeup array dest tag",
|
|
|
|
"AG payload array",
|
|
|
|
"EX payload array",
|
|
|
|
"IDRF array",
|
|
|
|
"Retire dispatch queue",
|
|
|
|
"Mapper checkpoint array",
|
|
|
|
"Physical register file EX0 port",
|
|
|
|
"Physical register file EX1 port",
|
|
|
|
"Physical register file AG0 port",
|
|
|
|
"Physical register file AG1 port",
|
|
|
|
"Flag register file",
|
2013-06-06 04:50:03 +08:00
|
|
|
"DE error occurred",
|
|
|
|
"Retire status queue"
|
2010-09-22 21:28:59 +08:00
|
|
|
};
|
|
|
|
|
2014-11-05 01:41:08 +08:00
|
|
|
static const char * const mc6_mce_desc[] = {
|
|
|
|
"Hardware Assertion",
|
|
|
|
"Free List",
|
|
|
|
"Physical Register File",
|
|
|
|
"Retire Queue",
|
|
|
|
"Scheduler table",
|
|
|
|
"Status Register File",
|
|
|
|
};
|
|
|
|
|
2016-03-07 21:02:18 +08:00
|
|
|
/* Scalable MCA error strings */
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_ls_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"Load queue parity error",
|
|
|
|
"Store queue parity error",
|
|
|
|
"Miss address buffer payload parity error",
|
|
|
|
"Level 1 TLB parity error",
|
2019-02-02 06:55:52 +08:00
|
|
|
"DC Tag error type 5",
|
2019-02-02 06:55:53 +08:00
|
|
|
"DC Tag error type 6",
|
|
|
|
"DC Tag error type 1",
|
2016-03-07 21:02:18 +08:00
|
|
|
"Internal error type 1",
|
|
|
|
"Internal error type 2",
|
2019-02-02 06:55:53 +08:00
|
|
|
"System Read Data Error Thread 0",
|
|
|
|
"System Read Data Error Thread 1",
|
|
|
|
"DC Tag error type 2",
|
|
|
|
"DC Data error type 1 and poison consumption",
|
|
|
|
"DC Data error type 2",
|
|
|
|
"DC Data error type 3",
|
|
|
|
"DC Tag error type 4",
|
|
|
|
"Level 2 TLB parity error",
|
2016-03-07 21:02:18 +08:00
|
|
|
"PDC parity error",
|
2019-02-02 06:55:53 +08:00
|
|
|
"DC Tag error type 3",
|
|
|
|
"DC Tag error type 5",
|
|
|
|
"L2 Fill Data error",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_if_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"Op Cache Microtag Probe Port Parity Error",
|
|
|
|
"IC Microtag or Full Tag Multi-hit Error",
|
|
|
|
"IC Full Tag Parity Error",
|
|
|
|
"IC Data Array Parity Error",
|
|
|
|
"Decoupling Queue PhysAddr Parity Error",
|
|
|
|
"L0 ITLB Parity Error",
|
|
|
|
"L1 ITLB Parity Error",
|
|
|
|
"L2 ITLB Parity Error",
|
|
|
|
"BPQ Thread 0 Snoop Parity Error",
|
|
|
|
"BPQ Thread 1 Snoop Parity Error",
|
|
|
|
"L1 BTB Multi-Match Error",
|
|
|
|
"L2 BTB Multi-Match Error",
|
|
|
|
"L2 Cache Response Poison Error",
|
|
|
|
"System Read Data Error",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_l2_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"L2M Tag Multiple-Way-Hit error",
|
|
|
|
"L2M Tag or State Array ECC Error",
|
|
|
|
"L2M Data Array ECC Error",
|
|
|
|
"Hardware Assert Error",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_de_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"Micro-op cache tag parity error",
|
|
|
|
"Micro-op cache data parity error",
|
|
|
|
"Instruction buffer parity error",
|
|
|
|
"Micro-op queue parity error",
|
|
|
|
"Instruction dispatch queue parity error",
|
|
|
|
"Fetch address FIFO parity error",
|
|
|
|
"Patch RAM data parity error",
|
|
|
|
"Patch RAM sequencer parity error",
|
|
|
|
"Micro-op buffer parity error"
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_ex_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"Watchdog Timeout error",
|
|
|
|
"Physical register file parity error",
|
|
|
|
"Flag register file parity error",
|
|
|
|
"Immediate displacement register file parity error",
|
|
|
|
"Address generator payload parity error",
|
|
|
|
"EX payload parity error",
|
|
|
|
"Checkpoint queue parity error",
|
|
|
|
"Retire dispatch queue parity error",
|
2016-09-12 15:59:32 +08:00
|
|
|
"Retire status queue parity error",
|
|
|
|
"Scheduling queue parity error",
|
|
|
|
"Branch buffer queue parity error",
|
2019-02-02 06:55:52 +08:00
|
|
|
"Hardware Assertion error",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_fp_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"Physical register file (PRF) parity error",
|
|
|
|
"Freelist (FL) parity error",
|
|
|
|
"Schedule queue parity error",
|
2016-03-07 21:02:18 +08:00
|
|
|
"NSQ parity error",
|
2019-02-02 06:55:53 +08:00
|
|
|
"Retire queue (RQ) parity error",
|
|
|
|
"Status register file (SRF) parity error",
|
2016-09-12 15:59:32 +08:00
|
|
|
"Hardware assertion",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_l3_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"Shadow Tag Macro ECC Error",
|
|
|
|
"Shadow Tag Macro Multi-way-hit Error",
|
|
|
|
"L3M Tag ECC Error",
|
|
|
|
"L3M Tag Multi-way-hit Error",
|
|
|
|
"L3M Data ECC Error",
|
|
|
|
"SDP Parity Error or SystemReadDataError from XI",
|
|
|
|
"L3 Victim Queue Parity Error",
|
|
|
|
"L3 Hardware Assertion",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_cs_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"Illegal Request",
|
|
|
|
"Address Violation",
|
|
|
|
"Security Violation",
|
|
|
|
"Illegal Response",
|
|
|
|
"Unexpected Response",
|
|
|
|
"Request or Probe Parity Error",
|
|
|
|
"Read Response Parity Error",
|
|
|
|
"Atomic Request Parity Error",
|
|
|
|
"Probe Filter ECC Error",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
x86/MCE/AMD, EDAC/mce_amd: Add new McaTypes for CS, PSP, and SMU units
The existing CS, PSP, and SMU SMCA bank types will see new versions (as
indicated by their McaTypes) in future SMCA systems.
Add the new (HWID, MCATYPE) tuples for these new versions. Reuse the
same names as the older versions, since they are logically the same to
the user. SMCA systems won't mix and match IP blocks with different
McaType versions in the same system, so there isn't a need to
distinguish them. The MCA_IPID register is saved when logging an MCA
error, and that can be used to triage the error.
Also, add the new error descriptions to edac_mce_amd. Some error types
(positions in the list) are overloaded compared to the previous
McaTypes. Therefore, just create new lists of the error descriptions to
keep things simple even if some of the error descriptions are the same
between versions.
Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: Borislav Petkov <bp@suse.de>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Kees Cook <keescook@chromium.org>
Cc: linux-edac <linux-edac@vger.kernel.org>
Cc: Mauro Carvalho Chehab <mchehab@kernel.org>
Cc: Pu Wen <puwen@hygon.cn>
Cc: Qiuxu Zhuo <qiuxu.zhuo@intel.com>
Cc: Shirish S <Shirish.S@amd.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Tony Luck <tony.luck@intel.com>
Cc: Vishal Verma <vishal.l.verma@intel.com>
Cc: x86-ml <x86@kernel.org>
Link: https://lkml.kernel.org/r/20190201225534.8177-3-Yazen.Ghannam@amd.com
2019-02-02 06:55:52 +08:00
|
|
|
static const char * const smca_cs2_mce_desc[] = {
|
|
|
|
"Illegal Request",
|
|
|
|
"Address Violation",
|
|
|
|
"Security Violation",
|
|
|
|
"Illegal Response",
|
|
|
|
"Unexpected Response",
|
|
|
|
"Request or Probe Parity Error",
|
|
|
|
"Read Response Parity Error",
|
|
|
|
"Atomic Request Parity Error",
|
|
|
|
"SDP read response had no match in the CS queue",
|
|
|
|
"Probe Filter Protocol Error",
|
|
|
|
"Probe Filter ECC Error",
|
|
|
|
"SDP read response had an unexpected RETRY error",
|
|
|
|
"Counter overflow error",
|
|
|
|
"Counter underflow error",
|
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_pie_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"Hardware Assert",
|
|
|
|
"Register security violation",
|
|
|
|
"Link Error",
|
|
|
|
"Poison data consumption",
|
2019-02-02 06:55:52 +08:00
|
|
|
"A deferred error was detected in the DF"
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_umc_mce_desc[] = {
|
2016-03-07 21:02:18 +08:00
|
|
|
"DRAM ECC error",
|
2019-02-02 06:55:53 +08:00
|
|
|
"Data poison error",
|
2016-03-07 21:02:18 +08:00
|
|
|
"SDP parity error",
|
|
|
|
"Advanced peripheral bus error",
|
2019-02-02 06:55:53 +08:00
|
|
|
"Address/Command parity error",
|
2016-03-07 21:02:18 +08:00
|
|
|
"Write data CRC error",
|
2019-02-02 06:55:52 +08:00
|
|
|
"DCQ SRAM ECC error",
|
|
|
|
"AES SRAM ECC error",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_pb_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"An ECC error in the Parameter Block RAM array",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_psp_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"An ECC or parity error in a PSP RAM instance",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
x86/MCE/AMD, EDAC/mce_amd: Add new McaTypes for CS, PSP, and SMU units
The existing CS, PSP, and SMU SMCA bank types will see new versions (as
indicated by their McaTypes) in future SMCA systems.
Add the new (HWID, MCATYPE) tuples for these new versions. Reuse the
same names as the older versions, since they are logically the same to
the user. SMCA systems won't mix and match IP blocks with different
McaType versions in the same system, so there isn't a need to
distinguish them. The MCA_IPID register is saved when logging an MCA
error, and that can be used to triage the error.
Also, add the new error descriptions to edac_mce_amd. Some error types
(positions in the list) are overloaded compared to the previous
McaTypes. Therefore, just create new lists of the error descriptions to
keep things simple even if some of the error descriptions are the same
between versions.
Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: Borislav Petkov <bp@suse.de>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Kees Cook <keescook@chromium.org>
Cc: linux-edac <linux-edac@vger.kernel.org>
Cc: Mauro Carvalho Chehab <mchehab@kernel.org>
Cc: Pu Wen <puwen@hygon.cn>
Cc: Qiuxu Zhuo <qiuxu.zhuo@intel.com>
Cc: Shirish S <Shirish.S@amd.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Tony Luck <tony.luck@intel.com>
Cc: Vishal Verma <vishal.l.verma@intel.com>
Cc: x86-ml <x86@kernel.org>
Link: https://lkml.kernel.org/r/20190201225534.8177-3-Yazen.Ghannam@amd.com
2019-02-02 06:55:52 +08:00
|
|
|
static const char * const smca_psp2_mce_desc[] = {
|
|
|
|
"High SRAM ECC or parity error",
|
|
|
|
"Low SRAM ECC or parity error",
|
|
|
|
"Instruction Cache Bank 0 ECC or parity error",
|
|
|
|
"Instruction Cache Bank 1 ECC or parity error",
|
|
|
|
"Instruction Tag Ram 0 parity error",
|
|
|
|
"Instruction Tag Ram 1 parity error",
|
|
|
|
"Data Cache Bank 0 ECC or parity error",
|
|
|
|
"Data Cache Bank 1 ECC or parity error",
|
|
|
|
"Data Cache Bank 2 ECC or parity error",
|
|
|
|
"Data Cache Bank 3 ECC or parity error",
|
|
|
|
"Data Tag Bank 0 parity error",
|
|
|
|
"Data Tag Bank 1 parity error",
|
|
|
|
"Data Tag Bank 2 parity error",
|
|
|
|
"Data Tag Bank 3 parity error",
|
|
|
|
"Dirty Data Ram parity error",
|
|
|
|
"TLB Bank 0 parity error",
|
|
|
|
"TLB Bank 1 parity error",
|
|
|
|
"System Hub Read Buffer ECC or parity error",
|
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:33 +08:00
|
|
|
static const char * const smca_smu_mce_desc[] = {
|
2019-02-02 06:55:53 +08:00
|
|
|
"An ECC or parity error in an SMU RAM instance",
|
2016-03-07 21:02:18 +08:00
|
|
|
};
|
|
|
|
|
x86/MCE/AMD, EDAC/mce_amd: Add new McaTypes for CS, PSP, and SMU units
The existing CS, PSP, and SMU SMCA bank types will see new versions (as
indicated by their McaTypes) in future SMCA systems.
Add the new (HWID, MCATYPE) tuples for these new versions. Reuse the
same names as the older versions, since they are logically the same to
the user. SMCA systems won't mix and match IP blocks with different
McaType versions in the same system, so there isn't a need to
distinguish them. The MCA_IPID register is saved when logging an MCA
error, and that can be used to triage the error.
Also, add the new error descriptions to edac_mce_amd. Some error types
(positions in the list) are overloaded compared to the previous
McaTypes. Therefore, just create new lists of the error descriptions to
keep things simple even if some of the error descriptions are the same
between versions.
Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: Borislav Petkov <bp@suse.de>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Kees Cook <keescook@chromium.org>
Cc: linux-edac <linux-edac@vger.kernel.org>
Cc: Mauro Carvalho Chehab <mchehab@kernel.org>
Cc: Pu Wen <puwen@hygon.cn>
Cc: Qiuxu Zhuo <qiuxu.zhuo@intel.com>
Cc: Shirish S <Shirish.S@amd.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Tony Luck <tony.luck@intel.com>
Cc: Vishal Verma <vishal.l.verma@intel.com>
Cc: x86-ml <x86@kernel.org>
Link: https://lkml.kernel.org/r/20190201225534.8177-3-Yazen.Ghannam@amd.com
2019-02-02 06:55:52 +08:00
|
|
|
static const char * const smca_smu2_mce_desc[] = {
|
|
|
|
"High SRAM ECC or parity error",
|
|
|
|
"Low SRAM ECC or parity error",
|
|
|
|
"Data Cache Bank A ECC or parity error",
|
|
|
|
"Data Cache Bank B ECC or parity error",
|
|
|
|
"Data Tag Cache Bank A ECC or parity error",
|
|
|
|
"Data Tag Cache Bank B ECC or parity error",
|
|
|
|
"Instruction Cache Bank A ECC or parity error",
|
|
|
|
"Instruction Cache Bank B ECC or parity error",
|
|
|
|
"Instruction Tag Cache Bank A ECC or parity error",
|
|
|
|
"Instruction Tag Cache Bank B ECC or parity error",
|
|
|
|
"System Hub Read Buffer ECC or parity error",
|
|
|
|
};
|
|
|
|
|
2019-02-02 06:55:51 +08:00
|
|
|
static const char * const smca_mp5_mce_desc[] = {
|
|
|
|
"High SRAM ECC or parity error",
|
|
|
|
"Low SRAM ECC or parity error",
|
|
|
|
"Data Cache Bank A ECC or parity error",
|
|
|
|
"Data Cache Bank B ECC or parity error",
|
|
|
|
"Data Tag Cache Bank A ECC or parity error",
|
|
|
|
"Data Tag Cache Bank B ECC or parity error",
|
|
|
|
"Instruction Cache Bank A ECC or parity error",
|
|
|
|
"Instruction Cache Bank B ECC or parity error",
|
|
|
|
"Instruction Tag Cache Bank A ECC or parity error",
|
|
|
|
"Instruction Tag Cache Bank B ECC or parity error",
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char * const smca_nbio_mce_desc[] = {
|
|
|
|
"ECC or Parity error",
|
|
|
|
"PCIE error",
|
|
|
|
"SDP ErrEvent error",
|
|
|
|
"SDP Egress Poison Error",
|
|
|
|
"IOHC Internal Poison Error",
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char * const smca_pcie_mce_desc[] = {
|
|
|
|
"CCIX PER Message logging",
|
|
|
|
"CCIX Read Response with Status: Non-Data Error",
|
|
|
|
"CCIX Write Response with Status: Non-Data Error",
|
|
|
|
"CCIX Read Response with Status: Data Error",
|
|
|
|
"CCIX Non-okay write response with data error",
|
|
|
|
};
|
|
|
|
|
2016-09-12 15:59:34 +08:00
|
|
|
struct smca_mce_desc {
|
|
|
|
const char * const *descs;
|
|
|
|
unsigned int num_descs;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct smca_mce_desc smca_mce_descs[] = {
|
|
|
|
[SMCA_LS] = { smca_ls_mce_desc, ARRAY_SIZE(smca_ls_mce_desc) },
|
|
|
|
[SMCA_IF] = { smca_if_mce_desc, ARRAY_SIZE(smca_if_mce_desc) },
|
|
|
|
[SMCA_L2_CACHE] = { smca_l2_mce_desc, ARRAY_SIZE(smca_l2_mce_desc) },
|
|
|
|
[SMCA_DE] = { smca_de_mce_desc, ARRAY_SIZE(smca_de_mce_desc) },
|
|
|
|
[SMCA_EX] = { smca_ex_mce_desc, ARRAY_SIZE(smca_ex_mce_desc) },
|
|
|
|
[SMCA_FP] = { smca_fp_mce_desc, ARRAY_SIZE(smca_fp_mce_desc) },
|
|
|
|
[SMCA_L3_CACHE] = { smca_l3_mce_desc, ARRAY_SIZE(smca_l3_mce_desc) },
|
|
|
|
[SMCA_CS] = { smca_cs_mce_desc, ARRAY_SIZE(smca_cs_mce_desc) },
|
x86/MCE/AMD, EDAC/mce_amd: Add new McaTypes for CS, PSP, and SMU units
The existing CS, PSP, and SMU SMCA bank types will see new versions (as
indicated by their McaTypes) in future SMCA systems.
Add the new (HWID, MCATYPE) tuples for these new versions. Reuse the
same names as the older versions, since they are logically the same to
the user. SMCA systems won't mix and match IP blocks with different
McaType versions in the same system, so there isn't a need to
distinguish them. The MCA_IPID register is saved when logging an MCA
error, and that can be used to triage the error.
Also, add the new error descriptions to edac_mce_amd. Some error types
(positions in the list) are overloaded compared to the previous
McaTypes. Therefore, just create new lists of the error descriptions to
keep things simple even if some of the error descriptions are the same
between versions.
Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: Borislav Petkov <bp@suse.de>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Kees Cook <keescook@chromium.org>
Cc: linux-edac <linux-edac@vger.kernel.org>
Cc: Mauro Carvalho Chehab <mchehab@kernel.org>
Cc: Pu Wen <puwen@hygon.cn>
Cc: Qiuxu Zhuo <qiuxu.zhuo@intel.com>
Cc: Shirish S <Shirish.S@amd.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Tony Luck <tony.luck@intel.com>
Cc: Vishal Verma <vishal.l.verma@intel.com>
Cc: x86-ml <x86@kernel.org>
Link: https://lkml.kernel.org/r/20190201225534.8177-3-Yazen.Ghannam@amd.com
2019-02-02 06:55:52 +08:00
|
|
|
[SMCA_CS_V2] = { smca_cs2_mce_desc, ARRAY_SIZE(smca_cs2_mce_desc) },
|
2016-09-12 15:59:34 +08:00
|
|
|
[SMCA_PIE] = { smca_pie_mce_desc, ARRAY_SIZE(smca_pie_mce_desc) },
|
|
|
|
[SMCA_UMC] = { smca_umc_mce_desc, ARRAY_SIZE(smca_umc_mce_desc) },
|
|
|
|
[SMCA_PB] = { smca_pb_mce_desc, ARRAY_SIZE(smca_pb_mce_desc) },
|
|
|
|
[SMCA_PSP] = { smca_psp_mce_desc, ARRAY_SIZE(smca_psp_mce_desc) },
|
x86/MCE/AMD, EDAC/mce_amd: Add new McaTypes for CS, PSP, and SMU units
The existing CS, PSP, and SMU SMCA bank types will see new versions (as
indicated by their McaTypes) in future SMCA systems.
Add the new (HWID, MCATYPE) tuples for these new versions. Reuse the
same names as the older versions, since they are logically the same to
the user. SMCA systems won't mix and match IP blocks with different
McaType versions in the same system, so there isn't a need to
distinguish them. The MCA_IPID register is saved when logging an MCA
error, and that can be used to triage the error.
Also, add the new error descriptions to edac_mce_amd. Some error types
(positions in the list) are overloaded compared to the previous
McaTypes. Therefore, just create new lists of the error descriptions to
keep things simple even if some of the error descriptions are the same
between versions.
Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: Borislav Petkov <bp@suse.de>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Kees Cook <keescook@chromium.org>
Cc: linux-edac <linux-edac@vger.kernel.org>
Cc: Mauro Carvalho Chehab <mchehab@kernel.org>
Cc: Pu Wen <puwen@hygon.cn>
Cc: Qiuxu Zhuo <qiuxu.zhuo@intel.com>
Cc: Shirish S <Shirish.S@amd.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Tony Luck <tony.luck@intel.com>
Cc: Vishal Verma <vishal.l.verma@intel.com>
Cc: x86-ml <x86@kernel.org>
Link: https://lkml.kernel.org/r/20190201225534.8177-3-Yazen.Ghannam@amd.com
2019-02-02 06:55:52 +08:00
|
|
|
[SMCA_PSP_V2] = { smca_psp2_mce_desc, ARRAY_SIZE(smca_psp2_mce_desc) },
|
2016-09-12 15:59:34 +08:00
|
|
|
[SMCA_SMU] = { smca_smu_mce_desc, ARRAY_SIZE(smca_smu_mce_desc) },
|
x86/MCE/AMD, EDAC/mce_amd: Add new McaTypes for CS, PSP, and SMU units
The existing CS, PSP, and SMU SMCA bank types will see new versions (as
indicated by their McaTypes) in future SMCA systems.
Add the new (HWID, MCATYPE) tuples for these new versions. Reuse the
same names as the older versions, since they are logically the same to
the user. SMCA systems won't mix and match IP blocks with different
McaType versions in the same system, so there isn't a need to
distinguish them. The MCA_IPID register is saved when logging an MCA
error, and that can be used to triage the error.
Also, add the new error descriptions to edac_mce_amd. Some error types
(positions in the list) are overloaded compared to the previous
McaTypes. Therefore, just create new lists of the error descriptions to
keep things simple even if some of the error descriptions are the same
between versions.
Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: Borislav Petkov <bp@suse.de>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Kees Cook <keescook@chromium.org>
Cc: linux-edac <linux-edac@vger.kernel.org>
Cc: Mauro Carvalho Chehab <mchehab@kernel.org>
Cc: Pu Wen <puwen@hygon.cn>
Cc: Qiuxu Zhuo <qiuxu.zhuo@intel.com>
Cc: Shirish S <Shirish.S@amd.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Tony Luck <tony.luck@intel.com>
Cc: Vishal Verma <vishal.l.verma@intel.com>
Cc: x86-ml <x86@kernel.org>
Link: https://lkml.kernel.org/r/20190201225534.8177-3-Yazen.Ghannam@amd.com
2019-02-02 06:55:52 +08:00
|
|
|
[SMCA_SMU_V2] = { smca_smu2_mce_desc, ARRAY_SIZE(smca_smu2_mce_desc) },
|
2019-02-02 06:55:51 +08:00
|
|
|
[SMCA_MP5] = { smca_mp5_mce_desc, ARRAY_SIZE(smca_mp5_mce_desc) },
|
|
|
|
[SMCA_NBIO] = { smca_nbio_mce_desc, ARRAY_SIZE(smca_nbio_mce_desc) },
|
|
|
|
[SMCA_PCIE] = { smca_pcie_mce_desc, ARRAY_SIZE(smca_pcie_mce_desc) },
|
2016-09-12 15:59:34 +08:00
|
|
|
};
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static bool f12h_mc0_mce(u16 ec, u8 xec)
|
2009-07-28 19:50:43 +08:00
|
|
|
{
|
2010-08-18 21:11:35 +08:00
|
|
|
bool ret = false;
|
2009-07-28 19:50:43 +08:00
|
|
|
|
2010-08-18 21:11:35 +08:00
|
|
|
if (MEM_ERROR(ec)) {
|
2010-09-22 22:08:37 +08:00
|
|
|
u8 ll = LL(ec);
|
2010-08-18 21:11:35 +08:00
|
|
|
ret = true;
|
2009-07-28 19:50:43 +08:00
|
|
|
|
2010-08-18 21:11:35 +08:00
|
|
|
if (ll == LL_L2)
|
|
|
|
pr_cont("during L1 linefill from L2.\n");
|
|
|
|
else if (ll == LL_L1)
|
2010-09-22 22:08:37 +08:00
|
|
|
pr_cont("Data/Tag %s error.\n", R4_MSG(ec));
|
2010-08-18 21:11:35 +08:00
|
|
|
else
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2009-07-28 19:50:43 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static bool f10h_mc0_mce(u16 ec, u8 xec)
|
2010-09-16 21:08:14 +08:00
|
|
|
{
|
2010-09-22 22:08:37 +08:00
|
|
|
if (R4(ec) == R4_GEN && LL(ec) == LL_L1) {
|
2010-09-16 21:08:14 +08:00
|
|
|
pr_cont("during data scrub.\n");
|
|
|
|
return true;
|
|
|
|
}
|
2012-09-12 00:57:43 +08:00
|
|
|
return f12h_mc0_mce(ec, xec);
|
2010-09-16 21:08:14 +08:00
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static bool k8_mc0_mce(u16 ec, u8 xec)
|
2010-08-18 21:11:35 +08:00
|
|
|
{
|
|
|
|
if (BUS_ERROR(ec)) {
|
|
|
|
pr_cont("during system linefill.\n");
|
|
|
|
return true;
|
|
|
|
}
|
2009-07-28 19:50:43 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
return f10h_mc0_mce(ec, xec);
|
2010-08-18 21:11:35 +08:00
|
|
|
}
|
|
|
|
|
2012-12-19 05:06:11 +08:00
|
|
|
static bool cat_mc0_mce(u16 ec, u8 xec)
|
2010-08-18 21:11:35 +08:00
|
|
|
{
|
2010-09-22 22:08:37 +08:00
|
|
|
u8 r4 = R4(ec);
|
2010-08-18 21:11:35 +08:00
|
|
|
bool ret = true;
|
|
|
|
|
|
|
|
if (MEM_ERROR(ec)) {
|
|
|
|
|
2010-09-22 22:08:37 +08:00
|
|
|
if (TT(ec) != TT_DATA || LL(ec) != LL_L1)
|
2010-08-18 21:11:35 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (r4) {
|
|
|
|
case R4_DRD:
|
|
|
|
case R4_DWR:
|
|
|
|
pr_cont("Data/Tag parity error due to %s.\n",
|
|
|
|
(r4 == R4_DRD ? "load/hw prf" : "store"));
|
|
|
|
break;
|
|
|
|
case R4_EVICT:
|
|
|
|
pr_cont("Copyback parity error on a tag miss.\n");
|
|
|
|
break;
|
|
|
|
case R4_SNOOP:
|
|
|
|
pr_cont("Tag parity error during snoop.\n");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
} else if (BUS_ERROR(ec)) {
|
|
|
|
|
2010-09-22 22:08:37 +08:00
|
|
|
if ((II(ec) != II_MEM && II(ec) != II_IO) || LL(ec) != LL_LG)
|
2010-08-18 21:11:35 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
pr_cont("System read data error on a ");
|
|
|
|
|
|
|
|
switch (r4) {
|
|
|
|
case R4_RD:
|
|
|
|
pr_cont("TLB reload.\n");
|
|
|
|
break;
|
|
|
|
case R4_DWR:
|
|
|
|
pr_cont("store.\n");
|
|
|
|
break;
|
|
|
|
case R4_DRD:
|
|
|
|
pr_cont("load.\n");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static bool f15h_mc0_mce(u16 ec, u8 xec)
|
2010-09-18 01:22:34 +08:00
|
|
|
{
|
|
|
|
bool ret = true;
|
|
|
|
|
|
|
|
if (MEM_ERROR(ec)) {
|
|
|
|
|
|
|
|
switch (xec) {
|
|
|
|
case 0x0:
|
|
|
|
pr_cont("Data Array access error.\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x1:
|
|
|
|
pr_cont("UC error during a linefill from L2/NB.\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x2:
|
|
|
|
case 0x11:
|
|
|
|
pr_cont("STQ access error.\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x3:
|
|
|
|
pr_cont("SCB access error.\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x10:
|
|
|
|
pr_cont("Tag error.\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x12:
|
|
|
|
pr_cont("LDQ access error.\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
} else if (BUS_ERROR(ec)) {
|
|
|
|
|
|
|
|
if (!xec)
|
2011-11-16 00:10:58 +08:00
|
|
|
pr_cont("System Read Data Error.\n");
|
2010-09-18 01:22:34 +08:00
|
|
|
else
|
2011-11-16 00:10:58 +08:00
|
|
|
pr_cont(" Internal error condition type %d.\n", xec);
|
2014-07-14 22:58:19 +08:00
|
|
|
} else if (INT_ERROR(ec)) {
|
|
|
|
if (xec <= 0x1f)
|
|
|
|
pr_cont("Hardware Assert.\n");
|
|
|
|
else
|
|
|
|
ret = false;
|
|
|
|
|
2010-09-18 01:22:34 +08:00
|
|
|
} else
|
|
|
|
ret = false;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static void decode_mc0_mce(struct mce *m)
|
2010-08-18 21:11:35 +08:00
|
|
|
{
|
2010-09-22 22:08:37 +08:00
|
|
|
u16 ec = EC(m->status);
|
|
|
|
u8 xec = XEC(m->status, xec_mask);
|
2010-08-18 21:11:35 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_emerg(HW_ERR "MC0 Error: ");
|
2010-08-18 21:11:35 +08:00
|
|
|
|
|
|
|
/* TLB error signatures are the same across families */
|
|
|
|
if (TLB_ERROR(ec)) {
|
2010-09-22 22:08:37 +08:00
|
|
|
if (TT(ec) == TT_DATA) {
|
2010-08-18 21:11:35 +08:00
|
|
|
pr_cont("%s TLB %s.\n", LL_MSG(ec),
|
2010-09-18 01:22:34 +08:00
|
|
|
((xec == 2) ? "locked miss"
|
|
|
|
: (xec ? "multimatch" : "parity")));
|
2010-08-18 21:11:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2012-09-12 00:57:43 +08:00
|
|
|
} else if (fam_ops->mc0_mce(ec, xec))
|
2010-09-18 01:22:34 +08:00
|
|
|
;
|
|
|
|
else
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_emerg(HW_ERR "Corrupted MC0 MCE info?\n");
|
2009-07-28 19:50:43 +08:00
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static bool k8_mc1_mce(u16 ec, u8 xec)
|
2009-07-28 20:06:26 +08:00
|
|
|
{
|
2010-09-22 22:08:37 +08:00
|
|
|
u8 ll = LL(ec);
|
2010-08-27 01:05:49 +08:00
|
|
|
bool ret = true;
|
2009-07-28 20:06:26 +08:00
|
|
|
|
2010-08-27 01:05:49 +08:00
|
|
|
if (!MEM_ERROR(ec))
|
|
|
|
return false;
|
2009-07-28 20:06:26 +08:00
|
|
|
|
2010-08-27 01:05:49 +08:00
|
|
|
if (ll == 0x2)
|
|
|
|
pr_cont("during a linefill from L2.\n");
|
|
|
|
else if (ll == 0x1) {
|
2010-09-22 22:08:37 +08:00
|
|
|
switch (R4(ec)) {
|
2010-08-27 01:05:49 +08:00
|
|
|
case R4_IRD:
|
|
|
|
pr_cont("Parity error during data load.\n");
|
|
|
|
break;
|
2009-07-28 20:06:26 +08:00
|
|
|
|
2010-08-27 01:05:49 +08:00
|
|
|
case R4_EVICT:
|
|
|
|
pr_cont("Copyback Parity/Victim error.\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R4_SNOOP:
|
|
|
|
pr_cont("Tag Snoop error.\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
ret = false;
|
|
|
|
break;
|
|
|
|
}
|
2009-07-28 20:06:26 +08:00
|
|
|
} else
|
2010-08-27 01:05:49 +08:00
|
|
|
ret = false;
|
2009-07-28 20:06:26 +08:00
|
|
|
|
2010-08-27 01:05:49 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-12-19 05:06:11 +08:00
|
|
|
static bool cat_mc1_mce(u16 ec, u8 xec)
|
2010-08-27 01:05:49 +08:00
|
|
|
{
|
2010-09-22 22:08:37 +08:00
|
|
|
u8 r4 = R4(ec);
|
2010-08-27 01:05:49 +08:00
|
|
|
bool ret = true;
|
2009-07-28 20:06:26 +08:00
|
|
|
|
2012-12-19 05:06:11 +08:00
|
|
|
if (!MEM_ERROR(ec))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (TT(ec) != TT_INSTR)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (r4 == R4_IRD)
|
|
|
|
pr_cont("Data/tag array parity error for a tag hit.\n");
|
|
|
|
else if (r4 == R4_SNOOP)
|
|
|
|
pr_cont("Tag error during snoop/victimization.\n");
|
|
|
|
else if (xec == 0x0)
|
|
|
|
pr_cont("Tag parity error from victim castout.\n");
|
|
|
|
else if (xec == 0x2)
|
|
|
|
pr_cont("Microcode patch RAM parity error.\n");
|
|
|
|
else
|
|
|
|
ret = false;
|
2010-08-27 01:05:49 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static bool f15h_mc1_mce(u16 ec, u8 xec)
|
2010-11-08 22:03:35 +08:00
|
|
|
{
|
|
|
|
bool ret = true;
|
|
|
|
|
|
|
|
if (!MEM_ERROR(ec))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (xec) {
|
|
|
|
case 0x0 ... 0xa:
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("%s.\n", f15h_mc1_mce_desc[xec]);
|
2010-11-08 22:03:35 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0xd:
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("%s.\n", f15h_mc1_mce_desc[xec-2]);
|
2010-11-08 22:03:35 +08:00
|
|
|
break;
|
|
|
|
|
2011-11-22 02:45:34 +08:00
|
|
|
case 0x10:
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("%s.\n", f15h_mc1_mce_desc[xec-4]);
|
2011-11-22 02:45:34 +08:00
|
|
|
break;
|
|
|
|
|
2014-07-14 22:58:19 +08:00
|
|
|
case 0x11 ... 0x15:
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("Decoder %s parity error.\n", f15h_mc1_mce_desc[xec-4]);
|
2010-11-08 22:03:35 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static void decode_mc1_mce(struct mce *m)
|
2010-08-27 01:05:49 +08:00
|
|
|
{
|
2010-09-22 22:08:37 +08:00
|
|
|
u16 ec = EC(m->status);
|
|
|
|
u8 xec = XEC(m->status, xec_mask);
|
2010-08-27 01:05:49 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_emerg(HW_ERR "MC1 Error: ");
|
2010-08-27 01:05:49 +08:00
|
|
|
|
|
|
|
if (TLB_ERROR(ec))
|
|
|
|
pr_cont("%s TLB %s.\n", LL_MSG(ec),
|
|
|
|
(xec ? "multimatch" : "parity error"));
|
|
|
|
else if (BUS_ERROR(ec)) {
|
2010-10-15 21:27:02 +08:00
|
|
|
bool k8 = (boot_cpu_data.x86 == 0xf && (m->status & BIT_64(58)));
|
2010-08-27 01:05:49 +08:00
|
|
|
|
|
|
|
pr_cont("during %s.\n", (k8 ? "system linefill" : "NB data read"));
|
2014-07-14 22:58:19 +08:00
|
|
|
} else if (INT_ERROR(ec)) {
|
|
|
|
if (xec <= 0x3f)
|
|
|
|
pr_cont("Hardware Assert.\n");
|
|
|
|
else
|
|
|
|
goto wrong_mc1_mce;
|
2012-09-12 00:57:43 +08:00
|
|
|
} else if (fam_ops->mc1_mce(ec, xec))
|
2010-08-27 01:05:49 +08:00
|
|
|
;
|
|
|
|
else
|
2014-07-14 22:58:19 +08:00
|
|
|
goto wrong_mc1_mce;
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
wrong_mc1_mce:
|
|
|
|
pr_emerg(HW_ERR "Corrupted MC1 MCE info?\n");
|
2009-07-28 20:06:26 +08:00
|
|
|
}
|
|
|
|
|
2012-12-19 05:06:10 +08:00
|
|
|
static bool k8_mc2_mce(u16 ec, u8 xec)
|
2009-07-28 20:14:24 +08:00
|
|
|
{
|
2012-12-19 05:06:10 +08:00
|
|
|
bool ret = true;
|
2009-07-28 20:14:24 +08:00
|
|
|
|
|
|
|
if (xec == 0x1)
|
|
|
|
pr_cont(" in the write data buffers.\n");
|
|
|
|
else if (xec == 0x3)
|
|
|
|
pr_cont(" in the victim data buffers.\n");
|
|
|
|
else if (xec == 0x2 && MEM_ERROR(ec))
|
2010-09-22 22:08:37 +08:00
|
|
|
pr_cont(": %s error in the L2 cache tags.\n", R4_MSG(ec));
|
2009-07-28 20:14:24 +08:00
|
|
|
else if (xec == 0x0) {
|
|
|
|
if (TLB_ERROR(ec))
|
2014-11-22 20:41:01 +08:00
|
|
|
pr_cont("%s error in a Page Descriptor Cache or Guest TLB.\n",
|
|
|
|
TT_MSG(ec));
|
2009-07-28 20:14:24 +08:00
|
|
|
else if (BUS_ERROR(ec))
|
|
|
|
pr_cont(": %s/ECC error in data read from NB: %s.\n",
|
2010-09-22 22:08:37 +08:00
|
|
|
R4_MSG(ec), PP_MSG(ec));
|
2009-07-28 20:14:24 +08:00
|
|
|
else if (MEM_ERROR(ec)) {
|
2010-09-22 22:08:37 +08:00
|
|
|
u8 r4 = R4(ec);
|
2009-07-28 20:14:24 +08:00
|
|
|
|
2010-09-22 22:08:37 +08:00
|
|
|
if (r4 >= 0x7)
|
2009-07-28 20:14:24 +08:00
|
|
|
pr_cont(": %s error during data copyback.\n",
|
2010-09-22 22:08:37 +08:00
|
|
|
R4_MSG(ec));
|
|
|
|
else if (r4 <= 0x1)
|
2009-07-28 20:14:24 +08:00
|
|
|
pr_cont(": %s parity/ECC error during data "
|
2010-09-22 22:08:37 +08:00
|
|
|
"access from L2.\n", R4_MSG(ec));
|
2009-07-28 20:14:24 +08:00
|
|
|
else
|
2012-12-19 05:06:10 +08:00
|
|
|
ret = false;
|
2009-07-28 20:14:24 +08:00
|
|
|
} else
|
2012-12-19 05:06:10 +08:00
|
|
|
ret = false;
|
2009-07-28 20:14:24 +08:00
|
|
|
} else
|
2012-12-19 05:06:10 +08:00
|
|
|
ret = false;
|
2009-07-28 20:14:24 +08:00
|
|
|
|
2012-12-19 05:06:10 +08:00
|
|
|
return ret;
|
2009-07-28 20:14:24 +08:00
|
|
|
}
|
|
|
|
|
2012-12-19 05:06:10 +08:00
|
|
|
static bool f15h_mc2_mce(u16 ec, u8 xec)
|
2010-09-22 02:45:10 +08:00
|
|
|
{
|
2012-12-19 05:06:10 +08:00
|
|
|
bool ret = true;
|
2010-09-22 02:45:10 +08:00
|
|
|
|
|
|
|
if (TLB_ERROR(ec)) {
|
|
|
|
if (xec == 0x0)
|
|
|
|
pr_cont("Data parity TLB read error.\n");
|
|
|
|
else if (xec == 0x1)
|
|
|
|
pr_cont("Poison data provided for TLB fill.\n");
|
|
|
|
else
|
2012-12-19 05:06:10 +08:00
|
|
|
ret = false;
|
2010-09-22 02:45:10 +08:00
|
|
|
} else if (BUS_ERROR(ec)) {
|
|
|
|
if (xec > 2)
|
2012-12-19 05:06:10 +08:00
|
|
|
ret = false;
|
2010-09-22 02:45:10 +08:00
|
|
|
|
|
|
|
pr_cont("Error during attempted NB data read.\n");
|
|
|
|
} else if (MEM_ERROR(ec)) {
|
|
|
|
switch (xec) {
|
|
|
|
case 0x4 ... 0xc:
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("%s.\n", f15h_mc2_mce_desc[xec - 0x4]);
|
2010-09-22 02:45:10 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x10 ... 0x14:
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("%s.\n", f15h_mc2_mce_desc[xec - 0x7]);
|
2010-09-22 02:45:10 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2012-12-19 05:06:10 +08:00
|
|
|
ret = false;
|
2010-09-22 02:45:10 +08:00
|
|
|
}
|
2014-07-14 22:58:19 +08:00
|
|
|
} else if (INT_ERROR(ec)) {
|
|
|
|
if (xec <= 0x3f)
|
|
|
|
pr_cont("Hardware Assert.\n");
|
|
|
|
else
|
|
|
|
ret = false;
|
2010-09-22 02:45:10 +08:00
|
|
|
}
|
|
|
|
|
2012-12-19 05:06:10 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-12-19 05:06:11 +08:00
|
|
|
static bool f16h_mc2_mce(u16 ec, u8 xec)
|
|
|
|
{
|
|
|
|
u8 r4 = R4(ec);
|
|
|
|
|
|
|
|
if (!MEM_ERROR(ec))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (xec) {
|
|
|
|
case 0x04 ... 0x05:
|
|
|
|
pr_cont("%cBUFF parity error.\n", (r4 == R4_RD) ? 'I' : 'O');
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x09 ... 0x0b:
|
|
|
|
case 0x0d ... 0x0f:
|
|
|
|
pr_cont("ECC error in L2 tag (%s).\n",
|
|
|
|
((r4 == R4_GEN) ? "BankReq" :
|
|
|
|
((r4 == R4_SNOOP) ? "Prb" : "Fill")));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x10 ... 0x19:
|
|
|
|
case 0x1b:
|
|
|
|
pr_cont("ECC error in L2 data array (%s).\n",
|
|
|
|
(((r4 == R4_RD) && !(xec & 0x3)) ? "Hit" :
|
|
|
|
((r4 == R4_GEN) ? "Attr" :
|
|
|
|
((r4 == R4_EVICT) ? "Vict" : "Fill"))));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x1c ... 0x1d:
|
|
|
|
case 0x1f:
|
|
|
|
pr_cont("Parity error in L2 attribute bits (%s).\n",
|
|
|
|
((r4 == R4_RD) ? "Hit" :
|
|
|
|
((r4 == R4_GEN) ? "Attr" : "Fill")));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-12-19 05:06:10 +08:00
|
|
|
static void decode_mc2_mce(struct mce *m)
|
|
|
|
{
|
|
|
|
u16 ec = EC(m->status);
|
|
|
|
u8 xec = XEC(m->status, xec_mask);
|
2010-09-22 02:45:10 +08:00
|
|
|
|
2012-12-19 05:06:10 +08:00
|
|
|
pr_emerg(HW_ERR "MC2 Error: ");
|
|
|
|
|
|
|
|
if (!fam_ops->mc2_mce(ec, xec))
|
|
|
|
pr_cont(HW_ERR "Corrupted MC2 MCE info?\n");
|
2010-09-22 02:45:10 +08:00
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static void decode_mc3_mce(struct mce *m)
|
2009-07-28 20:17:30 +08:00
|
|
|
{
|
2010-09-22 22:08:37 +08:00
|
|
|
u16 ec = EC(m->status);
|
|
|
|
u8 xec = XEC(m->status, xec_mask);
|
2010-08-27 23:03:34 +08:00
|
|
|
|
2010-09-22 17:53:32 +08:00
|
|
|
if (boot_cpu_data.x86 >= 0x14) {
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_emerg("You shouldn't be seeing MC3 MCE on this cpu family,"
|
2010-08-27 23:03:34 +08:00
|
|
|
" please report on LKML.\n");
|
|
|
|
return;
|
|
|
|
}
|
2009-07-28 20:17:30 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_emerg(HW_ERR "MC3 Error");
|
2009-07-28 20:17:30 +08:00
|
|
|
|
|
|
|
if (xec == 0x0) {
|
2010-09-22 22:08:37 +08:00
|
|
|
u8 r4 = R4(ec);
|
2009-07-28 20:17:30 +08:00
|
|
|
|
2010-08-27 23:03:34 +08:00
|
|
|
if (!BUS_ERROR(ec) || (r4 != R4_DRD && r4 != R4_DWR))
|
2012-09-12 00:57:43 +08:00
|
|
|
goto wrong_mc3_mce;
|
2009-07-28 20:17:30 +08:00
|
|
|
|
2010-09-22 22:08:37 +08:00
|
|
|
pr_cont(" during %s.\n", R4_MSG(ec));
|
2010-08-27 23:03:34 +08:00
|
|
|
} else
|
2012-09-12 00:57:43 +08:00
|
|
|
goto wrong_mc3_mce;
|
2010-08-27 23:03:34 +08:00
|
|
|
|
2009-07-28 20:17:30 +08:00
|
|
|
return;
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
wrong_mc3_mce:
|
|
|
|
pr_emerg(HW_ERR "Corrupted MC3 MCE info?\n");
|
2009-07-28 20:17:30 +08:00
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static void decode_mc4_mce(struct mce *m)
|
2010-09-01 00:28:08 +08:00
|
|
|
{
|
2017-07-25 17:07:03 +08:00
|
|
|
unsigned int fam = x86_family(m->cpuid);
|
2011-11-25 04:29:57 +08:00
|
|
|
int node_id = amd_get_nb_id(m->extcpu);
|
|
|
|
u16 ec = EC(m->status);
|
|
|
|
u8 xec = XEC(m->status, 0x1f);
|
|
|
|
u8 offset = 0;
|
2010-09-01 00:28:08 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_emerg(HW_ERR "MC4 Error (node %d): ", node_id);
|
2010-09-01 00:28:08 +08:00
|
|
|
|
2011-11-25 04:29:57 +08:00
|
|
|
switch (xec) {
|
|
|
|
case 0x0 ... 0xe:
|
2010-09-01 00:28:08 +08:00
|
|
|
|
2011-11-25 04:29:57 +08:00
|
|
|
/* special handling for DRAM ECCs */
|
|
|
|
if (xec == 0x0 || xec == 0x8) {
|
|
|
|
/* no ECCs on F11h */
|
2017-07-25 17:07:03 +08:00
|
|
|
if (fam == 0x11)
|
2012-09-12 00:57:43 +08:00
|
|
|
goto wrong_mc4_mce;
|
2010-09-01 00:28:08 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("%s.\n", mc4_mce_desc[xec]);
|
2010-09-01 00:28:08 +08:00
|
|
|
|
2016-11-18 06:57:29 +08:00
|
|
|
if (decode_dram_ecc)
|
|
|
|
decode_dram_ecc(node_id, m);
|
2011-11-25 04:29:57 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-09-01 00:28:08 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0xf:
|
|
|
|
if (TLB_ERROR(ec))
|
|
|
|
pr_cont("GART Table Walk data error.\n");
|
|
|
|
else if (BUS_ERROR(ec))
|
|
|
|
pr_cont("DMA Exclusion Vector Table Walk error.\n");
|
|
|
|
else
|
2012-09-12 00:57:43 +08:00
|
|
|
goto wrong_mc4_mce;
|
2011-11-25 04:29:57 +08:00
|
|
|
return;
|
2010-09-01 00:28:08 +08:00
|
|
|
|
2010-09-22 21:06:24 +08:00
|
|
|
case 0x19:
|
2017-07-25 17:07:03 +08:00
|
|
|
if (fam == 0x15 || fam == 0x16)
|
2010-09-22 21:06:24 +08:00
|
|
|
pr_cont("Compute Unit Data Error.\n");
|
|
|
|
else
|
2012-09-12 00:57:43 +08:00
|
|
|
goto wrong_mc4_mce;
|
2011-11-25 04:29:57 +08:00
|
|
|
return;
|
2010-09-22 21:06:24 +08:00
|
|
|
|
2010-09-01 00:28:08 +08:00
|
|
|
case 0x1c ... 0x1f:
|
2011-11-25 04:29:57 +08:00
|
|
|
offset = 13;
|
2010-09-01 00:28:08 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2012-09-12 00:57:43 +08:00
|
|
|
goto wrong_mc4_mce;
|
2011-11-25 04:29:57 +08:00
|
|
|
}
|
2010-09-01 00:28:08 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("%s.\n", mc4_mce_desc[xec - offset]);
|
2010-09-01 00:28:08 +08:00
|
|
|
return;
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
wrong_mc4_mce:
|
|
|
|
pr_emerg(HW_ERR "Corrupted MC4 MCE info?\n");
|
2009-07-28 16:56:15 +08:00
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static void decode_mc5_mce(struct mce *m)
|
2009-07-28 20:20:46 +08:00
|
|
|
{
|
2017-07-25 17:07:03 +08:00
|
|
|
unsigned int fam = x86_family(m->cpuid);
|
2014-07-14 22:58:19 +08:00
|
|
|
u16 ec = EC(m->status);
|
2010-09-22 22:08:37 +08:00
|
|
|
u8 xec = XEC(m->status, xec_mask);
|
2010-09-22 21:28:59 +08:00
|
|
|
|
2017-07-25 17:07:03 +08:00
|
|
|
if (fam == 0xf || fam == 0x11)
|
2012-09-12 00:57:43 +08:00
|
|
|
goto wrong_mc5_mce;
|
2010-09-01 00:38:24 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_emerg(HW_ERR "MC5 Error: ");
|
2010-09-22 21:28:59 +08:00
|
|
|
|
2014-07-14 22:58:19 +08:00
|
|
|
if (INT_ERROR(ec)) {
|
|
|
|
if (xec <= 0x1f) {
|
|
|
|
pr_cont("Hardware Assert.\n");
|
|
|
|
return;
|
|
|
|
} else
|
|
|
|
goto wrong_mc5_mce;
|
|
|
|
}
|
|
|
|
|
2010-09-22 21:28:59 +08:00
|
|
|
if (xec == 0x0 || xec == 0xc)
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("%s.\n", mc5_mce_desc[xec]);
|
2013-06-06 04:50:03 +08:00
|
|
|
else if (xec <= 0xd)
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_cont("%s parity error.\n", mc5_mce_desc[xec]);
|
2010-09-22 21:28:59 +08:00
|
|
|
else
|
2012-09-12 00:57:43 +08:00
|
|
|
goto wrong_mc5_mce;
|
2010-09-22 21:28:59 +08:00
|
|
|
|
|
|
|
return;
|
2010-09-01 00:38:24 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
wrong_mc5_mce:
|
|
|
|
pr_emerg(HW_ERR "Corrupted MC5 MCE info?\n");
|
2009-07-28 20:20:46 +08:00
|
|
|
}
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
static void decode_mc6_mce(struct mce *m)
|
2010-09-22 21:37:58 +08:00
|
|
|
{
|
2010-09-22 22:08:37 +08:00
|
|
|
u8 xec = XEC(m->status, xec_mask);
|
2010-09-22 21:37:58 +08:00
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
pr_emerg(HW_ERR "MC6 Error: ");
|
2010-09-22 21:37:58 +08:00
|
|
|
|
2014-11-05 01:41:08 +08:00
|
|
|
if (xec > 0x5)
|
2012-09-12 00:57:43 +08:00
|
|
|
goto wrong_mc6_mce;
|
2010-09-22 21:37:58 +08:00
|
|
|
|
2014-11-05 01:41:08 +08:00
|
|
|
pr_cont("%s parity error.\n", mc6_mce_desc[xec]);
|
2010-09-22 21:37:58 +08:00
|
|
|
return;
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
wrong_mc6_mce:
|
|
|
|
pr_emerg(HW_ERR "Corrupted MC6 MCE info?\n");
|
2010-09-22 21:37:58 +08:00
|
|
|
}
|
|
|
|
|
2016-03-07 21:02:18 +08:00
|
|
|
/* Decode errors according to Scalable MCA specification */
|
2017-07-25 16:44:59 +08:00
|
|
|
static void decode_smca_error(struct mce *m)
|
2016-03-07 21:02:18 +08:00
|
|
|
{
|
2016-11-02 19:48:01 +08:00
|
|
|
struct smca_hwid *hwid;
|
x86/mce/AMD, EDAC/mce_amd: Enumerate Reserved SMCA bank type
Currently, bank 4 is reserved on Fam17h, so we chose not to initialize
bank 4 in the smca_banks array. This means that when we check if a bank
is initialized, like during boot or resume, we will see that bank 4 is
not initialized and try to initialize it.
This will cause a call trace, when resuming from suspend, due to
rdmsr_*on_cpu() calls in the init path. The rdmsr_*on_cpu() calls issue
an IPI but we're running with interrupts disabled. This triggers:
WARNING: CPU: 0 PID: 11523 at kernel/smp.c:291 smp_call_function_single+0xdc/0xe0
...
Reserved banks will be read-as-zero, so their MCA_IPID register will be
zero. So, like the smca_banks array, the threshold_banks array will not
have an entry for a reserved bank since all its MCA_MISC* registers will
be zero.
Enumerate a "Reserved" bank type that matches on a HWID_MCATYPE of 0,0.
Use the "Reserved" type when checking if a bank is reserved. It's
possible that other bank numbers may be reserved on future systems.
Don't try to find the block address on reserved banks.
Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: Borislav Petkov <bp@suse.de>
Cc: <stable@vger.kernel.org> # 4.14.x
Cc: Borislav Petkov <bp@alien8.de>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Tony Luck <tony.luck@intel.com>
Cc: linux-edac <linux-edac@vger.kernel.org>
Link: http://lkml.kernel.org/r/20180221101900.10326-7-bp@alien8.de
Signed-off-by: Ingo Molnar <mingo@kernel.org>
2018-02-21 18:18:58 +08:00
|
|
|
enum smca_bank_types bank_type;
|
2016-03-07 21:02:18 +08:00
|
|
|
const char *ip_name;
|
2016-09-12 15:59:34 +08:00
|
|
|
u8 xec = XEC(m->status, xec_mask);
|
2016-03-07 21:02:18 +08:00
|
|
|
|
2016-09-12 15:59:34 +08:00
|
|
|
if (m->bank >= ARRAY_SIZE(smca_banks))
|
2016-03-07 21:02:18 +08:00
|
|
|
return;
|
|
|
|
|
2016-11-02 19:48:01 +08:00
|
|
|
hwid = smca_banks[m->bank].hwid;
|
|
|
|
if (!hwid)
|
2016-03-07 21:02:18 +08:00
|
|
|
return;
|
|
|
|
|
2016-11-02 19:48:01 +08:00
|
|
|
bank_type = hwid->bank_type;
|
x86/mce/AMD, EDAC/mce_amd: Enumerate Reserved SMCA bank type
Currently, bank 4 is reserved on Fam17h, so we chose not to initialize
bank 4 in the smca_banks array. This means that when we check if a bank
is initialized, like during boot or resume, we will see that bank 4 is
not initialized and try to initialize it.
This will cause a call trace, when resuming from suspend, due to
rdmsr_*on_cpu() calls in the init path. The rdmsr_*on_cpu() calls issue
an IPI but we're running with interrupts disabled. This triggers:
WARNING: CPU: 0 PID: 11523 at kernel/smp.c:291 smp_call_function_single+0xdc/0xe0
...
Reserved banks will be read-as-zero, so their MCA_IPID register will be
zero. So, like the smca_banks array, the threshold_banks array will not
have an entry for a reserved bank since all its MCA_MISC* registers will
be zero.
Enumerate a "Reserved" bank type that matches on a HWID_MCATYPE of 0,0.
Use the "Reserved" type when checking if a bank is reserved. It's
possible that other bank numbers may be reserved on future systems.
Don't try to find the block address on reserved banks.
Signed-off-by: Yazen Ghannam <yazen.ghannam@amd.com>
Signed-off-by: Borislav Petkov <bp@suse.de>
Cc: <stable@vger.kernel.org> # 4.14.x
Cc: Borislav Petkov <bp@alien8.de>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Tony Luck <tony.luck@intel.com>
Cc: linux-edac <linux-edac@vger.kernel.org>
Link: http://lkml.kernel.org/r/20180221101900.10326-7-bp@alien8.de
Signed-off-by: Ingo Molnar <mingo@kernel.org>
2018-02-21 18:18:58 +08:00
|
|
|
|
|
|
|
if (bank_type == SMCA_RESERVED) {
|
|
|
|
pr_emerg(HW_ERR "Bank %d is reserved.\n", m->bank);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-11-04 04:12:33 +08:00
|
|
|
ip_name = smca_get_long_name(bank_type);
|
2016-03-07 21:02:18 +08:00
|
|
|
|
2016-09-12 15:59:34 +08:00
|
|
|
pr_emerg(HW_ERR "%s Extended Error Code: %d\n", ip_name, xec);
|
2016-03-07 21:02:18 +08:00
|
|
|
|
2016-09-12 15:59:34 +08:00
|
|
|
/* Only print the decode of valid error codes */
|
|
|
|
if (xec < smca_mce_descs[bank_type].num_descs &&
|
2016-11-02 19:48:01 +08:00
|
|
|
(hwid->xec_bitmap & BIT_ULL(xec))) {
|
2016-09-12 15:59:34 +08:00
|
|
|
pr_emerg(HW_ERR "%s Error: ", ip_name);
|
|
|
|
pr_cont("%s.\n", smca_mce_descs[bank_type].descs[xec]);
|
|
|
|
}
|
2016-11-18 06:57:29 +08:00
|
|
|
|
|
|
|
if (bank_type == SMCA_UMC && xec == 0 && decode_dram_ecc)
|
2017-03-21 04:26:51 +08:00
|
|
|
decode_dram_ecc(cpu_to_node(m->extcpu), m);
|
2016-03-07 21:02:18 +08:00
|
|
|
}
|
|
|
|
|
2010-09-07 00:13:39 +08:00
|
|
|
static inline void amd_decode_err_code(u16 ec)
|
2009-07-28 16:56:15 +08:00
|
|
|
{
|
2012-12-19 05:06:11 +08:00
|
|
|
if (INT_ERROR(ec)) {
|
|
|
|
pr_emerg(HW_ERR "internal: %s\n", UU_MSG(ec));
|
|
|
|
return;
|
|
|
|
}
|
2010-09-22 23:42:27 +08:00
|
|
|
|
|
|
|
pr_emerg(HW_ERR "cache level: %s", LL_MSG(ec));
|
|
|
|
|
|
|
|
if (BUS_ERROR(ec))
|
|
|
|
pr_cont(", mem/io: %s", II_MSG(ec));
|
|
|
|
else
|
|
|
|
pr_cont(", tx: %s", TT_MSG(ec));
|
|
|
|
|
|
|
|
if (MEM_ERROR(ec) || BUS_ERROR(ec)) {
|
|
|
|
pr_cont(", mem-tx: %s", R4_MSG(ec));
|
|
|
|
|
|
|
|
if (BUS_ERROR(ec))
|
|
|
|
pr_cont(", part-proc: %s (%s)", PP_MSG(ec), TO_MSG(ec));
|
|
|
|
}
|
|
|
|
|
|
|
|
pr_cont("\n");
|
2009-07-24 19:51:42 +08:00
|
|
|
}
|
|
|
|
|
2010-09-01 00:28:08 +08:00
|
|
|
/*
|
|
|
|
* Filter out unwanted MCE signatures here.
|
|
|
|
*/
|
|
|
|
static bool amd_filter_mce(struct mce *m)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* NB GART TLB error reporting is disabled by default.
|
|
|
|
*/
|
2017-07-25 17:09:56 +08:00
|
|
|
if (m->bank == 4 && XEC(m->status, 0x1f) == 0x5 && !report_gart_errors)
|
2010-09-01 00:28:08 +08:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-09-15 02:25:37 +08:00
|
|
|
static const char *decode_error_status(struct mce *m)
|
|
|
|
{
|
|
|
|
if (m->status & MCI_STATUS_UC) {
|
|
|
|
if (m->status & MCI_STATUS_PCC)
|
|
|
|
return "System Fatal error.";
|
|
|
|
if (m->mcgstatus & MCG_STATUS_RIPV)
|
|
|
|
return "Uncorrected, software restartable error.";
|
|
|
|
return "Uncorrected, software containable error.";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m->status & MCI_STATUS_DEFERRED)
|
2017-01-25 06:32:23 +08:00
|
|
|
return "Deferred error, no action required.";
|
2012-09-15 02:25:37 +08:00
|
|
|
|
|
|
|
return "Corrected error, no action required.";
|
|
|
|
}
|
|
|
|
|
2017-01-24 02:35:11 +08:00
|
|
|
static int
|
|
|
|
amd_decode_mce(struct notifier_block *nb, unsigned long val, void *data)
|
2009-07-24 19:51:42 +08:00
|
|
|
{
|
2009-10-07 19:20:38 +08:00
|
|
|
struct mce *m = (struct mce *)data;
|
2017-07-25 17:07:03 +08:00
|
|
|
unsigned int fam = x86_family(m->cpuid);
|
2011-08-25 00:44:22 +08:00
|
|
|
int ecc;
|
2009-07-24 19:51:42 +08:00
|
|
|
|
2010-09-01 00:28:08 +08:00
|
|
|
if (amd_filter_mce(m))
|
|
|
|
return NOTIFY_STOP;
|
|
|
|
|
2014-02-18 03:51:52 +08:00
|
|
|
pr_emerg(HW_ERR "%s\n", decode_error_status(m));
|
|
|
|
|
|
|
|
pr_emerg(HW_ERR "CPU:%d (%x:%x:%x) MC%d_STATUS[%s|%s|%s|%s|%s",
|
|
|
|
m->extcpu,
|
2017-07-25 17:07:03 +08:00
|
|
|
fam, x86_model(m->cpuid), x86_stepping(m->cpuid),
|
2014-02-18 03:51:52 +08:00
|
|
|
m->bank,
|
|
|
|
((m->status & MCI_STATUS_OVER) ? "Over" : "-"),
|
2015-07-13 19:53:02 +08:00
|
|
|
((m->status & MCI_STATUS_UC) ? "UE" :
|
|
|
|
(m->status & MCI_STATUS_DEFERRED) ? "-" : "CE"),
|
2014-02-18 03:51:52 +08:00
|
|
|
((m->status & MCI_STATUS_MISCV) ? "MiscV" : "-"),
|
|
|
|
((m->status & MCI_STATUS_PCC) ? "PCC" : "-"),
|
|
|
|
((m->status & MCI_STATUS_ADDRV) ? "AddrV" : "-"));
|
|
|
|
|
2017-07-25 17:07:03 +08:00
|
|
|
if (fam >= 0x15) {
|
2016-11-18 22:10:22 +08:00
|
|
|
pr_cont("|%s", (m->status & MCI_STATUS_DEFERRED ? "Deferred" : "-"));
|
|
|
|
|
|
|
|
/* F15h, bank4, bit 43 is part of McaStatSubCache. */
|
2017-07-25 17:07:03 +08:00
|
|
|
if (fam != 0x15 || m->bank != 4)
|
2016-11-18 22:10:22 +08:00
|
|
|
pr_cont("|%s", (m->status & MCI_STATUS_POISON ? "Poison" : "-"));
|
|
|
|
}
|
2014-02-18 03:51:52 +08:00
|
|
|
|
2016-05-11 20:58:28 +08:00
|
|
|
if (boot_cpu_has(X86_FEATURE_SMCA)) {
|
2016-03-07 21:02:18 +08:00
|
|
|
u32 low, high;
|
|
|
|
u32 addr = MSR_AMD64_SMCA_MCx_CONFIG(m->bank);
|
|
|
|
|
2016-09-12 15:59:29 +08:00
|
|
|
pr_cont("|%s", ((m->status & MCI_STATUS_SYNDV) ? "SyndV" : "-"));
|
|
|
|
|
2016-03-07 21:02:18 +08:00
|
|
|
if (!rdmsr_safe(addr, &low, &high) &&
|
|
|
|
(low & MCI_CONFIG_MCAX))
|
|
|
|
pr_cont("|%s", ((m->status & MCI_STATUS_TCC) ? "TCC" : "-"));
|
|
|
|
}
|
|
|
|
|
2014-02-18 03:51:52 +08:00
|
|
|
/* do the two bits[14:13] together */
|
|
|
|
ecc = (m->status >> 45) & 0x3;
|
|
|
|
if (ecc)
|
|
|
|
pr_cont("|%sECC", ((ecc == 2) ? "C" : "U"));
|
|
|
|
|
|
|
|
pr_cont("]: 0x%016llx\n", m->status);
|
|
|
|
|
|
|
|
if (m->status & MCI_STATUS_ADDRV)
|
EDAC, mce_amd: Print IPID and Syndrome on a separate line
Currently, the IPID and Syndrome are printed on the same line as the
Address. There are cases when we can have a valid Syndrome but not a
valid Address.
For example, the MCA_SYND register can be used to hold more detailed
error info that the hardware folks can use. It's not just DRAM ECC
syndromes. There are some error types that aren't related to memory that
may have valid syndromes, like some errors related to links in the Data
Fabric, etc.
In these cases, the IPID and Syndrome are not printed at the same log
level as the rest of the stanza, so users won't see them on the console.
Console:
[Hardware Error]: CPU:16 (17:1:0) MC22_STATUS[Over|CE|MiscV|-|-|-|-|SyndV|-]: 0xd82000000002080b
[Hardware Error]: Power, Interrupts, etc. Extended Error Code: 2
Dmesg:
[Hardware Error]: CPU:16 (17:1:0) MC22_STATUS[Over|CE|MiscV|-|-|-|-|SyndV|-]: 0xd82000000002080b
, Syndrome: 0x000000010b404000, IPID: 0x0001002e00000002
[Hardware Error]: Power, Interrupts, etc. Extended Error Code: 2
Print the IPID first and on a new line. The IPID should always be
printed on SMCA systems. The Syndrome will then be printed with the IPID
and at the same log level when valid:
[Hardware Error]: CPU:16 (17:1:0) MC22_STATUS[Over|CE|MiscV|-|-|-|-|SyndV|-]: 0xd82000000002080b
[Hardware Error]: IPID: 0x0001002e00000002, Syndrome: 0x000000010b404000
[Hardware Error]: Power, Interrupts, etc. Extended Error Code: 2
Signed-off-by: Yazen Ghannam <Yazen.Ghannam@amd.com>
Cc: linux-edac <linux-edac@vger.kernel.org>
Link: http://lkml.kernel.org/r/1487192182-2474-1-git-send-email-Yazen.Ghannam@amd.com
Signed-off-by: Borislav Petkov <bp@suse.de>
2017-02-16 04:56:22 +08:00
|
|
|
pr_emerg(HW_ERR "Error Addr: 0x%016llx\n", m->addr);
|
2014-02-18 03:51:52 +08:00
|
|
|
|
2016-05-11 20:58:28 +08:00
|
|
|
if (boot_cpu_has(X86_FEATURE_SMCA)) {
|
EDAC, mce_amd: Print IPID and Syndrome on a separate line
Currently, the IPID and Syndrome are printed on the same line as the
Address. There are cases when we can have a valid Syndrome but not a
valid Address.
For example, the MCA_SYND register can be used to hold more detailed
error info that the hardware folks can use. It's not just DRAM ECC
syndromes. There are some error types that aren't related to memory that
may have valid syndromes, like some errors related to links in the Data
Fabric, etc.
In these cases, the IPID and Syndrome are not printed at the same log
level as the rest of the stanza, so users won't see them on the console.
Console:
[Hardware Error]: CPU:16 (17:1:0) MC22_STATUS[Over|CE|MiscV|-|-|-|-|SyndV|-]: 0xd82000000002080b
[Hardware Error]: Power, Interrupts, etc. Extended Error Code: 2
Dmesg:
[Hardware Error]: CPU:16 (17:1:0) MC22_STATUS[Over|CE|MiscV|-|-|-|-|SyndV|-]: 0xd82000000002080b
, Syndrome: 0x000000010b404000, IPID: 0x0001002e00000002
[Hardware Error]: Power, Interrupts, etc. Extended Error Code: 2
Print the IPID first and on a new line. The IPID should always be
printed on SMCA systems. The Syndrome will then be printed with the IPID
and at the same log level when valid:
[Hardware Error]: CPU:16 (17:1:0) MC22_STATUS[Over|CE|MiscV|-|-|-|-|SyndV|-]: 0xd82000000002080b
[Hardware Error]: IPID: 0x0001002e00000002, Syndrome: 0x000000010b404000
[Hardware Error]: Power, Interrupts, etc. Extended Error Code: 2
Signed-off-by: Yazen Ghannam <Yazen.Ghannam@amd.com>
Cc: linux-edac <linux-edac@vger.kernel.org>
Link: http://lkml.kernel.org/r/1487192182-2474-1-git-send-email-Yazen.Ghannam@amd.com
Signed-off-by: Borislav Petkov <bp@suse.de>
2017-02-16 04:56:22 +08:00
|
|
|
pr_emerg(HW_ERR "IPID: 0x%016llx", m->ipid);
|
|
|
|
|
2016-09-12 15:59:29 +08:00
|
|
|
if (m->status & MCI_STATUS_SYNDV)
|
|
|
|
pr_cont(", Syndrome: 0x%016llx", m->synd);
|
|
|
|
|
|
|
|
pr_cont("\n");
|
|
|
|
|
2017-07-25 16:44:59 +08:00
|
|
|
decode_smca_error(m);
|
2016-03-07 21:02:18 +08:00
|
|
|
goto err_code;
|
EDAC, mce_amd: Print IPID and Syndrome on a separate line
Currently, the IPID and Syndrome are printed on the same line as the
Address. There are cases when we can have a valid Syndrome but not a
valid Address.
For example, the MCA_SYND register can be used to hold more detailed
error info that the hardware folks can use. It's not just DRAM ECC
syndromes. There are some error types that aren't related to memory that
may have valid syndromes, like some errors related to links in the Data
Fabric, etc.
In these cases, the IPID and Syndrome are not printed at the same log
level as the rest of the stanza, so users won't see them on the console.
Console:
[Hardware Error]: CPU:16 (17:1:0) MC22_STATUS[Over|CE|MiscV|-|-|-|-|SyndV|-]: 0xd82000000002080b
[Hardware Error]: Power, Interrupts, etc. Extended Error Code: 2
Dmesg:
[Hardware Error]: CPU:16 (17:1:0) MC22_STATUS[Over|CE|MiscV|-|-|-|-|SyndV|-]: 0xd82000000002080b
, Syndrome: 0x000000010b404000, IPID: 0x0001002e00000002
[Hardware Error]: Power, Interrupts, etc. Extended Error Code: 2
Print the IPID first and on a new line. The IPID should always be
printed on SMCA systems. The Syndrome will then be printed with the IPID
and at the same log level when valid:
[Hardware Error]: CPU:16 (17:1:0) MC22_STATUS[Over|CE|MiscV|-|-|-|-|SyndV|-]: 0xd82000000002080b
[Hardware Error]: IPID: 0x0001002e00000002, Syndrome: 0x000000010b404000
[Hardware Error]: Power, Interrupts, etc. Extended Error Code: 2
Signed-off-by: Yazen Ghannam <Yazen.Ghannam@amd.com>
Cc: linux-edac <linux-edac@vger.kernel.org>
Link: http://lkml.kernel.org/r/1487192182-2474-1-git-send-email-Yazen.Ghannam@amd.com
Signed-off-by: Borislav Petkov <bp@suse.de>
2017-02-16 04:56:22 +08:00
|
|
|
}
|
2016-03-07 21:02:18 +08:00
|
|
|
|
2017-01-24 02:35:12 +08:00
|
|
|
if (m->tsc)
|
|
|
|
pr_emerg(HW_ERR "TSC: %llu\n", m->tsc);
|
|
|
|
|
2014-02-18 03:51:52 +08:00
|
|
|
if (!fam_ops)
|
|
|
|
goto err_code;
|
|
|
|
|
2009-07-28 19:50:43 +08:00
|
|
|
switch (m->bank) {
|
|
|
|
case 0:
|
2012-09-12 00:57:43 +08:00
|
|
|
decode_mc0_mce(m);
|
2009-07-28 19:50:43 +08:00
|
|
|
break;
|
2009-07-28 16:56:15 +08:00
|
|
|
|
2009-07-28 20:06:26 +08:00
|
|
|
case 1:
|
2012-09-12 00:57:43 +08:00
|
|
|
decode_mc1_mce(m);
|
2009-07-28 20:06:26 +08:00
|
|
|
break;
|
|
|
|
|
2009-07-28 20:14:24 +08:00
|
|
|
case 2:
|
2012-12-19 05:06:10 +08:00
|
|
|
decode_mc2_mce(m);
|
2009-07-28 20:14:24 +08:00
|
|
|
break;
|
|
|
|
|
2009-07-28 20:17:30 +08:00
|
|
|
case 3:
|
2012-09-12 00:57:43 +08:00
|
|
|
decode_mc3_mce(m);
|
2009-07-28 20:17:30 +08:00
|
|
|
break;
|
|
|
|
|
2009-07-28 19:50:43 +08:00
|
|
|
case 4:
|
2012-09-12 00:57:43 +08:00
|
|
|
decode_mc4_mce(m);
|
2009-07-28 19:50:43 +08:00
|
|
|
break;
|
|
|
|
|
2009-07-28 20:20:46 +08:00
|
|
|
case 5:
|
2012-09-12 00:57:43 +08:00
|
|
|
decode_mc5_mce(m);
|
2009-07-28 20:20:46 +08:00
|
|
|
break;
|
|
|
|
|
2010-09-22 21:37:58 +08:00
|
|
|
case 6:
|
2012-09-12 00:57:43 +08:00
|
|
|
decode_mc6_mce(m);
|
2010-09-22 21:37:58 +08:00
|
|
|
break;
|
|
|
|
|
2009-07-28 19:50:43 +08:00
|
|
|
default:
|
|
|
|
break;
|
2009-07-27 22:21:14 +08:00
|
|
|
}
|
2009-07-28 19:50:43 +08:00
|
|
|
|
2014-02-18 03:51:52 +08:00
|
|
|
err_code:
|
2009-07-28 19:50:43 +08:00
|
|
|
amd_decode_err_code(m->status & 0xffff);
|
2009-10-07 19:20:38 +08:00
|
|
|
|
|
|
|
return NOTIFY_STOP;
|
2009-07-24 19:51:42 +08:00
|
|
|
}
|
2009-10-01 22:14:32 +08:00
|
|
|
|
2009-10-07 19:20:38 +08:00
|
|
|
static struct notifier_block amd_mce_dec_nb = {
|
|
|
|
.notifier_call = amd_decode_mce,
|
2017-01-24 02:35:14 +08:00
|
|
|
.priority = MCE_PRIO_EDAC,
|
2009-10-07 19:20:38 +08:00
|
|
|
};
|
|
|
|
|
2009-10-01 22:14:32 +08:00
|
|
|
static int __init mce_amd_init(void)
|
|
|
|
{
|
2010-09-22 23:44:51 +08:00
|
|
|
struct cpuinfo_x86 *c = &boot_cpu_data;
|
|
|
|
|
2018-09-27 22:31:28 +08:00
|
|
|
if (c->x86_vendor != X86_VENDOR_AMD &&
|
|
|
|
c->x86_vendor != X86_VENDOR_HYGON)
|
2014-02-18 03:51:52 +08:00
|
|
|
return -ENODEV;
|
2010-08-07 00:55:45 +08:00
|
|
|
|
2010-08-18 21:11:35 +08:00
|
|
|
fam_ops = kzalloc(sizeof(struct amd_decoder_ops), GFP_KERNEL);
|
|
|
|
if (!fam_ops)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2010-09-22 23:44:51 +08:00
|
|
|
switch (c->x86) {
|
2010-08-18 21:11:35 +08:00
|
|
|
case 0xf:
|
2012-09-12 00:57:43 +08:00
|
|
|
fam_ops->mc0_mce = k8_mc0_mce;
|
|
|
|
fam_ops->mc1_mce = k8_mc1_mce;
|
2012-12-19 05:06:10 +08:00
|
|
|
fam_ops->mc2_mce = k8_mc2_mce;
|
2010-08-18 21:11:35 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x10:
|
2012-09-12 00:57:43 +08:00
|
|
|
fam_ops->mc0_mce = f10h_mc0_mce;
|
|
|
|
fam_ops->mc1_mce = k8_mc1_mce;
|
2012-12-19 05:06:10 +08:00
|
|
|
fam_ops->mc2_mce = k8_mc2_mce;
|
2010-08-18 21:11:35 +08:00
|
|
|
break;
|
|
|
|
|
2010-10-06 01:07:16 +08:00
|
|
|
case 0x11:
|
2012-09-12 00:57:43 +08:00
|
|
|
fam_ops->mc0_mce = k8_mc0_mce;
|
|
|
|
fam_ops->mc1_mce = k8_mc1_mce;
|
2012-12-19 05:06:10 +08:00
|
|
|
fam_ops->mc2_mce = k8_mc2_mce;
|
2010-10-06 01:07:16 +08:00
|
|
|
break;
|
|
|
|
|
2010-09-16 21:08:14 +08:00
|
|
|
case 0x12:
|
2012-09-12 00:57:43 +08:00
|
|
|
fam_ops->mc0_mce = f12h_mc0_mce;
|
|
|
|
fam_ops->mc1_mce = k8_mc1_mce;
|
2012-12-19 05:06:10 +08:00
|
|
|
fam_ops->mc2_mce = k8_mc2_mce;
|
2010-09-16 21:08:14 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-18 21:11:35 +08:00
|
|
|
case 0x14:
|
2012-12-19 05:06:11 +08:00
|
|
|
fam_ops->mc0_mce = cat_mc0_mce;
|
|
|
|
fam_ops->mc1_mce = cat_mc1_mce;
|
2012-12-19 05:06:10 +08:00
|
|
|
fam_ops->mc2_mce = k8_mc2_mce;
|
2010-08-18 21:11:35 +08:00
|
|
|
break;
|
|
|
|
|
2010-09-18 01:11:47 +08:00
|
|
|
case 0x15:
|
2014-07-14 22:58:19 +08:00
|
|
|
xec_mask = c->x86_model == 0x60 ? 0x3f : 0x1f;
|
|
|
|
|
2012-09-12 00:57:43 +08:00
|
|
|
fam_ops->mc0_mce = f15h_mc0_mce;
|
|
|
|
fam_ops->mc1_mce = f15h_mc1_mce;
|
2012-12-19 05:06:10 +08:00
|
|
|
fam_ops->mc2_mce = f15h_mc2_mce;
|
2010-09-18 01:11:47 +08:00
|
|
|
break;
|
|
|
|
|
2012-12-19 05:06:11 +08:00
|
|
|
case 0x16:
|
|
|
|
xec_mask = 0x1f;
|
|
|
|
fam_ops->mc0_mce = cat_mc0_mce;
|
|
|
|
fam_ops->mc1_mce = cat_mc1_mce;
|
|
|
|
fam_ops->mc2_mce = f16h_mc2_mce;
|
|
|
|
break;
|
|
|
|
|
2016-03-07 21:02:18 +08:00
|
|
|
case 0x17:
|
2018-09-27 22:31:28 +08:00
|
|
|
case 0x18:
|
2016-03-07 21:02:18 +08:00
|
|
|
xec_mask = 0x3f;
|
2016-05-11 20:58:28 +08:00
|
|
|
if (!boot_cpu_has(X86_FEATURE_SMCA)) {
|
2016-03-07 21:02:18 +08:00
|
|
|
printk(KERN_WARNING "Decoding supported only on Scalable MCA processors.\n");
|
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2010-08-18 21:11:35 +08:00
|
|
|
default:
|
2012-04-04 20:21:02 +08:00
|
|
|
printk(KERN_WARNING "Huh? What family is it: 0x%x?!\n", c->x86);
|
2016-03-07 21:02:18 +08:00
|
|
|
goto err_out;
|
2010-08-18 21:11:35 +08:00
|
|
|
}
|
|
|
|
|
2010-09-06 21:05:45 +08:00
|
|
|
pr_info("MCE: In-kernel MCE decoding enabled.\n");
|
|
|
|
|
2011-12-04 22:12:09 +08:00
|
|
|
mce_register_decode_chain(&amd_mce_dec_nb);
|
2009-10-01 22:14:32 +08:00
|
|
|
|
|
|
|
return 0;
|
2016-03-07 21:02:18 +08:00
|
|
|
|
|
|
|
err_out:
|
|
|
|
kfree(fam_ops);
|
|
|
|
fam_ops = NULL;
|
|
|
|
return -EINVAL;
|
2009-10-01 22:14:32 +08:00
|
|
|
}
|
|
|
|
early_initcall(mce_amd_init);
|
2009-10-02 21:31:48 +08:00
|
|
|
|
|
|
|
#ifdef MODULE
|
|
|
|
static void __exit mce_amd_exit(void)
|
|
|
|
{
|
2011-12-04 22:12:09 +08:00
|
|
|
mce_unregister_decode_chain(&amd_mce_dec_nb);
|
2010-08-18 21:11:35 +08:00
|
|
|
kfree(fam_ops);
|
2009-10-02 21:31:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
MODULE_DESCRIPTION("AMD MCE decoder");
|
|
|
|
MODULE_ALIAS("edac-mce-amd");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
module_exit(mce_amd_exit);
|
|
|
|
#endif
|