cregit-Linux how code gets into the kernel

Release 4.11 drivers/edac/mce_amd.c

Directory: drivers/edac
#include <linux/module.h>
#include <linux/slab.h>

#include "mce_amd.h"


static struct amd_decoder_ops *fam_ops;


static u8 xec_mask	 = 0xf;


static bool report_gart_errors;

static void (*decode_dram_ecc)(int node_id, struct mce *m);


void amd_report_gart_errors(bool v) { report_gart_errors = v; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov12100.00%1100.00%
Total12100.00%1100.00%

EXPORT_SYMBOL_GPL(amd_report_gart_errors);
void amd_register_ecc_decoder(void (*f)(int, struct mce *)) { decode_dram_ecc = f; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov2195.45%266.67%
Yazen Ghannam14.55%133.33%
Total22100.00%3100.00%

EXPORT_SYMBOL_GPL(amd_register_ecc_decoder);
void amd_unregister_ecc_decoder(void (*f)(int, struct mce *)) { if (decode_dram_ecc) { WARN_ON(decode_dram_ecc != f); decode_dram_ecc = NULL; } }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov3291.43%266.67%
Yazen Ghannam38.57%133.33%
Total35100.00%3100.00%

EXPORT_SYMBOL_GPL(amd_unregister_ecc_decoder); /* * string representation for the different MCA reported error types, see F3x48 * or MSR0000_0411. */ /* transaction type */ static const char * const tt_msgs[] = { "INSN", "DATA", "GEN", "RESV" }; /* cache level */ static const char * const ll_msgs[] = { "RESV", "L1", "L2", "L3/GEN" }; /* memory transaction type */ static const char * const rrrr_msgs[] = { "GEN", "RD", "WR", "DRD", "DWR", "IRD", "PRF", "EV", "SNP" }; /* participating processor */ const char * const pp_msgs[] = { "SRC", "RES", "OBS", "GEN" }; EXPORT_SYMBOL_GPL(pp_msgs); /* request timeout */ static const char * const to_msgs[] = { "no timeout", "timed out" }; /* memory or i/o */ static const char * const ii_msgs[] = { "MEM", "RESV", "IO", "GEN" }; /* internal error type */ static const char * const uu_msgs[] = { "RESV", "RESV", "HWA", "RESV" }; static const char * const f15h_mc1_mce_desc[] = { "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 */ "Microcode Patch Buffer", /* xec = 010 */ "uop queue", "insn buffer", "predecode buffer", "fetch address FIFO", "dispatch uop queue" }; static const char * const f15h_mc2_mce_desc[] = { "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", "VB Data ECC or parity error", "L2 Tag ECC error", /* xec = 0x10 */ "Hard L2 Tag ECC error", "Multiple hits on L2 tag", "XAB parity error", "PRB address parity error" }; static const char * const mc4_mce_desc[] = { "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" }; static const char * const mc5_mce_desc[] = { "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", "DE error occurred", "Retire status queue" }; static const char * const mc6_mce_desc[] = { "Hardware Assertion", "Free List", "Physical Register File", "Retire Queue", "Scheduler table", "Status Register File", }; /* Scalable MCA error strings */ static const char * const smca_ls_mce_desc[] = { "Load queue parity", "Store queue parity", "Miss address buffer payload parity", "L1 TLB parity", "Reserved", "DC tag error type 6", "DC tag error type 1", "Internal error type 1", "Internal error type 2", "Sys Read data error thread 0", "Sys read data error thread 1", "DC tag error type 2", "DC data error type 1 (poison comsumption)", "DC data error type 2", "DC data error type 3", "DC tag error type 4", "L2 TLB parity", "PDC parity error", "DC tag error type 3", "DC tag error type 5", "L2 fill data error", }; static const char * const smca_if_mce_desc[] = { "microtag probe port parity error", "IC microtag or full tag multi-hit error", "IC full tag parity", "IC data array parity", "Decoupling queue phys addr parity error", "L0 ITLB parity error", "L1 ITLB parity error", "L2 ITLB parity error", "BPQ snoop parity on Thread 0", "BPQ snoop parity on Thread 1", "L1 BTB multi-match error", "L2 BTB multi-match error", "L2 Cache Response Poison error", "System Read Data error", }; static const char * const smca_l2_mce_desc[] = { "L2M tag multi-way-hit error", "L2M tag ECC error", "L2M data ECC error", "HW assert", }; static const char * const smca_de_mce_desc[] = { "uop cache tag parity error", "uop cache data parity error", "Insn buffer parity error", "uop queue parity error", "Insn dispatch queue parity error", "Fetch address FIFO parity", "Patch RAM data parity", "Patch RAM sequencer parity", "uop buffer parity" }; static const char * const smca_ex_mce_desc[] = { "Watchdog timeout error", "Phy register file parity", "Flag register file parity", "Immediate displacement register file parity", "Address generator payload parity", "EX payload parity", "Checkpoint queue parity", "Retire dispatch queue parity", "Retire status queue parity error", "Scheduling queue parity error", "Branch buffer queue parity error", }; static const char * const smca_fp_mce_desc[] = { "Physical register file parity", "Freelist parity error", "Schedule queue parity", "NSQ parity error", "Retire queue parity", "Status register file parity", "Hardware assertion", }; static const char * const smca_l3_mce_desc[] = { "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", "XI parity, L3 fill done channel error", "L3 victim queue parity", "L3 HW assert", }; static const char * const smca_cs_mce_desc[] = { "Illegal request from transport layer", "Address violation", "Security violation", "Illegal response from transport layer", "Unexpected response", "Parity error on incoming request or probe response data", "Parity error on incoming read response data", "Atomic request parity", "ECC error on probe filter access", }; static const char * const smca_pie_mce_desc[] = { "HW assert", "Internal PIE register security violation", "Error on GMI link", "Poison data written to internal PIE register", }; static const char * const smca_umc_mce_desc[] = { "DRAM ECC error", "Data poison error on DRAM", "SDP parity error", "Advanced peripheral bus error", "Command/address parity error", "Write data CRC error", }; static const char * const smca_pb_mce_desc[] = { "Parameter Block RAM ECC error", }; static const char * const smca_psp_mce_desc[] = { "PSP RAM ECC or parity error", }; static const char * const smca_smu_mce_desc[] = { "SMU RAM ECC or parity error", }; 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) }, [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) }, [SMCA_SMU] = { smca_smu_mce_desc, ARRAY_SIZE(smca_smu_mce_desc) }, };
static bool f12h_mc0_mce(u16 ec, u8 xec) { bool ret = false; if (MEM_ERROR(ec)) { u8 ll = LL(ec); ret = true; if (ll == LL_L2) pr_cont("during L1 linefill from L2.\n"); else if (ll == LL_L1) pr_cont("Data/Tag %s error.\n", R4_MSG(ec)); else ret = false; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov74100.00%5100.00%
Total74100.00%5100.00%


static bool f10h_mc0_mce(u16 ec, u8 xec) { if (R4(ec) == R4_GEN && LL(ec) == LL_L1) { pr_cont("during data scrub.\n"); return true; } return f12h_mc0_mce(ec, xec); }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov46100.00%4100.00%
Total46100.00%4100.00%


static bool k8_mc0_mce(u16 ec, u8 xec) { if (BUS_ERROR(ec)) { pr_cont("during system linefill.\n"); return true; } return f10h_mc0_mce(ec, xec); }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov37100.00%4100.00%
Total37100.00%4100.00%


static bool cat_mc0_mce(u16 ec, u8 xec) { u8 r4 = R4(ec); bool ret = true; if (MEM_ERROR(ec)) { if (TT(ec) != TT_DATA || LL(ec) != LL_L1) 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)) { if ((II(ec) != II_MEM && II(ec) != II_IO) || LL(ec) != LL_LG) 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; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov19499.49%480.00%
Jacob Shin10.51%120.00%
Total195100.00%5100.00%


static bool f15h_mc0_mce(u16 ec, u8 xec) { 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) pr_cont("System Read Data Error.\n"); else pr_cont(" Internal error condition type %d.\n", xec); } else if (INT_ERROR(ec)) { if (xec <= 0x1f) pr_cont("Hardware Assert.\n"); else ret = false; } else ret = false; return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov13083.33%375.00%
Aravind Gopalakrishnan2616.67%125.00%
Total156100.00%4100.00%


static void decode_mc0_mce(struct mce *m) { u16 ec = EC(m->status); u8 xec = XEC(m->status, xec_mask); pr_emerg(HW_ERR "MC0 Error: "); /* TLB error signatures are the same across families */ if (TLB_ERROR(ec)) { if (TT(ec) == TT_DATA) { pr_cont("%s TLB %s.\n", LL_MSG(ec), ((xec == 2) ? "locked miss" : (xec ? "multimatch" : "parity"))); return; } } else if (fam_ops->mc0_mce(ec, xec)) ; else pr_emerg(HW_ERR "Corrupted MC0 MCE info?\n"); }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov109100.00%7100.00%
Total109100.00%7100.00%


static bool k8_mc1_mce(u16 ec, u8 xec) { u8 ll = LL(ec); bool ret = true; if (!MEM_ERROR(ec)) return false; if (ll == 0x2) pr_cont("during a linefill from L2.\n"); else if (ll == 0x1) { switch (R4(ec)) { case R4_IRD: pr_cont("Parity error during data load.\n"); break; 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; } } else ret = false; return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov106100.00%6100.00%
Total106100.00%6100.00%


static bool cat_mc1_mce(u16 ec, u8 xec) { u8 r4 = R4(ec); bool ret = true; 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; return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov7168.93%480.00%
Jacob Shin3231.07%120.00%
Total103100.00%5100.00%


static bool f15h_mc1_mce(u16 ec, u8 xec) { bool ret = true; if (!MEM_ERROR(ec)) return false; switch (xec) { case 0x0 ... 0xa: pr_cont("%s.\n", f15h_mc1_mce_desc[xec]); break; case 0xd: pr_cont("%s.\n", f15h_mc1_mce_desc[xec-2]); break; case 0x10: pr_cont("%s.\n", f15h_mc1_mce_desc[xec-4]); break; case 0x11 ... 0x15: pr_cont("Decoder %s parity error.\n", f15h_mc1_mce_desc[xec-4]); break; default: ret = false; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov10799.07%375.00%
Aravind Gopalakrishnan10.93%125.00%
Total108100.00%4100.00%


static void decode_mc1_mce(struct mce *m) { u16 ec = EC(m->status); u8 xec = XEC(m->status, xec_mask); pr_emerg(HW_ERR "MC1 Error: "); if (TLB_ERROR(ec)) pr_cont("%s TLB %s.\n", LL_MSG(ec), (xec ? "multimatch" : "parity error")); else if (BUS_ERROR(ec)) { bool k8 = (boot_cpu_data.x86 == 0xf && (m->status & BIT_64(58))); pr_cont("during %s.\n", (k8 ? "system linefill" : "NB data read")); } else if (INT_ERROR(ec)) { if (xec <= 0x3f) pr_cont("Hardware Assert.\n"); else goto wrong_mc1_mce; } else if (fam_ops->mc1_mce(ec, xec)) ; else goto wrong_mc1_mce; return; wrong_mc1_mce: pr_emerg(HW_ERR "Corrupted MC1 MCE info?\n"); }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov12980.62%888.89%
Aravind Gopalakrishnan3119.38%111.11%
Total160100.00%9100.00%


static bool k8_mc2_mce(u16 ec, u8 xec) { bool ret = true; 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)) pr_cont(": %s error in the L2 cache tags.\n", R4_MSG(ec)); else if (xec == 0x0) { if (TLB_ERROR(ec)) pr_cont("%s error in a Page Descriptor Cache or Guest TLB.\n", TT_MSG(ec)); else if (BUS_ERROR(ec)) pr_cont(": %s/ECC error in data read from NB: %s.\n", R4_MSG(ec), PP_MSG(ec)); else if (MEM_ERROR(ec)) { u8 r4 = R4(ec); if (r4 >= 0x7) pr_cont(": %s error during data copyback.\n", R4_MSG(ec)); else if (r4 <= 0x1) pr_cont(": %s parity/ECC error during data " "access from L2.\n", R4_MSG(ec)); else ret = false; } else ret = false; } else ret = false; return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov15987.85%375.00%
Jacob Shin2212.15%125.00%
Total181100.00%4100.00%


static bool f15h_mc2_mce(u16 ec, u8 xec) { bool ret = true; 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 ret = false; } else if (BUS_ERROR(ec)) { if (xec > 2) ret = false; pr_cont("Error during attempted NB data read.\n"); } else if (MEM_ERROR(ec)) { switch (xec) { case 0x4 ... 0xc: pr_cont("%s.\n", f15h_mc2_mce_desc[xec - 0x4]); break; case 0x10 ... 0x14: pr_cont("%s.\n", f15h_mc2_mce_desc[xec - 0x7]); break; default: ret = false; } } else if (INT_ERROR(ec)) { if (xec <= 0x3f) pr_cont("Hardware Assert.\n"); else ret = false; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov11569.70%250.00%
Aravind Gopalakrishnan2615.76%125.00%
Jacob Shin2414.55%125.00%
Total165100.00%4100.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; }

Contributors

PersonTokensPropCommitsCommitProp
Jacob Shin194100.00%1100.00%
Total194100.00%1100.00%


static void decode_mc2_mce(struct mce *m) { u16 ec = EC(m->status); u8 xec = XEC(m->status, xec_mask); pr_emerg(HW_ERR "MC2 Error: "); if (!fam_ops->mc2_mce(ec, xec)) pr_cont(HW_ERR "Corrupted MC2 MCE info?\n"); }

Contributors

PersonTokensPropCommitsCommitProp
Jacob Shin5087.72%133.33%
Borislav Petkov712.28%266.67%
Total57100.00%3100.00%


static void decode_mc3_mce(struct mce *m) { u16 ec = EC(m->status); u8 xec = XEC(m->status, xec_mask); if (boot_cpu_data.x86 >= 0x14) { pr_emerg("You shouldn't be seeing MC3 MCE on this cpu family," " please report on LKML.\n"); return; } pr_emerg(HW_ERR "MC3 Error"); if (xec == 0x0) { u8 r4 = R4(ec); if (!BUS_ERROR(ec) || (r4 != R4_DRD && r4 != R4_DWR)) goto wrong_mc3_mce; pr_cont(" during %s.\n", R4_MSG(ec)); } else goto wrong_mc3_mce; return; wrong_mc3_mce: pr_emerg(HW_ERR "Corrupted MC3 MCE info?\n"); }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov116100.00%8100.00%
Total116100.00%8100.00%


static void decode_mc4_mce(struct mce *m) { struct cpuinfo_x86 *c = &boot_cpu_data; int node_id = amd_get_nb_id(m->extcpu); u16 ec = EC(m->status); u8 xec = XEC(m->status, 0x1f); u8 offset = 0; pr_emerg(HW_ERR "MC4 Error (node %d): ", node_id); switch (xec) { case 0x0 ... 0xe: /* special handling for DRAM ECCs */ if (xec == 0x0 || xec == 0x8) { /* no ECCs on F11h */ if (c->x86 == 0x11) goto wrong_mc4_mce; pr_cont("%s.\n", mc4_mce_desc[xec]); if (decode_dram_ecc) decode_dram_ecc(node_id, m); return; } 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 goto wrong_mc4_mce; return; case 0x19: if (boot_cpu_data.x86 == 0x15 || boot_cpu_data.x86 == 0x16) pr_cont("Compute Unit Data Error.\n"); else goto wrong_mc4_mce; return; case 0x1c ... 0x1f: offset = 13; break; default: goto wrong_mc4_mce; } pr_cont("%s.\n", mc4_mce_desc[xec - offset]); return; wrong_mc4_mce: pr_emerg(HW_ERR "Corrupted MC4 MCE info?\n"); }

Contributors

PersonTokensPropCommitsCommitProp
Borislav Petkov21096.33%675.00%
Jacob Shin62.75%112.50%
Yazen Ghannam20.92%1