1/* 2 * This file contains the 64-bit "server" PowerPC variant 3 * of the low level exception handling including exception 4 * vectors, exception return, part of the slb and stab 5 * handling and other fixed offset specific things. 6 * 7 * This file is meant to be #included from head_64.S due to 8 * position dependant assembly. 9 * 10 * Most of this originates from head_64.S and thus has the same 11 * copyright history. 12 * 13 */ 14 15#include <asm/exception-64s.h> 16#include <asm/ptrace.h> 17 18/* 19 * We layout physical memory as follows: 20 * 0x0000 - 0x00ff : Secondary processor spin code 21 * 0x0100 - 0x2fff : pSeries Interrupt prologs 22 * 0x3000 - 0x5fff : interrupt support, iSeries and common interrupt prologs 23 * 0x6000 - 0x6fff : Initial (CPU0) segment table 24 * 0x7000 - 0x7fff : FWNMI data area 25 * 0x8000 - : Early init and support code 26 */ 27 28/* 29 * This is the start of the interrupt handlers for pSeries 30 * This code runs with relocation off. 31 * Code from here to __end_interrupts gets copied down to real 32 * address 0x100 when we are running a relocatable kernel. 33 * Therefore any relative branches in this section must only 34 * branch to labels in this section. 35 */ 36 . = 0x100 37 .globl __start_interrupts 38__start_interrupts: 39 40 STD_EXCEPTION_PSERIES(0x100, system_reset) 41 42 . = 0x200 43_machine_check_pSeries: 44 HMT_MEDIUM 45 DO_KVM 0x200 46 mtspr SPRN_SPRG_SCRATCH0,r13 /* save r13 */ 47 EXCEPTION_PROLOG_PSERIES(PACA_EXMC, machine_check_common) 48 49 . = 0x300 50 .globl data_access_pSeries 51data_access_pSeries: 52 HMT_MEDIUM 53 DO_KVM 0x300 54 mtspr SPRN_SPRG_SCRATCH0,r13 55BEGIN_FTR_SECTION 56 mfspr r13,SPRN_SPRG_PACA 57 std r9,PACA_EXSLB+EX_R9(r13) 58 std r10,PACA_EXSLB+EX_R10(r13) 59 mfspr r10,SPRN_DAR 60 mfspr r9,SPRN_DSISR 61 srdi r10,r10,60 62 rlwimi r10,r9,16,0x20 63 mfcr r9 64 cmpwi r10,0x2c 65 beq do_stab_bolted_pSeries 66 ld r10,PACA_EXSLB+EX_R10(r13) 67 std r11,PACA_EXGEN+EX_R11(r13) 68 ld r11,PACA_EXSLB+EX_R9(r13) 69 std r12,PACA_EXGEN+EX_R12(r13) 70 mfspr r12,SPRN_SPRG_SCRATCH0 71 std r10,PACA_EXGEN+EX_R10(r13) 72 std r11,PACA_EXGEN+EX_R9(r13) 73 std r12,PACA_EXGEN+EX_R13(r13) 74 EXCEPTION_PROLOG_PSERIES_1(data_access_common) 75FTR_SECTION_ELSE 76 EXCEPTION_PROLOG_PSERIES(PACA_EXGEN, data_access_common) 77ALT_FTR_SECTION_END_IFCLR(CPU_FTR_SLB) 78 79 . = 0x380 80 .globl data_access_slb_pSeries 81data_access_slb_pSeries: 82 HMT_MEDIUM 83 DO_KVM 0x380 84 mtspr SPRN_SPRG_SCRATCH0,r13 85 mfspr r13,SPRN_SPRG_PACA /* get paca address into r13 */ 86 std r3,PACA_EXSLB+EX_R3(r13) 87 mfspr r3,SPRN_DAR 88 std r9,PACA_EXSLB+EX_R9(r13) /* save r9 - r12 */ 89 mfcr r9 90#ifdef __DISABLED__ 91 /* Keep that around for when we re-implement dynamic VSIDs */ 92 cmpdi r3,0 93 bge slb_miss_user_pseries 94#endif /* __DISABLED__ */ 95 std r10,PACA_EXSLB+EX_R10(r13) 96 std r11,PACA_EXSLB+EX_R11(r13) 97 std r12,PACA_EXSLB+EX_R12(r13) 98 mfspr r10,SPRN_SPRG_SCRATCH0 99 std r10,PACA_EXSLB+EX_R13(r13) 100 mfspr r12,SPRN_SRR1 /* and SRR1 */ 101#ifndef CONFIG_RELOCATABLE 102 b .slb_miss_realmode 103#else 104 /* 105 * We can't just use a direct branch to .slb_miss_realmode 106 * because the distance from here to there depends on where 107 * the kernel ends up being put. 108 */ 109 mfctr r11 110 ld r10,PACAKBASE(r13) 111 LOAD_HANDLER(r10, .slb_miss_realmode) 112 mtctr r10 113 bctr 114#endif 115 116 STD_EXCEPTION_PSERIES(0x400, instruction_access) 117 118 . = 0x480 119 .globl instruction_access_slb_pSeries 120instruction_access_slb_pSeries: 121 HMT_MEDIUM 122 DO_KVM 0x480 123 mtspr SPRN_SPRG_SCRATCH0,r13 124 mfspr r13,SPRN_SPRG_PACA /* get paca address into r13 */ 125 std r3,PACA_EXSLB+EX_R3(r13) 126 mfspr r3,SPRN_SRR0 /* SRR0 is faulting address */ 127 std r9,PACA_EXSLB+EX_R9(r13) /* save r9 - r12 */ 128 mfcr r9 129#ifdef __DISABLED__ 130 /* Keep that around for when we re-implement dynamic VSIDs */ 131 cmpdi r3,0 132 bge slb_miss_user_pseries 133#endif /* __DISABLED__ */ 134 std r10,PACA_EXSLB+EX_R10(r13) 135 std r11,PACA_EXSLB+EX_R11(r13) 136 std r12,PACA_EXSLB+EX_R12(r13) 137 mfspr r10,SPRN_SPRG_SCRATCH0 138 std r10,PACA_EXSLB+EX_R13(r13) 139 mfspr r12,SPRN_SRR1 /* and SRR1 */ 140#ifndef CONFIG_RELOCATABLE 141 b .slb_miss_realmode 142#else 143 mfctr r11 144 ld r10,PACAKBASE(r13) 145 LOAD_HANDLER(r10, .slb_miss_realmode) 146 mtctr r10 147 bctr 148#endif 149 150 MASKABLE_EXCEPTION_PSERIES(0x500, hardware_interrupt) 151 STD_EXCEPTION_PSERIES(0x600, alignment) 152 STD_EXCEPTION_PSERIES(0x700, program_check) 153 STD_EXCEPTION_PSERIES(0x800, fp_unavailable) 154 MASKABLE_EXCEPTION_PSERIES(0x900, decrementer) 155 STD_EXCEPTION_PSERIES(0xa00, trap_0a) 156 STD_EXCEPTION_PSERIES(0xb00, trap_0b) 157 158 . = 0xc00 159 .globl system_call_pSeries 160system_call_pSeries: 161 HMT_MEDIUM 162 DO_KVM 0xc00 163BEGIN_FTR_SECTION 164 cmpdi r0,0x1ebe 165 beq- 1f 166END_FTR_SECTION_IFSET(CPU_FTR_REAL_LE) 167 mr r9,r13 168 mfspr r13,SPRN_SPRG_PACA 169 mfspr r11,SPRN_SRR0 170 ld r12,PACAKBASE(r13) 171 ld r10,PACAKMSR(r13) 172 LOAD_HANDLER(r12, system_call_entry) 173 mtspr SPRN_SRR0,r12 174 mfspr r12,SPRN_SRR1 175 mtspr SPRN_SRR1,r10 176 rfid 177 b . /* prevent speculative execution */ 178 179/* Fast LE/BE switch system call */ 1801: mfspr r12,SPRN_SRR1 181 xori r12,r12,MSR_LE 182 mtspr SPRN_SRR1,r12 183 rfid /* return to userspace */ 184 b . 185 186 STD_EXCEPTION_PSERIES(0xd00, single_step) 187 STD_EXCEPTION_PSERIES(0xe00, trap_0e) 188 189 /* We need to deal with the Altivec unavailable exception 190 * here which is at 0xf20, thus in the middle of the 191 * prolog code of the PerformanceMonitor one. A little 192 * trickery is thus necessary 193 */ 194performance_monitor_pSeries_1: 195 . = 0xf00 196 DO_KVM 0xf00 197 b performance_monitor_pSeries 198 199altivec_unavailable_pSeries_1: 200 . = 0xf20 201 DO_KVM 0xf20 202 b altivec_unavailable_pSeries 203 204vsx_unavailable_pSeries_1: 205 . = 0xf40 206 DO_KVM 0xf40 207 b vsx_unavailable_pSeries 208 209#ifdef CONFIG_CBE_RAS 210 HSTD_EXCEPTION_PSERIES(0x1200, cbe_system_error) 211#endif /* CONFIG_CBE_RAS */ 212 STD_EXCEPTION_PSERIES(0x1300, instruction_breakpoint) 213#ifdef CONFIG_CBE_RAS 214 HSTD_EXCEPTION_PSERIES(0x1600, cbe_maintenance) 215#endif /* CONFIG_CBE_RAS */ 216 STD_EXCEPTION_PSERIES(0x1700, altivec_assist) 217#ifdef CONFIG_CBE_RAS 218 HSTD_EXCEPTION_PSERIES(0x1800, cbe_thermal) 219#endif /* CONFIG_CBE_RAS */ 220 221 . = 0x3000 222 223/*** pSeries interrupt support ***/ 224 225 /* moved from 0xf00 */ 226 STD_EXCEPTION_PSERIES(., performance_monitor) 227 STD_EXCEPTION_PSERIES(., altivec_unavailable) 228 STD_EXCEPTION_PSERIES(., vsx_unavailable) 229 230/* 231 * An interrupt came in while soft-disabled; clear EE in SRR1, 232 * clear paca->hard_enabled and return. 233 */ 234masked_interrupt: 235 stb r10,PACAHARDIRQEN(r13) 236 mtcrf 0x80,r9 237 ld r9,PACA_EXGEN+EX_R9(r13) 238 mfspr r10,SPRN_SRR1 239 rldicl r10,r10,48,1 /* clear MSR_EE */ 240 rotldi r10,r10,16 241 mtspr SPRN_SRR1,r10 242 ld r10,PACA_EXGEN+EX_R10(r13) 243 mfspr r13,SPRN_SPRG_SCRATCH0 244 rfid 245 b . 246 247 .align 7 248do_stab_bolted_pSeries: 249 std r11,PACA_EXSLB+EX_R11(r13) 250 std r12,PACA_EXSLB+EX_R12(r13) 251 mfspr r10,SPRN_SPRG_SCRATCH0 252 std r10,PACA_EXSLB+EX_R13(r13) 253 EXCEPTION_PROLOG_PSERIES_1(.do_stab_bolted) 254 255#ifdef CONFIG_PPC_PSERIES 256/* 257 * Vectors for the FWNMI option. Share common code. 258 */ 259 .globl system_reset_fwnmi 260 .align 7 261system_reset_fwnmi: 262 HMT_MEDIUM 263 mtspr SPRN_SPRG_SCRATCH0,r13 /* save r13 */ 264 EXCEPTION_PROLOG_PSERIES(PACA_EXGEN, system_reset_common) 265 266 .globl machine_check_fwnmi 267 .align 7 268machine_check_fwnmi: 269 HMT_MEDIUM 270 mtspr SPRN_SPRG_SCRATCH0,r13 /* save r13 */ 271 EXCEPTION_PROLOG_PSERIES(PACA_EXMC, machine_check_common) 272 273#endif /* CONFIG_PPC_PSERIES */ 274 275#ifdef __DISABLED__ 276/* 277 * This is used for when the SLB miss handler has to go virtual, 278 * which doesn't happen for now anymore but will once we re-implement 279 * dynamic VSIDs for shared page tables 280 */ 281slb_miss_user_pseries: 282 std r10,PACA_EXGEN+EX_R10(r13) 283 std r11,PACA_EXGEN+EX_R11(r13) 284 std r12,PACA_EXGEN+EX_R12(r13) 285 mfspr r10,SPRG_SCRATCH0 286 ld r11,PACA_EXSLB+EX_R9(r13) 287 ld r12,PACA_EXSLB+EX_R3(r13) 288 std r10,PACA_EXGEN+EX_R13(r13) 289 std r11,PACA_EXGEN+EX_R9(r13) 290 std r12,PACA_EXGEN+EX_R3(r13) 291 clrrdi r12,r13,32 292 mfmsr r10 293 mfspr r11,SRR0 /* save SRR0 */ 294 ori r12,r12,slb_miss_user_common@l /* virt addr of handler */ 295 ori r10,r10,MSR_IR|MSR_DR|MSR_RI 296 mtspr SRR0,r12 297 mfspr r12,SRR1 /* and SRR1 */ 298 mtspr SRR1,r10 299 rfid 300 b . /* prevent spec. execution */ 301#endif /* __DISABLED__ */ 302 303/* KVM's trampoline code needs to be close to the interrupt handlers */ 304 305#ifdef CONFIG_KVM_BOOK3S_64_HANDLER 306#include "../kvm/book3s_rmhandlers.S" 307#endif 308 309 .align 7 310 .globl __end_interrupts 311__end_interrupts: 312 313/* 314 * Code from here down to __end_handlers is invoked from the 315 * exception prologs above. Because the prologs assemble the 316 * addresses of these handlers using the LOAD_HANDLER macro, 317 * which uses an addi instruction, these handlers must be in 318 * the first 32k of the kernel image. 319 */ 320 321/*** Common interrupt handlers ***/ 322 323 STD_EXCEPTION_COMMON(0x100, system_reset, .system_reset_exception) 324 325 /* 326 * Machine check is different because we use a different 327 * save area: PACA_EXMC instead of PACA_EXGEN. 328 */ 329 .align 7 330 .globl machine_check_common 331machine_check_common: 332 EXCEPTION_PROLOG_COMMON(0x200, PACA_EXMC) 333 FINISH_NAP 334 DISABLE_INTS 335 bl .save_nvgprs 336 addi r3,r1,STACK_FRAME_OVERHEAD 337 bl .machine_check_exception 338 b .ret_from_except 339 340 STD_EXCEPTION_COMMON_LITE(0x900, decrementer, .timer_interrupt) 341 STD_EXCEPTION_COMMON(0xa00, trap_0a, .unknown_exception) 342 STD_EXCEPTION_COMMON(0xb00, trap_0b, .unknown_exception) 343 STD_EXCEPTION_COMMON(0xd00, single_step, .single_step_exception) 344 STD_EXCEPTION_COMMON(0xe00, trap_0e, .unknown_exception) 345 STD_EXCEPTION_COMMON_IDLE(0xf00, performance_monitor, .performance_monitor_exception) 346 STD_EXCEPTION_COMMON(0x1300, instruction_breakpoint, .instruction_breakpoint_exception) 347#ifdef CONFIG_ALTIVEC 348 STD_EXCEPTION_COMMON(0x1700, altivec_assist, .altivec_assist_exception) 349#else 350 STD_EXCEPTION_COMMON(0x1700, altivec_assist, .unknown_exception) 351#endif 352#ifdef CONFIG_CBE_RAS 353 STD_EXCEPTION_COMMON(0x1200, cbe_system_error, .cbe_system_error_exception) 354 STD_EXCEPTION_COMMON(0x1600, cbe_maintenance, .cbe_maintenance_exception) 355 STD_EXCEPTION_COMMON(0x1800, cbe_thermal, .cbe_thermal_exception) 356#endif /* CONFIG_CBE_RAS */ 357 358 .align 7 359system_call_entry: 360 b system_call_common 361 362/* 363 * Here we have detected that the kernel stack pointer is bad. 364 * R9 contains the saved CR, r13 points to the paca, 365 * r10 contains the (bad) kernel stack pointer, 366 * r11 and r12 contain the saved SRR0 and SRR1. 367 * We switch to using an emergency stack, save the registers there, 368 * and call kernel_bad_stack(), which panics. 369 */ 370bad_stack: 371 ld r1,PACAEMERGSP(r13) 372 subi r1,r1,64+INT_FRAME_SIZE 373 std r9,_CCR(r1) 374 std r10,GPR1(r1) 375 std r11,_NIP(r1) 376 std r12,_MSR(r1) 377 mfspr r11,SPRN_DAR 378 mfspr r12,SPRN_DSISR 379 std r11,_DAR(r1) 380 std r12,_DSISR(r1) 381 mflr r10 382 mfctr r11 383 mfxer r12 384 std r10,_LINK(r1) 385 std r11,_CTR(r1) 386 std r12,_XER(r1) 387 SAVE_GPR(0,r1) 388 SAVE_GPR(2,r1) 389 SAVE_4GPRS(3,r1) 390 SAVE_2GPRS(7,r1) 391 SAVE_10GPRS(12,r1) 392 SAVE_10GPRS(22,r1) 393 lhz r12,PACA_TRAP_SAVE(r13) 394 std r12,_TRAP(r1) 395 addi r11,r1,INT_FRAME_SIZE 396 std r11,0(r1) 397 li r12,0 398 std r12,0(r11) 399 ld r2,PACATOC(r13) 4001: addi r3,r1,STACK_FRAME_OVERHEAD 401 bl .kernel_bad_stack 402 b 1b 403 404/* 405 * Here r13 points to the paca, r9 contains the saved CR, 406 * SRR0 and SRR1 are saved in r11 and r12, 407 * r9 - r13 are saved in paca->exgen. 408 */ 409 .align 7 410 .globl data_access_common 411data_access_common: 412 mfspr r10,SPRN_DAR 413 std r10,PACA_EXGEN+EX_DAR(r13) 414 mfspr r10,SPRN_DSISR 415 stw r10,PACA_EXGEN+EX_DSISR(r13) 416 EXCEPTION_PROLOG_COMMON(0x300, PACA_EXGEN) 417 ld r3,PACA_EXGEN+EX_DAR(r13) 418 lwz r4,PACA_EXGEN+EX_DSISR(r13) 419 li r5,0x300 420 b .do_hash_page /* Try to handle as hpte fault */ 421 422 .align 7 423 .globl instruction_access_common 424instruction_access_common: 425 EXCEPTION_PROLOG_COMMON(0x400, PACA_EXGEN) 426 ld r3,_NIP(r1) 427 andis. r4,r12,0x5820 428 li r5,0x400 429 b .do_hash_page /* Try to handle as hpte fault */ 430 431/* 432 * Here is the common SLB miss user that is used when going to virtual 433 * mode for SLB misses, that is currently not used 434 */ 435#ifdef __DISABLED__ 436 .align 7 437 .globl slb_miss_user_common 438slb_miss_user_common: 439 mflr r10 440 std r3,PACA_EXGEN+EX_DAR(r13) 441 stw r9,PACA_EXGEN+EX_CCR(r13) 442 std r10,PACA_EXGEN+EX_LR(r13) 443 std r11,PACA_EXGEN+EX_SRR0(r13) 444 bl .slb_allocate_user 445 446 ld r10,PACA_EXGEN+EX_LR(r13) 447 ld r3,PACA_EXGEN+EX_R3(r13) 448 lwz r9,PACA_EXGEN+EX_CCR(r13) 449 ld r11,PACA_EXGEN+EX_SRR0(r13) 450 mtlr r10 451 beq- slb_miss_fault 452 453 andi. r10,r12,MSR_RI /* check for unrecoverable exception */ 454 beq- unrecov_user_slb 455 mfmsr r10 456 457.machine push 458.machine "power4" 459 mtcrf 0x80,r9 460.machine pop 461 462 clrrdi r10,r10,2 /* clear RI before setting SRR0/1 */ 463 mtmsrd r10,1 464 465 mtspr SRR0,r11 466 mtspr SRR1,r12 467 468 ld r9,PACA_EXGEN+EX_R9(r13) 469 ld r10,PACA_EXGEN+EX_R10(r13) 470 ld r11,PACA_EXGEN+EX_R11(r13) 471 ld r12,PACA_EXGEN+EX_R12(r13) 472 ld r13,PACA_EXGEN+EX_R13(r13) 473 rfid 474 b . 475 476slb_miss_fault: 477 EXCEPTION_PROLOG_COMMON(0x380, PACA_EXGEN) 478 ld r4,PACA_EXGEN+EX_DAR(r13) 479 li r5,0 480 std r4,_DAR(r1) 481 std r5,_DSISR(r1) 482 b handle_page_fault 483 484unrecov_user_slb: 485 EXCEPTION_PROLOG_COMMON(0x4200, PACA_EXGEN) 486 DISABLE_INTS 487 bl .save_nvgprs 4881: addi r3,r1,STACK_FRAME_OVERHEAD 489 bl .unrecoverable_exception 490 b 1b 491 492#endif /* __DISABLED__ */ 493 494 495/* 496 * r13 points to the PACA, r9 contains the saved CR, 497 * r12 contain the saved SRR1, SRR0 is still ready for return 498 * r3 has the faulting address 499 * r9 - r13 are saved in paca->exslb. 500 * r3 is saved in paca->slb_r3 501 * We assume we aren't going to take any exceptions during this procedure. 502 */ 503_GLOBAL(slb_miss_realmode) 504 mflr r10 505#ifdef CONFIG_RELOCATABLE 506 mtctr r11 507#endif 508 509 stw r9,PACA_EXSLB+EX_CCR(r13) /* save CR in exc. frame */ 510 std r10,PACA_EXSLB+EX_LR(r13) /* save LR */ 511 512 bl .slb_allocate_realmode 513 514 /* All done -- return from exception. */ 515 516 ld r10,PACA_EXSLB+EX_LR(r13) 517 ld r3,PACA_EXSLB+EX_R3(r13) 518 lwz r9,PACA_EXSLB+EX_CCR(r13) /* get saved CR */ 519#ifdef CONFIG_PPC_ISERIES 520BEGIN_FW_FTR_SECTION 521 ld r11,PACALPPACAPTR(r13) 522 ld r11,LPPACASRR0(r11) /* get SRR0 value */ 523END_FW_FTR_SECTION_IFSET(FW_FEATURE_ISERIES) 524#endif /* CONFIG_PPC_ISERIES */ 525 526 mtlr r10 527 528 andi. r10,r12,MSR_RI /* check for unrecoverable exception */ 529 beq- 2f 530 531.machine push 532.machine "power4" 533 mtcrf 0x80,r9 534 mtcrf 0x01,r9 /* slb_allocate uses cr0 and cr7 */ 535.machine pop 536 537#ifdef CONFIG_PPC_ISERIES 538BEGIN_FW_FTR_SECTION 539 mtspr SPRN_SRR0,r11 540 mtspr SPRN_SRR1,r12 541END_FW_FTR_SECTION_IFSET(FW_FEATURE_ISERIES) 542#endif /* CONFIG_PPC_ISERIES */ 543 ld r9,PACA_EXSLB+EX_R9(r13) 544 ld r10,PACA_EXSLB+EX_R10(r13) 545 ld r11,PACA_EXSLB+EX_R11(r13) 546 ld r12,PACA_EXSLB+EX_R12(r13) 547 ld r13,PACA_EXSLB+EX_R13(r13) 548 rfid 549 b . /* prevent speculative execution */ 550 5512: 552#ifdef CONFIG_PPC_ISERIES 553BEGIN_FW_FTR_SECTION 554 b unrecov_slb 555END_FW_FTR_SECTION_IFSET(FW_FEATURE_ISERIES) 556#endif /* CONFIG_PPC_ISERIES */ 557 mfspr r11,SPRN_SRR0 558 ld r10,PACAKBASE(r13) 559 LOAD_HANDLER(r10,unrecov_slb) 560 mtspr SPRN_SRR0,r10 561 ld r10,PACAKMSR(r13) 562 mtspr SPRN_SRR1,r10 563 rfid 564 b . 565 566unrecov_slb: 567 EXCEPTION_PROLOG_COMMON(0x4100, PACA_EXSLB) 568 DISABLE_INTS 569 bl .save_nvgprs 5701: addi r3,r1,STACK_FRAME_OVERHEAD 571 bl .unrecoverable_exception 572 b 1b 573 574 .align 7 575 .globl hardware_interrupt_common 576 .globl hardware_interrupt_entry 577hardware_interrupt_common: 578 EXCEPTION_PROLOG_COMMON(0x500, PACA_EXGEN) 579 FINISH_NAP 580hardware_interrupt_entry: 581 DISABLE_INTS 582BEGIN_FTR_SECTION 583 bl .ppc64_runlatch_on 584END_FTR_SECTION_IFSET(CPU_FTR_CTRL) 585 addi r3,r1,STACK_FRAME_OVERHEAD 586 bl .do_IRQ 587 b .ret_from_except_lite 588 589#ifdef CONFIG_PPC_970_NAP 590power4_fixup_nap: 591 andc r9,r9,r10 592 std r9,TI_LOCAL_FLAGS(r11) 593 ld r10,_LINK(r1) /* make idle task do the */ 594 std r10,_NIP(r1) /* equivalent of a blr */ 595 blr 596#endif 597 598 .align 7 599 .globl alignment_common 600alignment_common: 601 mfspr r10,SPRN_DAR 602 std r10,PACA_EXGEN+EX_DAR(r13) 603 mfspr r10,SPRN_DSISR 604 stw r10,PACA_EXGEN+EX_DSISR(r13) 605 EXCEPTION_PROLOG_COMMON(0x600, PACA_EXGEN) 606 ld r3,PACA_EXGEN+EX_DAR(r13) 607 lwz r4,PACA_EXGEN+EX_DSISR(r13) 608 std r3,_DAR(r1) 609 std r4,_DSISR(r1) 610 bl .save_nvgprs 611 addi r3,r1,STACK_FRAME_OVERHEAD 612 ENABLE_INTS 613 bl .alignment_exception 614 b .ret_from_except 615 616 .align 7 617 .globl program_check_common 618program_check_common: 619 EXCEPTION_PROLOG_COMMON(0x700, PACA_EXGEN) 620 bl .save_nvgprs 621 addi r3,r1,STACK_FRAME_OVERHEAD 622 ENABLE_INTS 623 bl .program_check_exception 624 b .ret_from_except 625 626 .align 7 627 .globl fp_unavailable_common 628fp_unavailable_common: 629 EXCEPTION_PROLOG_COMMON(0x800, PACA_EXGEN) 630 bne 1f /* if from user, just load it up */ 631 bl .save_nvgprs 632 addi r3,r1,STACK_FRAME_OVERHEAD 633 ENABLE_INTS 634 bl .kernel_fp_unavailable_exception 635 BUG_OPCODE 6361: bl .load_up_fpu 637 b fast_exception_return 638 639 .align 7 640 .globl altivec_unavailable_common 641altivec_unavailable_common: 642 EXCEPTION_PROLOG_COMMON(0xf20, PACA_EXGEN) 643#ifdef CONFIG_ALTIVEC 644BEGIN_FTR_SECTION 645 beq 1f 646 bl .load_up_altivec 647 b fast_exception_return 6481: 649END_FTR_SECTION_IFSET(CPU_FTR_ALTIVEC) 650#endif 651 bl .save_nvgprs 652 addi r3,r1,STACK_FRAME_OVERHEAD 653 ENABLE_INTS 654 bl .altivec_unavailable_exception 655 b .ret_from_except 656 657 .align 7 658 .globl vsx_unavailable_common 659vsx_unavailable_common: 660 EXCEPTION_PROLOG_COMMON(0xf40, PACA_EXGEN) 661#ifdef CONFIG_VSX 662BEGIN_FTR_SECTION 663 bne .load_up_vsx 6641: 665END_FTR_SECTION_IFSET(CPU_FTR_VSX) 666#endif 667 bl .save_nvgprs 668 addi r3,r1,STACK_FRAME_OVERHEAD 669 ENABLE_INTS 670 bl .vsx_unavailable_exception 671 b .ret_from_except 672 673 .align 7 674 .globl __end_handlers 675__end_handlers: 676 677/* 678 * Return from an exception with minimal checks. 679 * The caller is assumed to have done EXCEPTION_PROLOG_COMMON. 680 * If interrupts have been enabled, or anything has been 681 * done that might have changed the scheduling status of 682 * any task or sent any task a signal, you should use 683 * ret_from_except or ret_from_except_lite instead of this. 684 */ 685fast_exc_return_irq: /* restores irq state too */ 686 ld r3,SOFTE(r1) 687 TRACE_AND_RESTORE_IRQ(r3); 688 ld r12,_MSR(r1) 689 rldicl r4,r12,49,63 /* get MSR_EE to LSB */ 690 stb r4,PACAHARDIRQEN(r13) /* restore paca->hard_enabled */ 691 b 1f 692 693 .globl fast_exception_return 694fast_exception_return: 695 ld r12,_MSR(r1) 6961: ld r11,_NIP(r1) 697 andi. r3,r12,MSR_RI /* check if RI is set */ 698 beq- unrecov_fer 699 700#ifdef CONFIG_VIRT_CPU_ACCOUNTING 701 andi. r3,r12,MSR_PR 702 beq 2f 703 ACCOUNT_CPU_USER_EXIT(r3, r4) 7042: 705#endif 706 707 ld r3,_CCR(r1) 708 ld r4,_LINK(r1) 709 ld r5,_CTR(r1) 710 ld r6,_XER(r1) 711 mtcr r3 712 mtlr r4 713 mtctr r5 714 mtxer r6 715 REST_GPR(0, r1) 716 REST_8GPRS(2, r1) 717 718 mfmsr r10 719 rldicl r10,r10,48,1 /* clear EE */ 720 rldicr r10,r10,16,61 /* clear RI (LE is 0 already) */ 721 mtmsrd r10,1 722 723 mtspr SPRN_SRR1,r12 724 mtspr SPRN_SRR0,r11 725 REST_4GPRS(10, r1) 726 ld r1,GPR1(r1) 727 rfid 728 b . /* prevent speculative execution */ 729 730unrecov_fer: 731 bl .save_nvgprs 7321: addi r3,r1,STACK_FRAME_OVERHEAD 733 bl .unrecoverable_exception 734 b 1b 735 736 737/* 738 * Hash table stuff 739 */ 740 .align 7 741_STATIC(do_hash_page) 742 std r3,_DAR(r1) 743 std r4,_DSISR(r1) 744 745 andis. r0,r4,0xa410 /* weird error? */ 746 bne- handle_page_fault /* if not, try to insert a HPTE */ 747 andis. r0,r4,DSISR_DABRMATCH@h 748 bne- handle_dabr_fault 749 750BEGIN_FTR_SECTION 751 andis. r0,r4,0x0020 /* Is it a segment table fault? */ 752 bne- do_ste_alloc /* If so handle it */ 753END_FTR_SECTION_IFCLR(CPU_FTR_SLB) 754 755 clrrdi r11,r1,THREAD_SHIFT 756 lwz r0,TI_PREEMPT(r11) /* If we're in an "NMI" */ 757 andis. r0,r0,NMI_MASK@h /* (i.e. an irq when soft-disabled) */ 758 bne 77f /* then don't call hash_page now */ 759 760 /* 761 * On iSeries, we soft-disable interrupts here, then 762 * hard-enable interrupts so that the hash_page code can spin on 763 * the hash_table_lock without problems on a shared processor. 764 */ 765 DISABLE_INTS 766 767 /* 768 * Currently, trace_hardirqs_off() will be called by DISABLE_INTS 769 * and will clobber volatile registers when irq tracing is enabled 770 * so we need to reload them. It may be possible to be smarter here 771 * and move the irq tracing elsewhere but let's keep it simple for 772 * now 773 */ 774#ifdef CONFIG_TRACE_IRQFLAGS 775 ld r3,_DAR(r1) 776 ld r4,_DSISR(r1) 777 ld r5,_TRAP(r1) 778 ld r12,_MSR(r1) 779 clrrdi r5,r5,4 780#endif /* CONFIG_TRACE_IRQFLAGS */ 781 /* 782 * We need to set the _PAGE_USER bit if MSR_PR is set or if we are 783 * accessing a userspace segment (even from the kernel). We assume 784 * kernel addresses always have the high bit set. 785 */ 786 rlwinm r4,r4,32-25+9,31-9,31-9 /* DSISR_STORE -> _PAGE_RW */ 787 rotldi r0,r3,15 /* Move high bit into MSR_PR posn */ 788 orc r0,r12,r0 /* MSR_PR | ~high_bit */ 789 rlwimi r4,r0,32-13,30,30 /* becomes _PAGE_USER access bit */ 790 ori r4,r4,1 /* add _PAGE_PRESENT */ 791 rlwimi r4,r5,22+2,31-2,31-2 /* Set _PAGE_EXEC if trap is 0x400 */ 792 793 /* 794 * r3 contains the faulting address 795 * r4 contains the required access permissions 796 * r5 contains the trap number 797 * 798 * at return r3 = 0 for success 799 */ 800 bl .hash_page /* build HPTE if possible */ 801 cmpdi r3,0 /* see if hash_page succeeded */ 802 803BEGIN_FW_FTR_SECTION 804 /* 805 * If we had interrupts soft-enabled at the point where the 806 * DSI/ISI occurred, and an interrupt came in during hash_page, 807 * handle it now. 808 * We jump to ret_from_except_lite rather than fast_exception_return 809 * because ret_from_except_lite will check for and handle pending 810 * interrupts if necessary. 811 */ 812 beq 13f 813END_FW_FTR_SECTION_IFSET(FW_FEATURE_ISERIES) 814 815BEGIN_FW_FTR_SECTION 816 /* 817 * Here we have interrupts hard-disabled, so it is sufficient 818 * to restore paca->{soft,hard}_enable and get out. 819 */ 820 beq fast_exc_return_irq /* Return from exception on success */ 821END_FW_FTR_SECTION_IFCLR(FW_FEATURE_ISERIES) 822 823 /* For a hash failure, we don't bother re-enabling interrupts */ 824 ble- 12f 825 826 /* 827 * hash_page couldn't handle it, set soft interrupt enable back 828 * to what it was before the trap. Note that .arch_local_irq_restore 829 * handles any interrupts pending at this point. 830 */ 831 ld r3,SOFTE(r1) 832 TRACE_AND_RESTORE_IRQ_PARTIAL(r3, 11f) 833 bl .arch_local_irq_restore 834 b 11f 835 836/* We have a data breakpoint exception - handle it */ 837handle_dabr_fault: 838 bl .save_nvgprs 839 ld r4,_DAR(r1) 840 ld r5,_DSISR(r1) 841 addi r3,r1,STACK_FRAME_OVERHEAD 842 bl .do_dabr 843 b .ret_from_except_lite 844 845/* Here we have a page fault that hash_page can't handle. */ 846handle_page_fault: 847 ENABLE_INTS 84811: ld r4,_DAR(r1) 849 ld r5,_DSISR(r1) 850 addi r3,r1,STACK_FRAME_OVERHEAD 851 bl .do_page_fault 852 cmpdi r3,0 853 beq+ 13f 854 bl .save_nvgprs 855 mr r5,r3 856 addi r3,r1,STACK_FRAME_OVERHEAD 857 lwz r4,_DAR(r1) 858 bl .bad_page_fault 859 b .ret_from_except 860 86113: b .ret_from_except_lite 862 863/* We have a page fault that hash_page could handle but HV refused 864 * the PTE insertion 865 */ 86612: bl .save_nvgprs 867 mr r5,r3 868 addi r3,r1,STACK_FRAME_OVERHEAD 869 ld r4,_DAR(r1) 870 bl .low_hash_fault 871 b .ret_from_except 872 873/* 874 * We come here as a result of a DSI at a point where we don't want 875 * to call hash_page, such as when we are accessing memory (possibly 876 * user memory) inside a PMU interrupt that occurred while interrupts 877 * were soft-disabled. We want to invoke the exception handler for 878 * the access, or panic if there isn't a handler. 879 */ 88077: bl .save_nvgprs 881 mr r4,r3 882 addi r3,r1,STACK_FRAME_OVERHEAD 883 li r5,SIGSEGV 884 bl .bad_page_fault 885 b .ret_from_except 886 887 /* here we have a segment miss */ 888do_ste_alloc: 889 bl .ste_allocate /* try to insert stab entry */ 890 cmpdi r3,0 891 bne- handle_page_fault 892 b fast_exception_return 893 894/* 895 * r13 points to the PACA, r9 contains the saved CR, 896 * r11 and r12 contain the saved SRR0 and SRR1. 897 * r9 - r13 are saved in paca->exslb. 898 * We assume we aren't going to take any exceptions during this procedure. 899 * We assume (DAR >> 60) == 0xc. 900 */ 901 .align 7 902_GLOBAL(do_stab_bolted) 903 stw r9,PACA_EXSLB+EX_CCR(r13) /* save CR in exc. frame */ 904 std r11,PACA_EXSLB+EX_SRR0(r13) /* save SRR0 in exc. frame */ 905 906 /* Hash to the primary group */ 907 ld r10,PACASTABVIRT(r13) 908 mfspr r11,SPRN_DAR 909 srdi r11,r11,28 910 rldimi r10,r11,7,52 /* r10 = first ste of the group */ 911 912 /* Calculate VSID */ 913 /* This is a kernel address, so protovsid = ESID */ 914 ASM_VSID_SCRAMBLE(r11, r9, 256M) 915 rldic r9,r11,12,16 /* r9 = vsid << 12 */ 916 917 /* Search the primary group for a free entry */ 9181: ld r11,0(r10) /* Test valid bit of the current ste */ 919 andi. r11,r11,0x80 920 beq 2f 921 addi r10,r10,16 922 andi. r11,r10,0x70 923 bne 1b 924 925 /* Stick for only searching the primary group for now. */ 926 /* At least for now, we use a very simple random castout scheme */ 927 /* Use the TB as a random number ; OR in 1 to avoid entry 0 */ 928 mftb r11 929 rldic r11,r11,4,57 /* r11 = (r11 << 4) & 0x70 */ 930 ori r11,r11,0x10 931 932 /* r10 currently points to an ste one past the group of interest */ 933 /* make it point to the randomly selected entry */ 934 subi r10,r10,128 935 or r10,r10,r11 /* r10 is the entry to invalidate */ 936 937 isync /* mark the entry invalid */ 938 ld r11,0(r10) 939 rldicl r11,r11,56,1 /* clear the valid bit */ 940 rotldi r11,r11,8 941 std r11,0(r10) 942 sync 943 944 clrrdi r11,r11,28 /* Get the esid part of the ste */ 945 slbie r11 946 9472: std r9,8(r10) /* Store the vsid part of the ste */ 948 eieio 949 950 mfspr r11,SPRN_DAR /* Get the new esid */ 951 clrrdi r11,r11,28 /* Permits a full 32b of ESID */ 952 ori r11,r11,0x90 /* Turn on valid and kp */ 953 std r11,0(r10) /* Put new entry back into the stab */ 954 955 sync 956 957 /* All done -- return from exception. */ 958 lwz r9,PACA_EXSLB+EX_CCR(r13) /* get saved CR */ 959 ld r11,PACA_EXSLB+EX_SRR0(r13) /* get saved SRR0 */ 960 961 andi. r10,r12,MSR_RI 962 beq- unrecov_slb 963 964 mtcrf 0x80,r9 /* restore CR */ 965 966 mfmsr r10 967 clrrdi r10,r10,2 968 mtmsrd r10,1 969 970 mtspr SPRN_SRR0,r11 971 mtspr SPRN_SRR1,r12 972 ld r9,PACA_EXSLB+EX_R9(r13) 973 ld r10,PACA_EXSLB+EX_R10(r13) 974 ld r11,PACA_EXSLB+EX_R11(r13) 975 ld r12,PACA_EXSLB+EX_R12(r13) 976 ld r13,PACA_EXSLB+EX_R13(r13) 977 rfid 978 b . /* prevent speculative execution */ 979 980/* 981 * Space for CPU0's segment table. 982 * 983 * On iSeries, the hypervisor must fill in at least one entry before 984 * we get control (with relocate on). The address is given to the hv 985 * as a page number (see xLparMap below), so this must be at a 986 * fixed address (the linker can't compute (u64)&initial_stab >> 987 * PAGE_SHIFT). 988 */ 989 . = STAB0_OFFSET /* 0x6000 */ 990 .globl initial_stab 991initial_stab: 992 .space 4096 993 994#ifdef CONFIG_PPC_PSERIES 995/* 996 * Data area reserved for FWNMI option. 997 * This address (0x7000) is fixed by the RPA. 998 */ 999 .= 0x7000 1000 .globl fwnmi_data_area 1001fwnmi_data_area: 1002#endif /* CONFIG_PPC_PSERIES */ 1003 1004 /* iSeries does not use the FWNMI stuff, so it is safe to put 1005 * this here, even if we later allow kernels that will boot on 1006 * both pSeries and iSeries */ 1007#ifdef CONFIG_PPC_ISERIES 1008 . = LPARMAP_PHYS 1009 .globl xLparMap 1010xLparMap: 1011 .quad HvEsidsToMap /* xNumberEsids */ 1012 .quad HvRangesToMap /* xNumberRanges */ 1013 .quad STAB0_PAGE /* xSegmentTableOffs */ 1014 .zero 40 /* xRsvd */ 1015 /* xEsids (HvEsidsToMap entries of 2 quads) */ 1016 .quad PAGE_OFFSET_ESID /* xKernelEsid */ 1017 .quad PAGE_OFFSET_VSID /* xKernelVsid */ 1018 .quad VMALLOC_START_ESID /* xKernelEsid */ 1019 .quad VMALLOC_START_VSID /* xKernelVsid */ 1020 /* xRanges (HvRangesToMap entries of 3 quads) */ 1021 .quad HvPagesToMap /* xPages */ 1022 .quad 0 /* xOffset */ 1023 .quad PAGE_OFFSET_VSID << (SID_SHIFT - HW_PAGE_SHIFT) /* xVPN */ 1024 1025#endif /* CONFIG_PPC_ISERIES */ 1026 1027#ifdef CONFIG_PPC_PSERIES 1028 . = 0x8000 1029#endif /* CONFIG_PPC_PSERIES */ 1030