Merge git://git.kernel.org/pub/scm/linux/kernel/git/jejb/scsi-rc-fixes-2.6
[pandora-kernel.git] / arch / alpha / kernel / err_titan.c
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 \f
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 }