forked from mchehab/rasdaemon
-
Notifications
You must be signed in to change notification settings - Fork 0
/
mce-intel-nehalem.c
150 lines (133 loc) · 5 KB
/
mce-intel-nehalem.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
/*
* The code below came from Andi Kleen/Intel/SuSe mcelog code,
* released under GNU Public General License, v.2
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <string.h>
#include <stdio.h>
#include "ras-mce-handler.h"
#include "bitfield.h"
/* See IA32 SDM Vol3B Appendix E.3.2 ff */
/* MC1_STATUS error */
static struct field qpi_status[] = {
SBITFIELD(16, "QPI header had bad parity"),
SBITFIELD(17, "QPI Data packet had bad parity"),
SBITFIELD(18, "Number of QPI retries exceeded"),
SBITFIELD(19, "Received QPI data packet that was poisoned by sender"),
SBITFIELD(20, "QPI reserved 20"),
SBITFIELD(21, "QPI reserved 21"),
SBITFIELD(22, "QPI received unsupported message encoding"),
SBITFIELD(23, "QPI credit type is not supported"),
SBITFIELD(24, "Sender sent too many QPI flits to the receiver"),
SBITFIELD(25, "QPI Sender sent a failed response to receiver"),
SBITFIELD(26, "Clock jitter detected in internal QPI clocking"),
{}
};
static struct field qpi_misc[] = {
SBITFIELD(14, "QPI misc reserved 14"),
SBITFIELD(15, "QPI misc reserved 15"),
SBITFIELD(24, "QPI Interleave/Head Indication Bit (IIB)"),
{}
};
static struct numfield qpi_numbers[] = {
HEXNUMBER(0, 7, "QPI class and opcode of packet with error"),
HEXNUMBER(8, 13, "QPI Request Transaction ID"),
NUMBERFORCE(16, 18, "QPI Requestor/Home Node ID (RHNID)"),
HEXNUMBER(19, 23, "QPI miscreserved 19-23"),
{},
};
static struct field nhm_memory_status[] = {
SBITFIELD(16, "Memory read ECC error"),
SBITFIELD(17, "Memory ECC error occurred during scrub"),
SBITFIELD(18, "Memory write parity error"),
SBITFIELD(19, "Memory error in half of redundant memory"),
SBITFIELD(20, "Memory reserved 20"),
SBITFIELD(21, "Memory access out of range"),
SBITFIELD(22, "Memory internal RTID invalid"),
SBITFIELD(23, "Memory address parity error"),
SBITFIELD(24, "Memory byte enable parity error"),
{}
};
static struct numfield nhm_memory_status_numbers[] = {
HEXNUMBER(25, 37, "Memory MISC reserved 25..37"),
NUMBERFORCE(38, 52, "Memory corrected error count (CORE_ERR_CNT)"),
HEXNUMBER(53, 56, "Memory MISC reserved 53..56"),
{}
};
static struct numfield nhm_memory_misc_numbers[] = {
HEXNUMBERFORCE(0, 7, "Memory transaction Tracker ID (RTId)"),
NUMBERFORCE(16, 17, "Memory DIMM ID of error"),
NUMBERFORCE(18, 19, "Memory channel ID of error"),
HEXNUMBERFORCE(32, 63, "Memory ECC syndrome"),
{}
};
static char *internal_errors[] = {
[0x0] = "No Error",
[0x3] = "Reset firmware did not complete",
[0x8] = "Received an invalid CMPD",
[0xa] = "Invalid Power Management Request",
[0xd] = "Invalid S-state transition",
[0x11] = "VID controller does not match POC controller selected",
[0x1a] = "MSID from POC does not match CPU MSID",
};
static struct field internal_error_status[] = {
FIELD(24, internal_errors),
{}
};
static struct numfield internal_error_numbers[] = {
HEXNUMBER(16, 23, "Internal machine check status reserved 16..23"),
HEXNUMBER(32, 56, "Internal machine check status reserved 32..56"),
{},
};
/* Generic architectural memory controller encoding */
void nehalem_decode_model(struct mce_event *e)
{
uint64_t status = e->status;
uint32_t mca = status & 0xffff;
uint64_t misc = e->misc;
unsigned channel, dimm;
if ((mca >> 11) == 1) { /* bus and interconnect QPI */
decode_bitfield(e, status, qpi_status);
if (status & MCI_STATUS_MISCV) {
decode_numfield(e, misc, qpi_numbers);
decode_bitfield(e, misc, qpi_misc);
}
} else if (mca == 0x0001) { /* internal unspecified */
decode_bitfield(e, status, internal_error_status);
decode_numfield(e, status, internal_error_numbers);
} else if ((mca >> 7) == 1) { /* memory controller */
decode_bitfield(e, status, nhm_memory_status);
decode_numfield(e, status, nhm_memory_status_numbers);
if (status & MCI_STATUS_MISCV)
decode_numfield(e, misc, nhm_memory_misc_numbers);
}
if ((((status & 0xffff) >> 7) == 1) && (status & MCI_STATUS_MISCV)) {
channel = EXTRACT(e->misc, 18, 19);
dimm = EXTRACT(e->misc, 16, 17);
mce_snprintf(e->mc_location, "channel=%d, dimm=%d",
channel, dimm);
}
}
/* Only core errors supported. Same as Nehalem */
void xeon75xx_decode_model(struct mce_event *e)
{
uint64_t status = e->status;
uint32_t mca = status & 0xffff;
if (mca == 0x0001) { /* internal unspecified */
decode_bitfield(e, status, internal_error_status);
decode_numfield(e, status, internal_error_numbers);
}
}