mce_amd.c 16.3 KB
Newer Older
1
#include <linux/module.h>
2 3
#include <linux/slab.h>

B
Borislav Petkov 已提交
4
#include "mce_amd.h"
D
Doug Thompson 已提交
5

6 7
static struct amd_decoder_ops *fam_ops;

8
static u8 xec_mask	 = 0xf;
9 10
static u8 nb_err_cpumask = 0xf;

11
static bool report_gart_errors;
12
static void (*nb_bus_decoder)(int node_id, struct mce *m);
13 14 15 16 17 18 19

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

20
void amd_register_ecc_decoder(void (*f)(int, struct mce *))
21 22 23 24 25
{
	nb_bus_decoder = f;
}
EXPORT_SYMBOL_GPL(amd_register_ecc_decoder);

26
void amd_unregister_ecc_decoder(void (*f)(int, struct mce *))
27 28 29 30 31 32 33 34 35
{
	if (nb_bus_decoder) {
		WARN_ON(nb_bus_decoder != f);

		nb_bus_decoder = NULL;
	}
}
EXPORT_SYMBOL_GPL(amd_unregister_ecc_decoder);

D
Doug Thompson 已提交
36 37 38 39
/*
 * string representation for the different MCA reported error types, see F3x48
 * or MSR0000_0411.
 */
B
Borislav Petkov 已提交
40 41 42

/* transaction type */
const char *tt_msgs[] = { "INSN", "DATA", "GEN", "RESV" };
43
EXPORT_SYMBOL_GPL(tt_msgs);
D
Doug Thompson 已提交
44

B
Borislav Petkov 已提交
45 46
/* cache level */
const char *ll_msgs[] = { "RESV", "L1", "L2", "L3/GEN" };
47
EXPORT_SYMBOL_GPL(ll_msgs);
D
Doug Thompson 已提交
48

B
Borislav Petkov 已提交
49
/* memory transaction type */
D
Doug Thompson 已提交
50
const char *rrrr_msgs[] = {
B
Borislav Petkov 已提交
51
       "GEN", "RD", "WR", "DRD", "DWR", "IRD", "PRF", "EV", "SNP"
D
Doug Thompson 已提交
52
};
53
EXPORT_SYMBOL_GPL(rrrr_msgs);
D
Doug Thompson 已提交
54

B
Borislav Petkov 已提交
55 56
/* participating processor */
const char *pp_msgs[] = { "SRC", "RES", "OBS", "GEN" };
57
EXPORT_SYMBOL_GPL(pp_msgs);
D
Doug Thompson 已提交
58

B
Borislav Petkov 已提交
59 60
/* request timeout */
const char *to_msgs[] = { "no timeout",	"timed out" };
61
EXPORT_SYMBOL_GPL(to_msgs);
D
Doug Thompson 已提交
62

B
Borislav Petkov 已提交
63 64
/* memory or i/o */
const char *ii_msgs[] = { "MEM", "RESV", "IO", "GEN" };
65
EXPORT_SYMBOL_GPL(ii_msgs);
D
Doug Thompson 已提交
66

67 68 69 70 71 72 73 74 75 76 77 78 79
static const char * const f15h_ic_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 */
80
	"Microcode Patch Buffer",			/* xec = 010 */
81 82 83 84 85 86
	"uop queue",
	"insn buffer",
	"predecode buffer",
	"fetch address FIFO"
};

87 88 89 90 91 92 93 94 95
static const char * const f15h_cu_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",
96
	"VB Data ECC or parity error",
97 98 99 100 101 102 103
	"L2 Tag ECC error",				/* xec = 0x10 */
	"Hard L2 Tag ECC error",
	"Multiple hits on L2 tag",
	"XAB parity error",
	"PRB address parity error"
};

104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
static const char *nb_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"
};

126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141
static const char * const fr_ex_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 correctable error could not be corrected"
};

142
static bool f12h_dc_mce(u16 ec, u8 xec)
143
{
144
	bool ret = false;
145

146
	if (MEM_ERROR(ec)) {
147
		u8 ll = LL(ec);
148
		ret = true;
149

150 151 152
		if (ll == LL_L2)
			pr_cont("during L1 linefill from L2.\n");
		else if (ll == LL_L1)
153
			pr_cont("Data/Tag %s error.\n", R4_MSG(ec));
154 155 156 157 158
		else
			ret = false;
	}
	return ret;
}
159

160
static bool f10h_dc_mce(u16 ec, u8 xec)
161
{
162
	if (R4(ec) == R4_GEN && LL(ec) == LL_L1) {
163 164 165
		pr_cont("during data scrub.\n");
		return true;
	}
166
	return f12h_dc_mce(ec, xec);
167 168
}

169
static bool k8_dc_mce(u16 ec, u8 xec)
170 171 172 173 174
{
	if (BUS_ERROR(ec)) {
		pr_cont("during system linefill.\n");
		return true;
	}
175

176
	return f10h_dc_mce(ec, xec);
177 178
}

179
static bool f14h_dc_mce(u16 ec, u8 xec)
180
{
181
	u8 r4	 = R4(ec);
182 183 184 185
	bool ret = true;

	if (MEM_ERROR(ec)) {

186
		if (TT(ec) != TT_DATA || LL(ec) != LL_L1)
187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205
			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)) {

206
		if ((II(ec) != II_MEM && II(ec) != II_IO) || LL(ec) != LL_LG)
207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230
			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;
}

231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268
static bool f15h_dc_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)
269
			pr_cont("System Read Data Error.\n");
270
		else
271
			pr_cont(" Internal error condition type %d.\n", xec);
272 273 274 275 276 277
	} else
		ret = false;

	return ret;
}

278 279
static void amd_decode_dc_mce(struct mce *m)
{
280 281
	u16 ec = EC(m->status);
	u8 xec = XEC(m->status, xec_mask);
282 283 284 285 286

	pr_emerg(HW_ERR "Data Cache Error: ");

	/* TLB error signatures are the same across families */
	if (TLB_ERROR(ec)) {
287
		if (TT(ec) == TT_DATA) {
288
			pr_cont("%s TLB %s.\n", LL_MSG(ec),
289 290
				((xec == 2) ? "locked miss"
					    : (xec ? "multimatch" : "parity")));
291 292
			return;
		}
293 294 295 296
	} else if (fam_ops->dc_mce(ec, xec))
		;
	else
		pr_emerg(HW_ERR "Corrupted DC MCE info?\n");
297 298
}

299
static bool k8_ic_mce(u16 ec, u8 xec)
300
{
301
	u8 ll	 = LL(ec);
302
	bool ret = true;
303

304 305
	if (!MEM_ERROR(ec))
		return false;
306

307 308 309
	if (ll == 0x2)
		pr_cont("during a linefill from L2.\n");
	else if (ll == 0x1) {
310
		switch (R4(ec)) {
311 312 313
		case R4_IRD:
			pr_cont("Parity error during data load.\n");
			break;
314

315 316 317 318 319 320 321 322 323 324 325 326
		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;
		}
327
	} else
328
		ret = false;
329

330 331 332
	return ret;
}

333
static bool f14h_ic_mce(u16 ec, u8 xec)
334
{
335
	u8 r4    = R4(ec);
336
	bool ret = true;
337

338
	if (MEM_ERROR(ec)) {
339
		if (TT(ec) != 0 || LL(ec) != 1)
340 341 342 343 344 345 346 347 348 349 350 351
			ret = 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
			ret = false;
	}
	return ret;
}

352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367
static bool f15h_ic_mce(u16 ec, u8 xec)
{
	bool ret = true;

	if (!MEM_ERROR(ec))
		return false;

	switch (xec) {
	case 0x0 ... 0xa:
		pr_cont("%s.\n", f15h_ic_mce_desc[xec]);
		break;

	case 0xd:
		pr_cont("%s.\n", f15h_ic_mce_desc[xec-2]);
		break;

368 369 370 371 372
	case 0x10:
		pr_cont("%s.\n", f15h_ic_mce_desc[xec-4]);
		break;

	case 0x11 ... 0x14:
373 374 375 376 377 378 379 380 381
		pr_cont("Decoder %s parity error.\n", f15h_ic_mce_desc[xec-4]);
		break;

	default:
		ret = false;
	}
	return ret;
}

382 383
static void amd_decode_ic_mce(struct mce *m)
{
384 385
	u16 ec = EC(m->status);
	u8 xec = XEC(m->status, xec_mask);
386 387 388 389 390 391 392

	pr_emerg(HW_ERR "Instruction Cache Error: ");

	if (TLB_ERROR(ec))
		pr_cont("%s TLB %s.\n", LL_MSG(ec),
			(xec ? "multimatch" : "parity error"));
	else if (BUS_ERROR(ec)) {
393
		bool k8 = (boot_cpu_data.x86 == 0xf && (m->status & BIT_64(58)));
394 395

		pr_cont("during %s.\n", (k8 ? "system linefill" : "NB data read"));
396
	} else if (fam_ops->ic_mce(ec, xec))
397 398 399
		;
	else
		pr_emerg(HW_ERR "Corrupted IC MCE info?\n");
400 401
}

402
static void amd_decode_bu_mce(struct mce *m)
403
{
404 405
	u16 ec = EC(m->status);
	u8 xec = XEC(m->status, xec_mask);
406

B
Borislav Petkov 已提交
407
	pr_emerg(HW_ERR "Bus Unit Error");
408 409 410 411 412 413

	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))
414
		pr_cont(": %s error in the L2 cache tags.\n", R4_MSG(ec));
415 416 417 418 419 420
	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",
421
				R4_MSG(ec), PP_MSG(ec));
422
		else if (MEM_ERROR(ec)) {
423
			u8 r4 = R4(ec);
424

425
			if (r4 >= 0x7)
426
				pr_cont(": %s error during data copyback.\n",
427 428
					R4_MSG(ec));
			else if (r4 <= 0x1)
429
				pr_cont(": %s parity/ECC error during data "
430
					"access from L2.\n", R4_MSG(ec));
431 432 433 434 435 436 437 438 439 440
			else
				goto wrong_bu_mce;
		} else
			goto wrong_bu_mce;
	} else
		goto wrong_bu_mce;

	return;

wrong_bu_mce:
B
Borislav Petkov 已提交
441
	pr_emerg(HW_ERR "Corrupted BU MCE info?\n");
442 443
}

444 445
static void amd_decode_cu_mce(struct mce *m)
{
446 447
	u16 ec = EC(m->status);
	u8 xec = XEC(m->status, xec_mask);
448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483

	pr_emerg(HW_ERR "Combined Unit Error: ");

	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
			goto wrong_cu_mce;
	} else if (BUS_ERROR(ec)) {
		if (xec > 2)
			goto wrong_cu_mce;

		pr_cont("Error during attempted NB data read.\n");
	} else if (MEM_ERROR(ec)) {
		switch (xec) {
		case 0x4 ... 0xc:
			pr_cont("%s.\n", f15h_cu_mce_desc[xec - 0x4]);
			break;

		case 0x10 ... 0x14:
			pr_cont("%s.\n", f15h_cu_mce_desc[xec - 0x7]);
			break;

		default:
			goto wrong_cu_mce;
		}
	}

	return;

wrong_cu_mce:
	pr_emerg(HW_ERR "Corrupted CU MCE info?\n");
}

484
static void amd_decode_ls_mce(struct mce *m)
485
{
486 487
	u16 ec = EC(m->status);
	u8 xec = XEC(m->status, xec_mask);
488

489
	if (boot_cpu_data.x86 >= 0x14) {
490 491 492 493
		pr_emerg("You shouldn't be seeing an LS MCE on this cpu family,"
			 " please report on LKML.\n");
		return;
	}
494

B
Borislav Petkov 已提交
495
	pr_emerg(HW_ERR "Load Store Error");
496 497

	if (xec == 0x0) {
498
		u8 r4 = R4(ec);
499

500
		if (!BUS_ERROR(ec) || (r4 != R4_DRD && r4 != R4_DWR))
501 502
			goto wrong_ls_mce;

503
		pr_cont(" during %s.\n", R4_MSG(ec));
504 505 506
	} else
		goto wrong_ls_mce;

507 508 509
	return;

wrong_ls_mce:
B
Borislav Petkov 已提交
510
	pr_emerg(HW_ERR "Corrupted LS MCE info?\n");
511 512
}

513
void amd_decode_nb_mce(struct mce *m)
514
{
515 516 517 518 519
	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;
520

521
	pr_emerg(HW_ERR "Northbridge Error (node %d): ", node_id);
522

523 524
	switch (xec) {
	case 0x0 ... 0xe:
525

526 527 528 529 530
		/* special handling for DRAM ECCs */
		if (xec == 0x0 || xec == 0x8) {
			/* no ECCs on F11h */
			if (c->x86 == 0x11)
				goto wrong_nb_mce;
531

532
			pr_cont("%s.\n", nb_mce_desc[xec]);
533

534 535 536 537
			if (nb_bus_decoder)
				nb_bus_decoder(node_id, m);
			return;
		}
538 539 540 541 542 543 544 545
		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
546 547
			goto wrong_nb_mce;
		return;
548

549 550 551 552
	case 0x19:
		if (boot_cpu_data.x86 == 0x15)
			pr_cont("Compute Unit Data Error.\n");
		else
553 554
			goto wrong_nb_mce;
		return;
555

556
	case 0x1c ... 0x1f:
557
		offset = 13;
558 559 560 561
		break;

	default:
		goto wrong_nb_mce;
562
	}
563

564
	pr_cont("%s.\n", nb_mce_desc[xec - offset]);
565 566 567 568
	return;

wrong_nb_mce:
	pr_emerg(HW_ERR "Corrupted NB MCE info?\n");
569 570 571
}
EXPORT_SYMBOL_GPL(amd_decode_nb_mce);

572
static void amd_decode_fr_mce(struct mce *m)
B
Borislav Petkov 已提交
573
{
574
	struct cpuinfo_x86 *c = &boot_cpu_data;
575
	u8 xec = XEC(m->status, xec_mask);
576 577

	if (c->x86 == 0xf || c->x86 == 0x11)
B
Borislav Petkov 已提交
578 579
		goto wrong_fr_mce;

580 581 582 583 584 585 586 587 588 589 590 591 592 593
	if (c->x86 != 0x15 && xec != 0x0)
		goto wrong_fr_mce;

	pr_emerg(HW_ERR "%s Error: ",
		 (c->x86 == 0x15 ? "Execution Unit" : "FIROB"));

	if (xec == 0x0 || xec == 0xc)
		pr_cont("%s.\n", fr_ex_mce_desc[xec]);
	else if (xec < 0xd)
		pr_cont("%s parity error.\n", fr_ex_mce_desc[xec]);
	else
		goto wrong_fr_mce;

	return;
B
Borislav Petkov 已提交
594 595 596

wrong_fr_mce:
	pr_emerg(HW_ERR "Corrupted FR MCE info?\n");
B
Borislav Petkov 已提交
597 598
}

599 600
static void amd_decode_fp_mce(struct mce *m)
{
601
	u8 xec = XEC(m->status, xec_mask);
602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638

	pr_emerg(HW_ERR "Floating Point Unit Error: ");

	switch (xec) {
	case 0x1:
		pr_cont("Free List");
		break;

	case 0x2:
		pr_cont("Physical Register File");
		break;

	case 0x3:
		pr_cont("Retire Queue");
		break;

	case 0x4:
		pr_cont("Scheduler table");
		break;

	case 0x5:
		pr_cont("Status Register File");
		break;

	default:
		goto wrong_fp_mce;
		break;
	}

	pr_cont(" parity error.\n");

	return;

wrong_fp_mce:
	pr_emerg(HW_ERR "Corrupted FP MCE info?\n");
}

B
Borislav Petkov 已提交
639
static inline void amd_decode_err_code(u16 ec)
640
{
641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656

	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");
657 658
}

659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674
/*
 * Filter out unwanted MCE signatures here.
 */
static bool amd_filter_mce(struct mce *m)
{
	u8 xec = (m->status >> 16) & 0x1f;

	/*
	 * NB GART TLB error reporting is disabled by default.
	 */
	if (m->bank == 4 && xec == 0x5 && !report_gart_errors)
		return true;

	return false;
}

B
Borislav Petkov 已提交
675
int amd_decode_mce(struct notifier_block *nb, unsigned long val, void *data)
676
{
677
	struct mce *m = (struct mce *)data;
678
	struct cpuinfo_x86 *c = &boot_cpu_data;
679
	int ecc;
680

681 682 683
	if (amd_filter_mce(m))
		return NOTIFY_STOP;

684
	pr_emerg(HW_ERR "CPU:%d\tMC%d_STATUS[%s|%s|%s|%s|%s",
685
		m->extcpu, m->bank,
686 687 688 689 690
		((m->status & MCI_STATUS_OVER)	? "Over"  : "-"),
		((m->status & MCI_STATUS_UC)	? "UE"	  : "CE"),
		((m->status & MCI_STATUS_MISCV)	? "MiscV" : "-"),
		((m->status & MCI_STATUS_PCC)	? "PCC"	  : "-"),
		((m->status & MCI_STATUS_ADDRV)	? "AddrV" : "-"));
691

692 693
	if (c->x86 == 0x15)
		pr_cont("|%s|%s",
694 695
			((m->status & BIT_64(44)) ? "Deferred" : "-"),
			((m->status & BIT_64(43)) ? "Poison"   : "-"));
696

697
	/* do the two bits[14:13] together */
698
	ecc = (m->status >> 45) & 0x3;
699
	if (ecc)
700 701 702
		pr_cont("|%sECC", ((ecc == 2) ? "C" : "U"));

	pr_cont("]: 0x%016llx\n", m->status);
703

704 705
	if (m->status & MCI_STATUS_ADDRV)
		pr_emerg(HW_ERR "\tMC%d_ADDR: 0x%016llx\n", m->bank, m->addr);
706

707 708
	switch (m->bank) {
	case 0:
709
		amd_decode_dc_mce(m);
710
		break;
711

712
	case 1:
713
		amd_decode_ic_mce(m);
714 715
		break;

716
	case 2:
717
		if (c->x86 == 0x15)
718 719 720
			amd_decode_cu_mce(m);
		else
			amd_decode_bu_mce(m);
721 722
		break;

723
	case 3:
724
		amd_decode_ls_mce(m);
725 726
		break;

727
	case 4:
728
		amd_decode_nb_mce(m);
729 730
		break;

B
Borislav Petkov 已提交
731
	case 5:
732
		amd_decode_fr_mce(m);
B
Borislav Petkov 已提交
733 734
		break;

735 736 737 738
	case 6:
		amd_decode_fp_mce(m);
		break;

739 740
	default:
		break;
741
	}
742 743

	amd_decode_err_code(m->status & 0xffff);
744 745

	return NOTIFY_STOP;
746
}
B
Borislav Petkov 已提交
747
EXPORT_SYMBOL_GPL(amd_decode_mce);
748

749 750 751 752
static struct notifier_block amd_mce_dec_nb = {
	.notifier_call	= amd_decode_mce,
};

753 754
static int __init mce_amd_init(void)
{
755 756 757
	struct cpuinfo_x86 *c = &boot_cpu_data;

	if (c->x86_vendor != X86_VENDOR_AMD)
758 759
		return 0;

760 761 762
	if ((c->x86 < 0xf || c->x86 > 0x12) &&
	    (c->x86 != 0x14 || c->x86_model > 0xf) &&
	    (c->x86 != 0x15 || c->x86_model > 0xf))
763 764
		return 0;

765 766 767 768
	fam_ops = kzalloc(sizeof(struct amd_decoder_ops), GFP_KERNEL);
	if (!fam_ops)
		return -ENOMEM;

769
	switch (c->x86) {
770 771
	case 0xf:
		fam_ops->dc_mce = k8_dc_mce;
772
		fam_ops->ic_mce = k8_ic_mce;
773 774 775 776
		break;

	case 0x10:
		fam_ops->dc_mce = f10h_dc_mce;
777
		fam_ops->ic_mce = k8_ic_mce;
778 779
		break;

780 781 782 783 784
	case 0x11:
		fam_ops->dc_mce = k8_dc_mce;
		fam_ops->ic_mce = k8_ic_mce;
		break;

785 786
	case 0x12:
		fam_ops->dc_mce = f12h_dc_mce;
787
		fam_ops->ic_mce = k8_ic_mce;
788 789
		break;

790
	case 0x14:
791
		nb_err_cpumask  = 0x3;
792
		fam_ops->dc_mce = f14h_dc_mce;
793
		fam_ops->ic_mce = f14h_ic_mce;
794 795
		break;

796 797
	case 0x15:
		xec_mask = 0x1f;
798
		fam_ops->dc_mce = f15h_dc_mce;
799
		fam_ops->ic_mce = f15h_ic_mce;
800 801
		break;

802
	default:
803
		printk(KERN_WARNING "Huh? What family is that: %d?!\n", c->x86);
804 805 806 807
		kfree(fam_ops);
		return -EINVAL;
	}

808 809
	pr_info("MCE: In-kernel MCE decoding enabled.\n");

810
	mce_register_decode_chain(&amd_mce_dec_nb);
811 812 813 814

	return 0;
}
early_initcall(mce_amd_init);
815 816 817 818

#ifdef MODULE
static void __exit mce_amd_exit(void)
{
819
	mce_unregister_decode_chain(&amd_mce_dec_nb);
820
	kfree(fam_ops);
821 822 823 824 825 826 827
}

MODULE_DESCRIPTION("AMD MCE decoder");
MODULE_ALIAS("edac-mce-amd");
MODULE_LICENSE("GPL");
module_exit(mce_amd_exit);
#endif