Don't use magic numbers.
[gem5.git] / arch / alpha / ev5.cc
1 /* $Id$ */
2
3 #include "arch/alpha/alpha_memory.hh"
4 #include "arch/alpha/isa_traits.hh"
5 #include "arch/alpha/osfpal.hh"
6 #include "base/kgdb.h"
7 #include "base/remote_gdb.hh"
8 #include "base/stats/events.hh"
9 #include "cpu/base_cpu.hh"
10 #include "cpu/exec_context.hh"
11 #include "cpu/fast_cpu/fast_cpu.hh"
12 #include "kern/kernel_stats.hh"
13 #include "sim/debug.hh"
14 #include "sim/sim_events.hh"
15
16 #ifdef FULL_SYSTEM
17
18 #ifndef SYSTEM_EV5
19 #error This code is only valid for EV5 systems
20 #endif
21
22 ////////////////////////////////////////////////////////////////////////
23 //
24 //
25 //
26 void
27 AlphaISA::swap_palshadow(RegFile *regs, bool use_shadow)
28 {
29 if (regs->pal_shadow == use_shadow)
30 panic("swap_palshadow: wrong PAL shadow state");
31
32 regs->pal_shadow = use_shadow;
33
34 for (int i = 0; i < NumIntRegs; i++) {
35 if (reg_redir[i]) {
36 IntReg temp = regs->intRegFile[i];
37 regs->intRegFile[i] = regs->palregs[i];
38 regs->palregs[i] = temp;
39 }
40 }
41 }
42
43 ////////////////////////////////////////////////////////////////////////
44 //
45 // Machine dependent functions
46 //
47 void
48 AlphaISA::initCPU(RegFile *regs)
49 {
50 initIPRs(regs);
51 // CPU comes up with PAL regs enabled
52 swap_palshadow(regs, true);
53
54 regs->pc = regs->ipr[IPR_PAL_BASE] + fault_addr[Reset_Fault];
55 regs->npc = regs->pc + sizeof(MachInst);
56 }
57
58 ////////////////////////////////////////////////////////////////////////
59 //
60 // alpha exceptions - value equals trap address, update with MD_FAULT_TYPE
61 //
62 Addr
63 AlphaISA::fault_addr[Num_Faults] = {
64 0x0000, /* No_Fault */
65 0x0001, /* Reset_Fault */
66 0x0401, /* Machine_Check_Fault */
67 0x0501, /* Arithmetic_Fault */
68 0x0101, /* Interrupt_Fault */
69 0x0201, /* Ndtb_Miss_Fault */
70 0x0281, /* Pdtb_Miss_Fault */
71 0x0301, /* Alignment_Fault */
72 0x0381, /* DTB_Fault_Fault */
73 0x0381, /* DTB_Acv_Fault */
74 0x0181, /* ITB_Miss_Fault */
75 0x0181, /* ITB_Fault_Fault */
76 0x0081, /* ITB_Acv_Fault */
77 0x0481, /* Unimplemented_Opcode_Fault */
78 0x0581, /* Fen_Fault */
79 0x2001, /* Pal_Fault */
80 0x0501, /* Integer_Overflow_Fault: maps to Arithmetic_Fault */
81 };
82
83 const int AlphaISA::reg_redir[AlphaISA::NumIntRegs] = {
84 /* 0 */ 0, 0, 0, 0, 0, 0, 0, 0,
85 /* 8 */ 1, 1, 1, 1, 1, 1, 1, 0,
86 /* 16 */ 0, 0, 0, 0, 0, 0, 0, 0,
87 /* 24 */ 0, 1, 0, 0, 0, 0, 0, 0 };
88
89 ////////////////////////////////////////////////////////////////////////
90 //
91 //
92 //
93 void
94 AlphaISA::initIPRs(RegFile *regs)
95 {
96 uint64_t *ipr = regs->ipr;
97
98 bzero((char *)ipr, NumInternalProcRegs * sizeof(InternalProcReg));
99 ipr[IPR_PAL_BASE] = PAL_BASE;
100 ipr[IPR_MCSR] = 0x6;
101 }
102
103
104 template <class XC>
105 void
106 AlphaISA::processInterrupts(XC *xc)
107 {
108 //Check if there are any outstanding interrupts
109 //Handle the interrupts
110 int ipl = 0;
111 int summary = 0;
112 IntReg *ipr = xc->getIprPtr();
113
114 check_interrupts = 0;
115
116 if (ipr[IPR_ASTRR])
117 panic("asynchronous traps not implemented\n");
118
119 if (ipr[IPR_SIRR]) {
120 for (int i = INTLEVEL_SOFTWARE_MIN;
121 i < INTLEVEL_SOFTWARE_MAX; i++) {
122 if (ipr[IPR_SIRR] & (ULL(1) << i)) {
123 // See table 4-19 of the 21164 hardware reference
124 ipl = (i - INTLEVEL_SOFTWARE_MIN) + 1;
125 summary |= (ULL(1) << i);
126 }
127 }
128 }
129
130 uint64_t interrupts = xc->intr_status();
131
132 if (interrupts) {
133 for (int i = INTLEVEL_EXTERNAL_MIN;
134 i < INTLEVEL_EXTERNAL_MAX; i++) {
135 if (interrupts & (ULL(1) << i)) {
136 // See table 4-19 of the 21164 hardware reference
137 ipl = i;
138 summary |= (ULL(1) << i);
139 }
140 }
141 }
142
143 if (ipl && ipl > ipr[IPR_IPLR]) {
144 ipr[IPR_ISR] = summary;
145 ipr[IPR_INTID] = ipl;
146 xc->trap(Interrupt_Fault);
147 DPRINTF(Flow, "Interrupt! IPLR=%d ipl=%d summary=%x\n",
148 ipr[IPR_IPLR], ipl, summary);
149 }
150
151 }
152
153 template <class XC>
154 void
155 AlphaISA::zeroRegisters(XC *xc)
156 {
157 // Insure ISA semantics
158 // (no longer very clean due to the change in setIntReg() in the
159 // cpu model. Consider changing later.)
160 xc->xc->setIntReg(ZeroReg, 0);
161 xc->xc->setFloatRegDouble(ZeroReg, 0.0);
162 }
163
164 void
165 ExecContext::ev5_trap(Fault fault)
166 {
167 DPRINTF(Fault, "Fault %s at PC: %#x\n", FaultName(fault), regs.pc);
168 cpu->recordEvent(csprintf("Fault %s", FaultName(fault)));
169
170 assert(!misspeculating());
171 kernelStats->fault(fault);
172
173 if (fault == Arithmetic_Fault)
174 panic("Arithmetic traps are unimplemented!");
175
176 AlphaISA::InternalProcReg *ipr = regs.ipr;
177
178 // exception restart address
179 if (fault != Interrupt_Fault || !PC_PAL(regs.pc))
180 ipr[AlphaISA::IPR_EXC_ADDR] = regs.pc;
181
182 if (fault == Pal_Fault || fault == Arithmetic_Fault /* ||
183 fault == Interrupt_Fault && !PC_PAL(regs.pc) */) {
184 // traps... skip faulting instruction
185 ipr[AlphaISA::IPR_EXC_ADDR] += 4;
186 }
187
188 if (!PC_PAL(regs.pc))
189 AlphaISA::swap_palshadow(&regs, true);
190
191 regs.pc = ipr[AlphaISA::IPR_PAL_BASE] + AlphaISA::fault_addr[fault];
192 regs.npc = regs.pc + sizeof(MachInst);
193 }
194
195
196 void
197 AlphaISA::intr_post(RegFile *regs, Fault fault, Addr pc)
198 {
199 InternalProcReg *ipr = regs->ipr;
200 bool use_pc = (fault == No_Fault);
201
202 if (fault == Arithmetic_Fault)
203 panic("arithmetic faults NYI...");
204
205 // compute exception restart address
206 if (use_pc || fault == Pal_Fault || fault == Arithmetic_Fault) {
207 // traps... skip faulting instruction
208 ipr[IPR_EXC_ADDR] = regs->pc + 4;
209 } else {
210 // fault, post fault at excepting instruction
211 ipr[IPR_EXC_ADDR] = regs->pc;
212 }
213
214 // jump to expection address (PAL PC bit set here as well...)
215 if (!use_pc)
216 regs->npc = ipr[IPR_PAL_BASE] + fault_addr[fault];
217 else
218 regs->npc = ipr[IPR_PAL_BASE] + pc;
219
220 // that's it! (orders of magnitude less painful than x86)
221 }
222
223 bool AlphaISA::check_interrupts = false;
224
225 Fault
226 ExecContext::hwrei()
227 {
228 uint64_t *ipr = regs.ipr;
229
230 if (!PC_PAL(regs.pc))
231 return Unimplemented_Opcode_Fault;
232
233 setNextPC(ipr[AlphaISA::IPR_EXC_ADDR]);
234
235 if (!misspeculating()) {
236 kernelStats->hwrei();
237
238 if ((ipr[AlphaISA::IPR_EXC_ADDR] & 1) == 0)
239 AlphaISA::swap_palshadow(&regs, false);
240
241 AlphaISA::check_interrupts = true;
242 }
243
244 // FIXME: XXX check for interrupts? XXX
245 return No_Fault;
246 }
247
248 uint64_t
249 ExecContext::readIpr(int idx, Fault &fault)
250 {
251 uint64_t *ipr = regs.ipr;
252 uint64_t retval = 0; // return value, default 0
253
254 switch (idx) {
255 case AlphaISA::IPR_PALtemp0:
256 case AlphaISA::IPR_PALtemp1:
257 case AlphaISA::IPR_PALtemp2:
258 case AlphaISA::IPR_PALtemp3:
259 case AlphaISA::IPR_PALtemp4:
260 case AlphaISA::IPR_PALtemp5:
261 case AlphaISA::IPR_PALtemp6:
262 case AlphaISA::IPR_PALtemp7:
263 case AlphaISA::IPR_PALtemp8:
264 case AlphaISA::IPR_PALtemp9:
265 case AlphaISA::IPR_PALtemp10:
266 case AlphaISA::IPR_PALtemp11:
267 case AlphaISA::IPR_PALtemp12:
268 case AlphaISA::IPR_PALtemp13:
269 case AlphaISA::IPR_PALtemp14:
270 case AlphaISA::IPR_PALtemp15:
271 case AlphaISA::IPR_PALtemp16:
272 case AlphaISA::IPR_PALtemp17:
273 case AlphaISA::IPR_PALtemp18:
274 case AlphaISA::IPR_PALtemp19:
275 case AlphaISA::IPR_PALtemp20:
276 case AlphaISA::IPR_PALtemp21:
277 case AlphaISA::IPR_PALtemp22:
278 case AlphaISA::IPR_PALtemp23:
279 case AlphaISA::IPR_PAL_BASE:
280
281 case AlphaISA::IPR_IVPTBR:
282 case AlphaISA::IPR_DC_MODE:
283 case AlphaISA::IPR_MAF_MODE:
284 case AlphaISA::IPR_ISR:
285 case AlphaISA::IPR_EXC_ADDR:
286 case AlphaISA::IPR_IC_PERR_STAT:
287 case AlphaISA::IPR_DC_PERR_STAT:
288 case AlphaISA::IPR_MCSR:
289 case AlphaISA::IPR_ASTRR:
290 case AlphaISA::IPR_ASTER:
291 case AlphaISA::IPR_SIRR:
292 case AlphaISA::IPR_ICSR:
293 case AlphaISA::IPR_ICM:
294 case AlphaISA::IPR_DTB_CM:
295 case AlphaISA::IPR_IPLR:
296 case AlphaISA::IPR_INTID:
297 case AlphaISA::IPR_PMCTR:
298 // no side-effect
299 retval = ipr[idx];
300 break;
301
302 case AlphaISA::IPR_CC:
303 retval |= ipr[idx] & ULL(0xffffffff00000000);
304 retval |= curTick & ULL(0x00000000ffffffff);
305 break;
306
307 case AlphaISA::IPR_VA:
308 retval = ipr[idx];
309 break;
310
311 case AlphaISA::IPR_VA_FORM:
312 case AlphaISA::IPR_MM_STAT:
313 case AlphaISA::IPR_IFAULT_VA_FORM:
314 case AlphaISA::IPR_EXC_MASK:
315 case AlphaISA::IPR_EXC_SUM:
316 retval = ipr[idx];
317 break;
318
319 case AlphaISA::IPR_DTB_PTE:
320 {
321 AlphaISA::PTE &pte = dtb->index(!misspeculating());
322
323 retval |= ((u_int64_t)pte.ppn & ULL(0x7ffffff)) << 32;
324 retval |= ((u_int64_t)pte.xre & ULL(0xf)) << 8;
325 retval |= ((u_int64_t)pte.xwe & ULL(0xf)) << 12;
326 retval |= ((u_int64_t)pte.fonr & ULL(0x1)) << 1;
327 retval |= ((u_int64_t)pte.fonw & ULL(0x1))<< 2;
328 retval |= ((u_int64_t)pte.asma & ULL(0x1)) << 4;
329 retval |= ((u_int64_t)pte.asn & ULL(0x7f)) << 57;
330 }
331 break;
332
333 // write only registers
334 case AlphaISA::IPR_HWINT_CLR:
335 case AlphaISA::IPR_SL_XMIT:
336 case AlphaISA::IPR_DC_FLUSH:
337 case AlphaISA::IPR_IC_FLUSH:
338 case AlphaISA::IPR_ALT_MODE:
339 case AlphaISA::IPR_DTB_IA:
340 case AlphaISA::IPR_DTB_IAP:
341 case AlphaISA::IPR_ITB_IA:
342 case AlphaISA::IPR_ITB_IAP:
343 fault = Unimplemented_Opcode_Fault;
344 break;
345
346 default:
347 // invalid IPR
348 fault = Unimplemented_Opcode_Fault;
349 break;
350 }
351
352 return retval;
353 }
354
355 #ifdef DEBUG
356 // Cause the simulator to break when changing to the following IPL
357 int break_ipl = -1;
358 #endif
359
360 Fault
361 ExecContext::setIpr(int idx, uint64_t val)
362 {
363 uint64_t *ipr = regs.ipr;
364 uint64_t old;
365
366 if (misspeculating())
367 return No_Fault;
368
369 switch (idx) {
370 case AlphaISA::IPR_PALtemp0:
371 case AlphaISA::IPR_PALtemp1:
372 case AlphaISA::IPR_PALtemp2:
373 case AlphaISA::IPR_PALtemp3:
374 case AlphaISA::IPR_PALtemp4:
375 case AlphaISA::IPR_PALtemp5:
376 case AlphaISA::IPR_PALtemp6:
377 case AlphaISA::IPR_PALtemp7:
378 case AlphaISA::IPR_PALtemp8:
379 case AlphaISA::IPR_PALtemp9:
380 case AlphaISA::IPR_PALtemp10:
381 case AlphaISA::IPR_PALtemp11:
382 case AlphaISA::IPR_PALtemp12:
383 case AlphaISA::IPR_PALtemp13:
384 case AlphaISA::IPR_PALtemp14:
385 case AlphaISA::IPR_PALtemp15:
386 case AlphaISA::IPR_PALtemp16:
387 case AlphaISA::IPR_PALtemp17:
388 case AlphaISA::IPR_PALtemp18:
389 case AlphaISA::IPR_PALtemp19:
390 case AlphaISA::IPR_PALtemp20:
391 case AlphaISA::IPR_PALtemp21:
392 case AlphaISA::IPR_PALtemp22:
393 case AlphaISA::IPR_PAL_BASE:
394 case AlphaISA::IPR_IC_PERR_STAT:
395 case AlphaISA::IPR_DC_PERR_STAT:
396 case AlphaISA::IPR_PMCTR:
397 // write entire quad w/ no side-effect
398 ipr[idx] = val;
399 break;
400
401 case AlphaISA::IPR_CC_CTL:
402 // This IPR resets the cycle counter. We assume this only
403 // happens once... let's verify that.
404 assert(ipr[idx] == 0);
405 ipr[idx] = 1;
406 break;
407
408 case AlphaISA::IPR_CC:
409 // This IPR only writes the upper 64 bits. It's ok to write
410 // all 64 here since we mask out the lower 32 in rpcc (see
411 // isa_desc).
412 ipr[idx] = val;
413 break;
414
415 case AlphaISA::IPR_PALtemp23:
416 // write entire quad w/ no side-effect
417 old = ipr[idx];
418 ipr[idx] = val;
419 kernelStats->context(old, val);
420 break;
421
422 case AlphaISA::IPR_DTB_PTE:
423 // write entire quad w/ no side-effect, tag is forthcoming
424 ipr[idx] = val;
425 break;
426
427 case AlphaISA::IPR_EXC_ADDR:
428 // second least significant bit in PC is always zero
429 ipr[idx] = val & ~2;
430 break;
431
432 case AlphaISA::IPR_ASTRR:
433 case AlphaISA::IPR_ASTER:
434 // only write least significant four bits - privilege mask
435 ipr[idx] = val & 0xf;
436 break;
437
438 case AlphaISA::IPR_IPLR:
439 #ifdef DEBUG
440 if (break_ipl != -1 && break_ipl == (val & 0x1f))
441 debug_break();
442 #endif
443
444 // only write least significant five bits - interrupt level
445 ipr[idx] = val & 0x1f;
446 kernelStats->swpipl(ipr[idx]);
447 break;
448
449 case AlphaISA::IPR_DTB_CM:
450 if (val & 0x18)
451 kernelStats->mode(Kernel::user);
452 else
453 kernelStats->mode(Kernel::kernel);
454
455 case AlphaISA::IPR_ICM:
456 // only write two mode bits - processor mode
457 ipr[idx] = val & 0x18;
458 break;
459
460 case AlphaISA::IPR_ALT_MODE:
461 // only write two mode bits - processor mode
462 ipr[idx] = val & 0x18;
463 break;
464
465 case AlphaISA::IPR_MCSR:
466 // more here after optimization...
467 ipr[idx] = val;
468 break;
469
470 case AlphaISA::IPR_SIRR:
471 // only write software interrupt mask
472 ipr[idx] = val & 0x7fff0;
473 break;
474
475 case AlphaISA::IPR_ICSR:
476 ipr[idx] = val & ULL(0xffffff0300);
477 break;
478
479 case AlphaISA::IPR_IVPTBR:
480 case AlphaISA::IPR_MVPTBR:
481 ipr[idx] = val & ULL(0xffffffffc0000000);
482 break;
483
484 case AlphaISA::IPR_DC_TEST_CTL:
485 ipr[idx] = val & 0x1ffb;
486 break;
487
488 case AlphaISA::IPR_DC_MODE:
489 case AlphaISA::IPR_MAF_MODE:
490 ipr[idx] = val & 0x3f;
491 break;
492
493 case AlphaISA::IPR_ITB_ASN:
494 ipr[idx] = val & 0x7f0;
495 break;
496
497 case AlphaISA::IPR_DTB_ASN:
498 ipr[idx] = val & ULL(0xfe00000000000000);
499 break;
500
501 case AlphaISA::IPR_EXC_SUM:
502 case AlphaISA::IPR_EXC_MASK:
503 // any write to this register clears it
504 ipr[idx] = 0;
505 break;
506
507 case AlphaISA::IPR_INTID:
508 case AlphaISA::IPR_SL_RCV:
509 case AlphaISA::IPR_MM_STAT:
510 case AlphaISA::IPR_ITB_PTE_TEMP:
511 case AlphaISA::IPR_DTB_PTE_TEMP:
512 // read-only registers
513 return Unimplemented_Opcode_Fault;
514
515 case AlphaISA::IPR_HWINT_CLR:
516 case AlphaISA::IPR_SL_XMIT:
517 case AlphaISA::IPR_DC_FLUSH:
518 case AlphaISA::IPR_IC_FLUSH:
519 // the following are write only
520 ipr[idx] = val;
521 break;
522
523 case AlphaISA::IPR_DTB_IA:
524 // really a control write
525 ipr[idx] = 0;
526
527 dtb->flushAll();
528 break;
529
530 case AlphaISA::IPR_DTB_IAP:
531 // really a control write
532 ipr[idx] = 0;
533
534 dtb->flushProcesses();
535 break;
536
537 case AlphaISA::IPR_DTB_IS:
538 // really a control write
539 ipr[idx] = val;
540
541 dtb->flushAddr(val, DTB_ASN_ASN(ipr[AlphaISA::IPR_DTB_ASN]));
542 break;
543
544 case AlphaISA::IPR_DTB_TAG: {
545 struct AlphaISA::PTE pte;
546
547 // FIXME: granularity hints NYI...
548 if (DTB_PTE_GH(ipr[AlphaISA::IPR_DTB_PTE]) != 0)
549 panic("PTE GH field != 0");
550
551 // write entire quad
552 ipr[idx] = val;
553
554 // construct PTE for new entry
555 pte.ppn = DTB_PTE_PPN(ipr[AlphaISA::IPR_DTB_PTE]);
556 pte.xre = DTB_PTE_XRE(ipr[AlphaISA::IPR_DTB_PTE]);
557 pte.xwe = DTB_PTE_XWE(ipr[AlphaISA::IPR_DTB_PTE]);
558 pte.fonr = DTB_PTE_FONR(ipr[AlphaISA::IPR_DTB_PTE]);
559 pte.fonw = DTB_PTE_FONW(ipr[AlphaISA::IPR_DTB_PTE]);
560 pte.asma = DTB_PTE_ASMA(ipr[AlphaISA::IPR_DTB_PTE]);
561 pte.asn = DTB_ASN_ASN(ipr[AlphaISA::IPR_DTB_ASN]);
562
563 // insert new TAG/PTE value into data TLB
564 dtb->insert(val, pte);
565 }
566 break;
567
568 case AlphaISA::IPR_ITB_PTE: {
569 struct AlphaISA::PTE pte;
570
571 // FIXME: granularity hints NYI...
572 if (ITB_PTE_GH(val) != 0)
573 panic("PTE GH field != 0");
574
575 // write entire quad
576 ipr[idx] = val;
577
578 // construct PTE for new entry
579 pte.ppn = ITB_PTE_PPN(val);
580 pte.xre = ITB_PTE_XRE(val);
581 pte.xwe = 0;
582 pte.fonr = ITB_PTE_FONR(val);
583 pte.fonw = ITB_PTE_FONW(val);
584 pte.asma = ITB_PTE_ASMA(val);
585 pte.asn = ITB_ASN_ASN(ipr[AlphaISA::IPR_ITB_ASN]);
586
587 // insert new TAG/PTE value into data TLB
588 itb->insert(ipr[AlphaISA::IPR_ITB_TAG], pte);
589 }
590 break;
591
592 case AlphaISA::IPR_ITB_IA:
593 // really a control write
594 ipr[idx] = 0;
595
596 itb->flushAll();
597 break;
598
599 case AlphaISA::IPR_ITB_IAP:
600 // really a control write
601 ipr[idx] = 0;
602
603 itb->flushProcesses();
604 break;
605
606 case AlphaISA::IPR_ITB_IS:
607 // really a control write
608 ipr[idx] = val;
609
610 itb->flushAddr(val, ITB_ASN_ASN(ipr[AlphaISA::IPR_ITB_ASN]));
611 break;
612
613 default:
614 // invalid IPR
615 return Unimplemented_Opcode_Fault;
616 }
617
618 // no error...
619 return No_Fault;
620 }
621
622 /**
623 * Check for special simulator handling of specific PAL calls.
624 * If return value is false, actual PAL call will be suppressed.
625 */
626 bool
627 ExecContext::simPalCheck(int palFunc)
628 {
629 kernelStats->callpal(palFunc);
630
631 switch (palFunc) {
632 case PAL::halt:
633 halt();
634 if (--System::numSystemsRunning == 0)
635 new SimExitEvent("all cpus halted");
636 break;
637
638 case PAL::bpt:
639 case PAL::bugchk:
640 if (system->breakpoint())
641 return false;
642 break;
643 }
644
645 return true;
646 }
647
648 //Forward instantiation for FastCPU object
649 template
650 void AlphaISA::processInterrupts(FastCPU *xc);
651
652 //Forward instantiation for FastCPU object
653 template
654 void AlphaISA::zeroRegisters(FastCPU *xc);
655
656 #endif // FULL_SYSTEM