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