xref: /openbmc/linux/arch/alpha/kernel/err_titan.c (revision 4f727ecefefbd180de10e25b3e74c03dce3f1e75)
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  *	linux/arch/alpha/kernel/err_titan.c
4  *
5  *	Copyright (C) 2000 Jeff Wiedemeier (Compaq Computer Corporation)
6  *
7  *	Error handling code supporting TITAN systems
8  */
9 
10 #include <linux/init.h>
11 #include <linux/pci.h>
12 #include <linux/sched.h>
13 
14 #include <asm/io.h>
15 #include <asm/core_titan.h>
16 #include <asm/hwrpb.h>
17 #include <asm/smp.h>
18 #include <asm/err_common.h>
19 #include <asm/err_ev6.h>
20 #include <asm/irq_regs.h>
21 
22 #include "err_impl.h"
23 #include "proto.h"
24 
25 
26 static int
27 titan_parse_c_misc(u64 c_misc, int print)
28 {
29 #ifdef CONFIG_VERBOSE_MCHECK
30 	char *src;
31 	int nxs = 0;
32 #endif
33 	int status = MCHK_DISPOSITION_REPORT;
34 
35 #define TITAN__CCHIP_MISC__NXM		(1UL << 28)
36 #define TITAN__CCHIP_MISC__NXS__S	(29)
37 #define TITAN__CCHIP_MISC__NXS__M	(0x7)
38 
39 	if (!(c_misc & TITAN__CCHIP_MISC__NXM))
40 		return MCHK_DISPOSITION_UNKNOWN_ERROR;
41 
42 #ifdef CONFIG_VERBOSE_MCHECK
43 	if (!print)
44 		return status;
45 
46 	nxs = EXTRACT(c_misc, TITAN__CCHIP_MISC__NXS);
47 	switch(nxs) {
48 	case 0:	/* CPU 0 */
49 	case 1:	/* CPU 1 */
50 	case 2:	/* CPU 2 */
51 	case 3:	/* CPU 3 */
52 		src = "CPU";
53 		/* num is already the CPU number */
54 		break;
55 	case 4:	/* Pchip 0 */
56 	case 5:	/* Pchip 1 */
57 		src = "Pchip";
58 		nxs -= 4;
59 		break;
60 	default:/* reserved */
61 		src = "Unknown, NXS =";
62 		/* leave num untouched */
63 		break;
64 	}
65 
66 	printk("%s    Non-existent memory access from: %s %d\n",
67 	       err_print_prefix, src, nxs);
68 #endif /* CONFIG_VERBOSE_MCHECK */
69 
70 	return status;
71 }
72 
73 static int
74 titan_parse_p_serror(int which, u64 serror, int print)
75 {
76 	int status = MCHK_DISPOSITION_REPORT;
77 
78 #ifdef CONFIG_VERBOSE_MCHECK
79 	static const char * const serror_src[] = {
80 		"GPCI", "APCI", "AGP HP", "AGP LP"
81 	};
82 	static const char * const serror_cmd[] = {
83 		"DMA Read", "DMA RMW", "SGTE Read", "Reserved"
84 	};
85 #endif /* CONFIG_VERBOSE_MCHECK */
86 
87 #define TITAN__PCHIP_SERROR__LOST_UECC	(1UL << 0)
88 #define TITAN__PCHIP_SERROR__UECC	(1UL << 1)
89 #define TITAN__PCHIP_SERROR__CRE	(1UL << 2)
90 #define TITAN__PCHIP_SERROR__NXIO	(1UL << 3)
91 #define TITAN__PCHIP_SERROR__LOST_CRE	(1UL << 4)
92 #define TITAN__PCHIP_SERROR__ECCMASK	(TITAN__PCHIP_SERROR__UECC |	  \
93 					 TITAN__PCHIP_SERROR__CRE)
94 #define TITAN__PCHIP_SERROR__ERRMASK	(TITAN__PCHIP_SERROR__LOST_UECC | \
95 					 TITAN__PCHIP_SERROR__UECC |	  \
96 					 TITAN__PCHIP_SERROR__CRE |	  \
97 					 TITAN__PCHIP_SERROR__NXIO |	  \
98 					 TITAN__PCHIP_SERROR__LOST_CRE)
99 #define TITAN__PCHIP_SERROR__SRC__S	(52)
100 #define TITAN__PCHIP_SERROR__SRC__M	(0x3)
101 #define TITAN__PCHIP_SERROR__CMD__S	(54)
102 #define TITAN__PCHIP_SERROR__CMD__M	(0x3)
103 #define TITAN__PCHIP_SERROR__SYN__S	(56)
104 #define TITAN__PCHIP_SERROR__SYN__M	(0xff)
105 #define TITAN__PCHIP_SERROR__ADDR__S	(15)
106 #define TITAN__PCHIP_SERROR__ADDR__M	(0xffffffffUL)
107 
108 	if (!(serror & TITAN__PCHIP_SERROR__ERRMASK))
109 		return MCHK_DISPOSITION_UNKNOWN_ERROR;
110 
111 #ifdef CONFIG_VERBOSE_MCHECK
112 	if (!print)
113 		return status;
114 
115 	printk("%s  PChip %d SERROR: %016llx\n",
116 	       err_print_prefix, which, serror);
117 	if (serror & TITAN__PCHIP_SERROR__ECCMASK) {
118 		printk("%s    %sorrectable ECC Error:\n"
119 		       "      Source: %-6s  Command: %-8s  Syndrome: 0x%08x\n"
120 		       "      Address: 0x%llx\n",
121 		       err_print_prefix,
122 		       (serror & TITAN__PCHIP_SERROR__UECC) ? "Unc" : "C",
123 		       serror_src[EXTRACT(serror, TITAN__PCHIP_SERROR__SRC)],
124 		       serror_cmd[EXTRACT(serror, TITAN__PCHIP_SERROR__CMD)],
125 		       (unsigned)EXTRACT(serror, TITAN__PCHIP_SERROR__SYN),
126 		       EXTRACT(serror, TITAN__PCHIP_SERROR__ADDR));
127 	}
128 	if (serror & TITAN__PCHIP_SERROR__NXIO)
129 		printk("%s    Non Existent I/O Error\n", err_print_prefix);
130 	if (serror & TITAN__PCHIP_SERROR__LOST_UECC)
131 		printk("%s    Lost Uncorrectable ECC Error\n",
132 		       err_print_prefix);
133 	if (serror & TITAN__PCHIP_SERROR__LOST_CRE)
134 		printk("%s    Lost Correctable ECC Error\n", err_print_prefix);
135 #endif /* CONFIG_VERBOSE_MCHECK */
136 
137 	return status;
138 }
139 
140 static int
141 titan_parse_p_perror(int which, int port, u64 perror, int print)
142 {
143 	int cmd;
144 	unsigned long addr;
145 	int status = MCHK_DISPOSITION_REPORT;
146 
147 #ifdef CONFIG_VERBOSE_MCHECK
148 	static const char * const perror_cmd[] = {
149 		"Interrupt Acknowledge", "Special Cycle",
150 		"I/O Read",		"I/O Write",
151 		"Reserved",		"Reserved",
152 		"Memory Read",		"Memory Write",
153 		"Reserved",		"Reserved",
154 		"Configuration Read",	"Configuration Write",
155 		"Memory Read Multiple",	"Dual Address Cycle",
156 		"Memory Read Line",	"Memory Write and Invalidate"
157 	};
158 #endif /* CONFIG_VERBOSE_MCHECK */
159 
160 #define TITAN__PCHIP_PERROR__LOST	(1UL << 0)
161 #define TITAN__PCHIP_PERROR__SERR	(1UL << 1)
162 #define TITAN__PCHIP_PERROR__PERR	(1UL << 2)
163 #define TITAN__PCHIP_PERROR__DCRTO	(1UL << 3)
164 #define TITAN__PCHIP_PERROR__SGE	(1UL << 4)
165 #define TITAN__PCHIP_PERROR__APE	(1UL << 5)
166 #define TITAN__PCHIP_PERROR__TA		(1UL << 6)
167 #define TITAN__PCHIP_PERROR__DPE	(1UL << 7)
168 #define TITAN__PCHIP_PERROR__NDS	(1UL << 8)
169 #define TITAN__PCHIP_PERROR__IPTPR	(1UL << 9)
170 #define TITAN__PCHIP_PERROR__IPTPW	(1UL << 10)
171 #define TITAN__PCHIP_PERROR__ERRMASK	(TITAN__PCHIP_PERROR__LOST |	\
172 					 TITAN__PCHIP_PERROR__SERR |	\
173 					 TITAN__PCHIP_PERROR__PERR |	\
174 					 TITAN__PCHIP_PERROR__DCRTO |	\
175 					 TITAN__PCHIP_PERROR__SGE |	\
176 					 TITAN__PCHIP_PERROR__APE |	\
177 					 TITAN__PCHIP_PERROR__TA |	\
178 					 TITAN__PCHIP_PERROR__DPE |	\
179 					 TITAN__PCHIP_PERROR__NDS |	\
180 					 TITAN__PCHIP_PERROR__IPTPR |	\
181 					 TITAN__PCHIP_PERROR__IPTPW)
182 #define TITAN__PCHIP_PERROR__DAC	(1UL << 47)
183 #define TITAN__PCHIP_PERROR__MWIN	(1UL << 48)
184 #define TITAN__PCHIP_PERROR__CMD__S	(52)
185 #define TITAN__PCHIP_PERROR__CMD__M	(0x0f)
186 #define TITAN__PCHIP_PERROR__ADDR__S	(14)
187 #define TITAN__PCHIP_PERROR__ADDR__M	(0x1fffffffful)
188 
189 	if (!(perror & TITAN__PCHIP_PERROR__ERRMASK))
190 		return MCHK_DISPOSITION_UNKNOWN_ERROR;
191 
192 	cmd = EXTRACT(perror, TITAN__PCHIP_PERROR__CMD);
193 	addr = EXTRACT(perror, TITAN__PCHIP_PERROR__ADDR) << 2;
194 
195 	/*
196 	 * Initializing the BIOS on a video card on a bus without
197 	 * a south bridge (subtractive decode agent) can result in
198 	 * master aborts as the BIOS probes the capabilities of the
199 	 * card. XFree86 does such initialization. If the error
200 	 * is a master abort (No DevSel as PCI Master) and the command
201 	 * is an I/O read or write below the address where we start
202 	 * assigning PCI I/O spaces (SRM uses 0x1000), then mark the
203 	 * error as dismissable so starting XFree86 doesn't result
204 	 * in a series of uncorrectable errors being reported. Also
205 	 * dismiss master aborts to VGA frame buffer space
206 	 * (0xA0000 - 0xC0000) and legacy BIOS space (0xC0000 - 0x100000)
207 	 * for the same reason.
208 	 *
209 	 * Also mark the error dismissible if it looks like the right
210 	 * error but only the Lost bit is set. Since the BIOS initialization
211 	 * can cause multiple master aborts and the error interrupt can
212 	 * be handled on a different CPU than the BIOS code is run on,
213 	 * it is possible for a second master abort to occur between the
214 	 * time the PALcode reads PERROR and the time it writes PERROR
215 	 * to acknowledge the error. If this timing happens, a second
216 	 * error will be signalled after the first, and if no additional
217 	 * errors occur, will look like a Lost error with no additional
218 	 * errors on the same transaction as the previous error.
219 	 */
220 	if (((perror & TITAN__PCHIP_PERROR__NDS) ||
221 	     ((perror & TITAN__PCHIP_PERROR__ERRMASK) ==
222 	      TITAN__PCHIP_PERROR__LOST)) &&
223 	    ((((cmd & 0xE) == 2) && (addr < 0x1000)) ||
224 	     (((cmd & 0xE) == 6) && (addr >= 0xA0000) && (addr < 0x100000)))) {
225 		status = MCHK_DISPOSITION_DISMISS;
226 	}
227 
228 #ifdef CONFIG_VERBOSE_MCHECK
229 	if (!print)
230 		return status;
231 
232 	printk("%s  PChip %d %cPERROR: %016llx\n",
233 	       err_print_prefix, which,
234 	       port ? 'A' : 'G', perror);
235 	if (perror & TITAN__PCHIP_PERROR__IPTPW)
236 		printk("%s    Invalid Peer-to-Peer Write\n", err_print_prefix);
237 	if (perror & TITAN__PCHIP_PERROR__IPTPR)
238 		printk("%s    Invalid Peer-to-Peer Read\n", err_print_prefix);
239 	if (perror & TITAN__PCHIP_PERROR__NDS)
240 		printk("%s    No DEVSEL as PCI Master [Master Abort]\n",
241 		       err_print_prefix);
242 	if (perror & TITAN__PCHIP_PERROR__DPE)
243 		printk("%s    Data Parity Error\n", err_print_prefix);
244 	if (perror & TITAN__PCHIP_PERROR__TA)
245 		printk("%s    Target Abort\n", err_print_prefix);
246 	if (perror & TITAN__PCHIP_PERROR__APE)
247 		printk("%s    Address Parity Error\n", err_print_prefix);
248 	if (perror & TITAN__PCHIP_PERROR__SGE)
249 		printk("%s    Scatter-Gather Error, Invalid PTE\n",
250 		       err_print_prefix);
251 	if (perror & TITAN__PCHIP_PERROR__DCRTO)
252 		printk("%s    Delayed-Completion Retry Timeout\n",
253 		       err_print_prefix);
254 	if (perror & TITAN__PCHIP_PERROR__PERR)
255 		printk("%s    PERR Asserted\n", err_print_prefix);
256 	if (perror & TITAN__PCHIP_PERROR__SERR)
257 		printk("%s    SERR Asserted\n", err_print_prefix);
258 	if (perror & TITAN__PCHIP_PERROR__LOST)
259 		printk("%s    Lost Error\n", err_print_prefix);
260 	printk("%s      Command: 0x%x - %s\n"
261 		 "      Address: 0x%lx\n",
262 	       err_print_prefix,
263 	       cmd, perror_cmd[cmd],
264 	       addr);
265 	if (perror & TITAN__PCHIP_PERROR__DAC)
266 		printk("%s      Dual Address Cycle\n", err_print_prefix);
267 	if (perror & TITAN__PCHIP_PERROR__MWIN)
268 		printk("%s      Hit in Monster Window\n", err_print_prefix);
269 #endif /* CONFIG_VERBOSE_MCHECK */
270 
271 	return status;
272 }
273 
274 static int
275 titan_parse_p_agperror(int which, u64 agperror, int print)
276 {
277 	int status = MCHK_DISPOSITION_REPORT;
278 #ifdef CONFIG_VERBOSE_MCHECK
279 	int cmd, len;
280 	unsigned long addr;
281 
282 	static const char * const agperror_cmd[] = {
283 		"Read (low-priority)",	"Read (high-priority)",
284 		"Write (low-priority)",	"Write (high-priority)",
285 		"Reserved",		"Reserved",
286 		"Flush",		"Fence"
287 	};
288 #endif /* CONFIG_VERBOSE_MCHECK */
289 
290 #define TITAN__PCHIP_AGPERROR__LOST	(1UL << 0)
291 #define TITAN__PCHIP_AGPERROR__LPQFULL	(1UL << 1)
292 #define TITAN__PCHIP_AGPERROR__HPQFULL	(1UL << 2)
293 #define TITAN__PCHIP_AGPERROR__RESCMD	(1UL << 3)
294 #define TITAN__PCHIP_AGPERROR__IPTE	(1UL << 4)
295 #define TITAN__PCHIP_AGPERROR__PTP	(1UL << 5)
296 #define TITAN__PCHIP_AGPERROR__NOWINDOW	(1UL << 6)
297 #define TITAN__PCHIP_AGPERROR__ERRMASK	(TITAN__PCHIP_AGPERROR__LOST |    \
298 					 TITAN__PCHIP_AGPERROR__LPQFULL | \
299 					 TITAN__PCHIP_AGPERROR__HPQFULL | \
300 					 TITAN__PCHIP_AGPERROR__RESCMD |  \
301 					 TITAN__PCHIP_AGPERROR__IPTE |    \
302 					 TITAN__PCHIP_AGPERROR__PTP |     \
303 					 TITAN__PCHIP_AGPERROR__NOWINDOW)
304 #define TITAN__PCHIP_AGPERROR__DAC	(1UL << 48)
305 #define TITAN__PCHIP_AGPERROR__MWIN	(1UL << 49)
306 #define TITAN__PCHIP_AGPERROR__FENCE	(1UL << 59)
307 #define TITAN__PCHIP_AGPERROR__CMD__S	(50)
308 #define TITAN__PCHIP_AGPERROR__CMD__M	(0x07)
309 #define TITAN__PCHIP_AGPERROR__ADDR__S	(15)
310 #define TITAN__PCHIP_AGPERROR__ADDR__M  (0xffffffffUL)
311 #define TITAN__PCHIP_AGPERROR__LEN__S	(53)
312 #define TITAN__PCHIP_AGPERROR__LEN__M	(0x3f)
313 
314 	if (!(agperror & TITAN__PCHIP_AGPERROR__ERRMASK))
315 		return MCHK_DISPOSITION_UNKNOWN_ERROR;
316 
317 #ifdef CONFIG_VERBOSE_MCHECK
318 	if (!print)
319 		return status;
320 
321 	cmd = EXTRACT(agperror, TITAN__PCHIP_AGPERROR__CMD);
322 	addr = EXTRACT(agperror, TITAN__PCHIP_AGPERROR__ADDR) << 3;
323 	len = EXTRACT(agperror, TITAN__PCHIP_AGPERROR__LEN);
324 
325 	printk("%s  PChip %d AGPERROR: %016llx\n", err_print_prefix,
326 	       which, agperror);
327 	if (agperror & TITAN__PCHIP_AGPERROR__NOWINDOW)
328 		printk("%s    No Window\n", err_print_prefix);
329 	if (agperror & TITAN__PCHIP_AGPERROR__PTP)
330 		printk("%s    Peer-to-Peer set\n", err_print_prefix);
331 	if (agperror & TITAN__PCHIP_AGPERROR__IPTE)
332 		printk("%s    Invalid PTE\n", err_print_prefix);
333 	if (agperror & TITAN__PCHIP_AGPERROR__RESCMD)
334 		printk("%s    Reserved Command\n", err_print_prefix);
335 	if (agperror & TITAN__PCHIP_AGPERROR__HPQFULL)
336 		printk("%s    HP Transaction Received while Queue Full\n",
337 		       err_print_prefix);
338 	if (agperror & TITAN__PCHIP_AGPERROR__LPQFULL)
339 		printk("%s    LP Transaction Received while Queue Full\n",
340 		       err_print_prefix);
341 	if (agperror & TITAN__PCHIP_AGPERROR__LOST)
342 		printk("%s    Lost Error\n", err_print_prefix);
343 	printk("%s      Command: 0x%x - %s, %d Quadwords%s\n"
344 		 "      Address: 0x%lx\n",
345 	       err_print_prefix, cmd, agperror_cmd[cmd], len,
346 	       (agperror & TITAN__PCHIP_AGPERROR__FENCE) ? ", FENCE" : "",
347 	       addr);
348 	if (agperror & TITAN__PCHIP_AGPERROR__DAC)
349 		printk("%s      Dual Address Cycle\n", err_print_prefix);
350 	if (agperror & TITAN__PCHIP_AGPERROR__MWIN)
351 		printk("%s      Hit in Monster Window\n", err_print_prefix);
352 #endif /* CONFIG_VERBOSE_MCHECK */
353 
354 	return status;
355 }
356 
357 static int
358 titan_parse_p_chip(int which, u64 serror, u64 gperror,
359 		   u64 aperror, u64 agperror, int print)
360 {
361 	int status = MCHK_DISPOSITION_UNKNOWN_ERROR;
362 	status |= titan_parse_p_serror(which, serror, print);
363 	status |= titan_parse_p_perror(which, 0, gperror, print);
364 	status |= titan_parse_p_perror(which, 1, aperror, print);
365 	status |= titan_parse_p_agperror(which, agperror, print);
366 	return status;
367 }
368 
369 int
370 titan_process_logout_frame(struct el_common *mchk_header, int print)
371 {
372 	struct el_TITAN_sysdata_mcheck *tmchk =
373 		(struct el_TITAN_sysdata_mcheck *)
374 		((unsigned long)mchk_header + mchk_header->sys_offset);
375 	int status = MCHK_DISPOSITION_UNKNOWN_ERROR;
376 
377 	status |= titan_parse_c_misc(tmchk->c_misc, print);
378 	status |= titan_parse_p_chip(0, tmchk->p0_serror, tmchk->p0_gperror,
379 				     tmchk->p0_aperror, tmchk->p0_agperror,
380 				     print);
381 	status |= titan_parse_p_chip(1, tmchk->p1_serror, tmchk->p1_gperror,
382 				     tmchk->p1_aperror, tmchk->p1_agperror,
383 				     print);
384 
385 	return status;
386 }
387 
388 void
389 titan_machine_check(unsigned long vector, unsigned long la_ptr)
390 {
391 	struct el_common *mchk_header = (struct el_common *)la_ptr;
392 	struct el_TITAN_sysdata_mcheck *tmchk =
393 		(struct el_TITAN_sysdata_mcheck *)
394 		((unsigned long)mchk_header + mchk_header->sys_offset);
395 	u64 irqmask;
396 
397 	/*
398 	 * Mask of Titan interrupt sources which are reported as machine checks
399 	 *
400 	 * 63 - CChip Error
401 	 * 62 - PChip 0 H_Error
402 	 * 61 - PChip 1 H_Error
403 	 * 60 - PChip 0 C_Error
404 	 * 59 - PChip 1 C_Error
405 	 */
406 #define TITAN_MCHECK_INTERRUPT_MASK	0xF800000000000000UL
407 
408 	/*
409 	 * Sync the processor
410 	 */
411 	mb();
412 	draina();
413 
414 	/*
415 	 * Only handle system errors here
416 	 */
417 	if ((vector != SCB_Q_SYSMCHK) && (vector != SCB_Q_SYSERR)) {
418 		ev6_machine_check(vector, la_ptr);
419 		return;
420 	}
421 
422 	/*
423 	 * It's a system error, handle it here
424 	 *
425 	 * The PALcode has already cleared the error, so just parse it
426 	 */
427 
428 	/*
429 	 * Parse the logout frame without printing first. If the only error(s)
430 	 * found are classified as "dismissable", then just dismiss them and
431 	 * don't print any message
432 	 */
433 	if (titan_process_logout_frame(mchk_header, 0) !=
434 	    MCHK_DISPOSITION_DISMISS) {
435 		char *saved_err_prefix = err_print_prefix;
436 		err_print_prefix = KERN_CRIT;
437 
438 		/*
439 		 * Either a nondismissable error was detected or no
440 		 * recognized error was detected  in the logout frame
441 		 * -- report the error in either case
442 		 */
443 		printk("%s"
444 		       "*System %s Error (Vector 0x%x) reported on CPU %d:\n",
445 		       err_print_prefix,
446 		       (vector == SCB_Q_SYSERR)?"Correctable":"Uncorrectable",
447 		       (unsigned int)vector, (int)smp_processor_id());
448 
449 #ifdef CONFIG_VERBOSE_MCHECK
450 		titan_process_logout_frame(mchk_header, alpha_verbose_mcheck);
451 		if (alpha_verbose_mcheck)
452 			dik_show_regs(get_irq_regs(), NULL);
453 #endif /* CONFIG_VERBOSE_MCHECK */
454 
455 		err_print_prefix = saved_err_prefix;
456 
457 		/*
458 		 * Convert any pending interrupts which report as system
459 		 * machine checks to interrupts
460 		 */
461 		irqmask = tmchk->c_dirx & TITAN_MCHECK_INTERRUPT_MASK;
462 		titan_dispatch_irqs(irqmask);
463 	}
464 
465 
466 	/*
467 	 * Release the logout frame
468 	 */
469 	wrmces(0x7);
470 	mb();
471 }
472 
473 /*
474  * Subpacket Annotations
475  */
476 static char *el_titan_pchip0_extended_annotation[] = {
477 	"Subpacket Header", 	"P0_SCTL",	"P0_SERREN",
478 	"P0_APCTL",		"P0_APERREN",	"P0_AGPERREN",
479 	"P0_ASPRST",		"P0_AWSBA0",	"P0_AWSBA1",
480 	"P0_AWSBA2",		"P0_AWSBA3",	"P0_AWSM0",
481 	"P0_AWSM1",		"P0_AWSM2",	"P0_AWSM3",
482 	"P0_ATBA0",		"P0_ATBA1",	"P0_ATBA2",
483 	"P0_ATBA3",		"P0_GPCTL",	"P0_GPERREN",
484 	"P0_GSPRST",		"P0_GWSBA0",	"P0_GWSBA1",
485 	"P0_GWSBA2",		"P0_GWSBA3",	"P0_GWSM0",
486 	"P0_GWSM1",		"P0_GWSM2",	"P0_GWSM3",
487 	"P0_GTBA0",		"P0_GTBA1",	"P0_GTBA2",
488 	"P0_GTBA3",		NULL
489 };
490 static char *el_titan_pchip1_extended_annotation[] = {
491 	"Subpacket Header", 	"P1_SCTL",	"P1_SERREN",
492 	"P1_APCTL",		"P1_APERREN",	"P1_AGPERREN",
493 	"P1_ASPRST",		"P1_AWSBA0",	"P1_AWSBA1",
494 	"P1_AWSBA2",		"P1_AWSBA3",	"P1_AWSM0",
495 	"P1_AWSM1",		"P1_AWSM2",	"P1_AWSM3",
496 	"P1_ATBA0",		"P1_ATBA1",	"P1_ATBA2",
497 	"P1_ATBA3",		"P1_GPCTL",	"P1_GPERREN",
498 	"P1_GSPRST",		"P1_GWSBA0",	"P1_GWSBA1",
499 	"P1_GWSBA2",		"P1_GWSBA3",	"P1_GWSM0",
500 	"P1_GWSM1",		"P1_GWSM2",	"P1_GWSM3",
501 	"P1_GTBA0",		"P1_GTBA1",	"P1_GTBA2",
502 	"P1_GTBA3",		NULL
503 };
504 static char *el_titan_memory_extended_annotation[] = {
505 	"Subpacket Header", 	"AAR0",		"AAR1",
506 	"AAR2",			"AAR3",		"P0_SCTL",
507 	"P0_GPCTL",		"P0_APCTL",	"P1_SCTL",
508 	"P1_GPCTL",		"P1_SCTL",	NULL
509 };
510 
511 static struct el_subpacket_annotation el_titan_annotations[] = {
512 	SUBPACKET_ANNOTATION(EL_CLASS__REGATTA_FAMILY,
513 			     EL_TYPE__REGATTA__TITAN_PCHIP0_EXTENDED,
514 			     1,
515 			     "Titan PChip 0 Extended Frame",
516 			     el_titan_pchip0_extended_annotation),
517 	SUBPACKET_ANNOTATION(EL_CLASS__REGATTA_FAMILY,
518 			     EL_TYPE__REGATTA__TITAN_PCHIP1_EXTENDED,
519 			     1,
520 			     "Titan PChip 1 Extended Frame",
521 			     el_titan_pchip1_extended_annotation),
522 	SUBPACKET_ANNOTATION(EL_CLASS__REGATTA_FAMILY,
523 			     EL_TYPE__REGATTA__TITAN_MEMORY_EXTENDED,
524 			     1,
525 			     "Titan Memory Extended Frame",
526 			     el_titan_memory_extended_annotation),
527 	SUBPACKET_ANNOTATION(EL_CLASS__REGATTA_FAMILY,
528 			     EL_TYPE__TERMINATION__TERMINATION,
529 			     1,
530 			     "Termination Subpacket",
531 			     NULL)
532 };
533 
534 static struct el_subpacket *
535 el_process_regatta_subpacket(struct el_subpacket *header)
536 {
537 	if (header->class != EL_CLASS__REGATTA_FAMILY) {
538 		printk("%s  ** Unexpected header CLASS %d TYPE %d, aborting\n",
539 		       err_print_prefix,
540 		       header->class, header->type);
541 		return NULL;
542 	}
543 
544 	switch(header->type) {
545 	case EL_TYPE__REGATTA__PROCESSOR_ERROR_FRAME:
546 	case EL_TYPE__REGATTA__SYSTEM_ERROR_FRAME:
547 	case EL_TYPE__REGATTA__ENVIRONMENTAL_FRAME:
548 	case EL_TYPE__REGATTA__PROCESSOR_DBL_ERROR_HALT:
549 	case EL_TYPE__REGATTA__SYSTEM_DBL_ERROR_HALT:
550 		printk("%s  ** Occurred on CPU %d:\n",
551 		       err_print_prefix,
552 		       (int)header->by_type.regatta_frame.cpuid);
553 		privateer_process_logout_frame((struct el_common *)
554 			header->by_type.regatta_frame.data_start, 1);
555 		break;
556 	default:
557 		printk("%s  ** REGATTA TYPE %d SUBPACKET\n",
558 		       err_print_prefix, header->type);
559 		el_annotate_subpacket(header);
560 		break;
561 	}
562 
563 
564 	return (struct el_subpacket *)((unsigned long)header + header->length);
565 }
566 
567 static struct el_subpacket_handler titan_subpacket_handler =
568 	SUBPACKET_HANDLER_INIT(EL_CLASS__REGATTA_FAMILY,
569 			       el_process_regatta_subpacket);
570 
571 void __init
572 titan_register_error_handlers(void)
573 {
574 	size_t i;
575 
576 	for (i = 0; i < ARRAY_SIZE (el_titan_annotations); i++)
577 		cdl_register_subpacket_annotation(&el_titan_annotations[i]);
578 
579 	cdl_register_subpacket_handler(&titan_subpacket_handler);
580 
581 	ev6_register_error_handlers();
582 }
583 
584 
585 /*
586  * Privateer
587  */
588 
589 static int
590 privateer_process_680_frame(struct el_common *mchk_header, int print)
591 {
592 	int status = MCHK_DISPOSITION_UNKNOWN_ERROR;
593 #ifdef CONFIG_VERBOSE_MCHECK
594 	struct el_PRIVATEER_envdata_mcheck *emchk =
595 		(struct el_PRIVATEER_envdata_mcheck *)
596 		((unsigned long)mchk_header + mchk_header->sys_offset);
597 
598 	/* TODO - categorize errors, for now, no error */
599 
600 	if (!print)
601 		return status;
602 
603 	/* TODO - decode instead of just dumping... */
604 	printk("%s  Summary Flags:         %016llx\n"
605  	         "  CChip DIRx:            %016llx\n"
606 		 "  System Management IR:  %016llx\n"
607 		 "  CPU IR:                %016llx\n"
608 		 "  Power Supply IR:       %016llx\n"
609 		 "  LM78 Fault Status:     %016llx\n"
610 		 "  System Doors:          %016llx\n"
611 		 "  Temperature Warning:   %016llx\n"
612 		 "  Fan Control:           %016llx\n"
613 		 "  Fatal Power Down Code: %016llx\n",
614 	       err_print_prefix,
615 	       emchk->summary,
616 	       emchk->c_dirx,
617 	       emchk->smir,
618 	       emchk->cpuir,
619 	       emchk->psir,
620 	       emchk->fault,
621 	       emchk->sys_doors,
622 	       emchk->temp_warn,
623 	       emchk->fan_ctrl,
624 	       emchk->code);
625 #endif /* CONFIG_VERBOSE_MCHECK */
626 
627 	return status;
628 }
629 
630 int
631 privateer_process_logout_frame(struct el_common *mchk_header, int print)
632 {
633 	struct el_common_EV6_mcheck *ev6mchk =
634 		(struct el_common_EV6_mcheck *)mchk_header;
635 	int status = MCHK_DISPOSITION_UNKNOWN_ERROR;
636 
637 	/*
638 	 * Machine check codes
639 	 */
640 #define PRIVATEER_MCHK__CORR_ECC		0x86	/* 630 */
641 #define PRIVATEER_MCHK__DC_TAG_PERR		0x9E	/* 630 */
642 #define PRIVATEER_MCHK__PAL_BUGCHECK		0x8E	/* 670 */
643 #define PRIVATEER_MCHK__OS_BUGCHECK		0x90	/* 670 */
644 #define PRIVATEER_MCHK__PROC_HRD_ERR		0x98	/* 670 */
645 #define PRIVATEER_MCHK__ISTREAM_CMOV_PRX	0xA0	/* 670 */
646 #define PRIVATEER_MCHK__ISTREAM_CMOV_FLT	0xA2	/* 670 */
647 #define PRIVATEER_MCHK__SYS_HRD_ERR		0x202	/* 660 */
648 #define PRIVATEER_MCHK__SYS_CORR_ERR		0x204	/* 620 */
649 #define PRIVATEER_MCHK__SYS_ENVIRON		0x206	/* 680 */
650 
651 	switch(ev6mchk->MCHK_Code) {
652 	/*
653 	 * Vector 630 - Processor, Correctable
654 	 */
655 	case PRIVATEER_MCHK__CORR_ECC:
656 	case PRIVATEER_MCHK__DC_TAG_PERR:
657 		/*
658 		 * Fall through to vector 670 for processing...
659 		 */
660 	/*
661 	 * Vector 670 - Processor, Uncorrectable
662 	 */
663 	case PRIVATEER_MCHK__PAL_BUGCHECK:
664 	case PRIVATEER_MCHK__OS_BUGCHECK:
665 	case PRIVATEER_MCHK__PROC_HRD_ERR:
666 	case PRIVATEER_MCHK__ISTREAM_CMOV_PRX:
667 	case PRIVATEER_MCHK__ISTREAM_CMOV_FLT:
668 		status |= ev6_process_logout_frame(mchk_header, print);
669 		break;
670 
671 	/*
672 	 * Vector 620 - System, Correctable
673 	 */
674 	case PRIVATEER_MCHK__SYS_CORR_ERR:
675 		/*
676 		 * Fall through to vector 660 for processing...
677 		 */
678 	/*
679 	 * Vector 660 - System, Uncorrectable
680 	 */
681 	case PRIVATEER_MCHK__SYS_HRD_ERR:
682 		status |= titan_process_logout_frame(mchk_header, print);
683 		break;
684 
685 	/*
686 	 * Vector 680 - System, Environmental
687 	 */
688 	case PRIVATEER_MCHK__SYS_ENVIRON:	/* System, Environmental */
689 		status |= privateer_process_680_frame(mchk_header, print);
690 		break;
691 
692 	/*
693 	 * Unknown
694 	 */
695 	default:
696 		status |= MCHK_DISPOSITION_REPORT;
697 		if (print) {
698 			printk("%s** Unknown Error, frame follows\n",
699 			       err_print_prefix);
700 			mchk_dump_logout_frame(mchk_header);
701 		}
702 
703 	}
704 
705 	return status;
706 }
707 
708 void
709 privateer_machine_check(unsigned long vector, unsigned long la_ptr)
710 {
711 	struct el_common *mchk_header = (struct el_common *)la_ptr;
712 	struct el_TITAN_sysdata_mcheck *tmchk =
713 		(struct el_TITAN_sysdata_mcheck *)
714 		(la_ptr + mchk_header->sys_offset);
715 	u64 irqmask;
716 	char *saved_err_prefix = err_print_prefix;
717 
718 #define PRIVATEER_680_INTERRUPT_MASK		(0xE00UL)
719 #define PRIVATEER_HOTPLUG_INTERRUPT_MASK	(0xE00UL)
720 
721 	/*
722 	 * Sync the processor.
723 	 */
724 	mb();
725 	draina();
726 
727 	/*
728 	 * Only handle system events here.
729 	 */
730 	if (vector != SCB_Q_SYSEVENT)
731 		return titan_machine_check(vector, la_ptr);
732 
733 	/*
734 	 * Report the event - System Events should be reported even if no
735 	 * error is indicated since the event could indicate the return
736 	 * to normal status.
737 	 */
738 	err_print_prefix = KERN_CRIT;
739 	printk("%s*System Event (Vector 0x%x) reported on CPU %d:\n",
740 	       err_print_prefix,
741 	       (unsigned int)vector, (int)smp_processor_id());
742 	privateer_process_680_frame(mchk_header, 1);
743 	err_print_prefix = saved_err_prefix;
744 
745 	/*
746 	 * Convert any pending interrupts which report as 680 machine
747 	 * checks to interrupts.
748 	 */
749 	irqmask = tmchk->c_dirx & PRIVATEER_680_INTERRUPT_MASK;
750 
751 	/*
752 	 * Dispatch the interrupt(s).
753 	 */
754 	titan_dispatch_irqs(irqmask);
755 
756 	/*
757 	 * Release the logout frame.
758 	 */
759 	wrmces(0x7);
760 	mb();
761 }
762